mirror of
https://github.com/emsesp/EMS-ESP32.git
synced 2026-05-09 07:25:49 +00:00
Compare commits
66 Commits
60b7d6d795
...
v3.7.2
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
eaa277fef0 | ||
|
|
e418b7d8e7 | ||
|
|
a4e3be6a69 | ||
|
|
a1bc5bb055 | ||
|
|
f444ca31e0 | ||
|
|
1b70b55989 | ||
|
|
1487f30c43 | ||
|
|
e00eb8e64f | ||
|
|
f41bb3671c | ||
|
|
22c75e6df3 | ||
|
|
5ab10b7aa6 | ||
|
|
ee5fd4d0eb | ||
|
|
46f35bc67c | ||
|
|
ec85a7ec24 | ||
|
|
02f2389587 | ||
|
|
7f140021aa | ||
|
|
6796962c1e | ||
|
|
df6de21cf4 | ||
|
|
df9f75a5c9 | ||
|
|
7bd8710eb6 | ||
|
|
32f2c6d341 | ||
|
|
86919c1684 | ||
|
|
86e29515e7 | ||
|
|
46eb4185d7 | ||
|
|
8da6761a48 | ||
|
|
9233f0dfcc | ||
|
|
292f743b14 | ||
|
|
dd6dfffd57 | ||
|
|
ec705a5307 | ||
|
|
f45f071710 | ||
|
|
f3858546de | ||
|
|
d0ac0b7804 | ||
|
|
d8284ec09f | ||
|
|
6e982acde8 | ||
|
|
8c94ce99b2 | ||
|
|
fc057d18c9 | ||
|
|
18e9b99413 | ||
|
|
a47e0e8266 | ||
|
|
f412ddc716 | ||
|
|
29110e96e5 | ||
|
|
b65866217a | ||
|
|
611e3b1243 | ||
|
|
2ca0a0c634 | ||
|
|
7eb1f061b7 | ||
|
|
50459a23fe | ||
|
|
5bf53c3389 | ||
|
|
4b7aa95be3 | ||
|
|
70943f5758 | ||
|
|
3bc280b817 | ||
|
|
62b15a5319 | ||
|
|
8dd18802d6 | ||
|
|
57a516a83a | ||
|
|
a57fdaa4b3 | ||
|
|
4841e42286 | ||
|
|
8c2d2b06ed | ||
|
|
38c8b1b7f0 | ||
|
|
6fb5933a02 | ||
|
|
c0944433be | ||
|
|
478e6362c9 | ||
|
|
4d6354db78 | ||
|
|
beab0f0c77 | ||
|
|
c17749bd22 | ||
|
|
2bad769c5c | ||
|
|
8ad89ca64b | ||
|
|
9244d8daec | ||
|
|
02d01334b2 |
@@ -1,27 +0,0 @@
|
|||||||
{
|
|
||||||
"name": "EMS-ESP Devcontainer",
|
|
||||||
"image": "mcr.microsoft.com/devcontainers/typescript-node:1-22-bookworm",
|
|
||||||
"features": {
|
|
||||||
"ghcr.io/devcontainers/features/node:1": {},
|
|
||||||
"ghcr.io/devcontainers-extra/features/pnpm:2": {},
|
|
||||||
"ghcr.io/devcontainers/features/python:1": {},
|
|
||||||
"ghcr.io/shyim/devcontainers-features/bun:0": {}
|
|
||||||
},
|
|
||||||
|
|
||||||
"forwardPorts": [
|
|
||||||
3000,
|
|
||||||
3080
|
|
||||||
],
|
|
||||||
|
|
||||||
// Use 'postCreateCommand' to run commands after the container is created.
|
|
||||||
"postCreateCommand": "cd mock-api && pnpm install && cd .. && cd interface && pnpm install",
|
|
||||||
|
|
||||||
// Configure tool-specific properties.
|
|
||||||
"customizations": {
|
|
||||||
"vscode": {
|
|
||||||
"extensions": [
|
|
||||||
"platformio.platformio-ide"
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
4
.github/ISSUE_TEMPLATE/bug_report.md
vendored
4
.github/ISSUE_TEMPLATE/bug_report.md
vendored
@@ -21,8 +21,8 @@ _Make sure your have performed every step and checked the applicable boxes befor
|
|||||||
|
|
||||||
- [ ] Searched the issue in [issues](https://github.com/emsesp/EMS-ESP32/issues)
|
- [ ] Searched the issue in [issues](https://github.com/emsesp/EMS-ESP32/issues)
|
||||||
- [ ] Searched the issue in [discussions](https://github.com/emsesp/EMS-ESP32/discussions)
|
- [ ] Searched the issue in [discussions](https://github.com/emsesp/EMS-ESP32/discussions)
|
||||||
- [ ] Searched the issue in the [docs](https://emsesp.org/Troubleshooting/)
|
- [ ] Searched the issue in the [docs](https://docs.emsesp.org/Troubleshooting/)
|
||||||
- [ ] Searched the issue in the [chat](https://discord.gg/GP9DPSgeJq)
|
- [ ] Searched the issue in the [chat](https://discord.gg/3J3GgnzpyT)
|
||||||
- [ ] Provide the System information in the area below, taken from `http://<IP>/api/system`
|
- [ ] Provide the System information in the area below, taken from `http://<IP>/api/system`
|
||||||
|
|
||||||
```json
|
```json
|
||||||
|
|||||||
4
.github/ISSUE_TEMPLATE/config.yml
vendored
4
.github/ISSUE_TEMPLATE/config.yml
vendored
@@ -1,11 +1,11 @@
|
|||||||
blank_issues_enabled: false
|
blank_issues_enabled: false
|
||||||
contact_links:
|
contact_links:
|
||||||
- name: EMS-ESP Docs
|
- name: EMS-ESP Docs
|
||||||
url: https://emsesp.org
|
url: https://docs.emsesp.org
|
||||||
about: All the information related to EMS-ESP.
|
about: All the information related to EMS-ESP.
|
||||||
- name: EMS-ESP Discussions and Support
|
- name: EMS-ESP Discussions and Support
|
||||||
url: https://github.com/emsesp/EMS-ESP32/discussions
|
url: https://github.com/emsesp/EMS-ESP32/discussions
|
||||||
about: EMS-ESP usage Questions, Feature Requests and Projects.
|
about: EMS-ESP usage Questions, Feature Requests and Projects.
|
||||||
- name: EMS-ESP Users Chat
|
- name: EMS-ESP Users Chat
|
||||||
url: https://discord.gg/GP9DPSgeJq
|
url: https://discord.gg/3J3GgnzpyT
|
||||||
about: Chat for feedback, questions and troubleshooting.
|
about: Chat for feedback, questions and troubleshooting.
|
||||||
|
|||||||
79
.github/workflows/dev_release.yml
vendored
79
.github/workflows/dev_release.yml
vendored
@@ -1,79 +0,0 @@
|
|||||||
name: 'Build dev release'
|
|
||||||
|
|
||||||
on:
|
|
||||||
workflow_dispatch:
|
|
||||||
push:
|
|
||||||
paths:
|
|
||||||
- 'src/emsesp_version.h'
|
|
||||||
branches:
|
|
||||||
- 'dev'
|
|
||||||
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
pre-release:
|
|
||||||
name: 'Build Dev Release'
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
|
|
||||||
- name: Install python 3.13
|
|
||||||
uses: actions/setup-python@v6
|
|
||||||
with:
|
|
||||||
python-version: '3.13'
|
|
||||||
|
|
||||||
- name: Install Node.js 24
|
|
||||||
uses: actions/setup-node@v6
|
|
||||||
with:
|
|
||||||
node-version: 24
|
|
||||||
|
|
||||||
- name: Checkout repository
|
|
||||||
uses: actions/checkout@v5
|
|
||||||
|
|
||||||
- name: Enable Corepack
|
|
||||||
run: corepack enable pnpm
|
|
||||||
|
|
||||||
- name: Get the EMS-ESP version
|
|
||||||
id: build_info
|
|
||||||
run: |
|
|
||||||
version=`grep -E '^#define EMSESP_APP_VERSION' ./src/emsesp_version.h | awk -F'"' '{print $2}'`
|
|
||||||
echo "VERSION=$version" >> $GITHUB_OUTPUT
|
|
||||||
|
|
||||||
- name: Install PlatformIO
|
|
||||||
run: |
|
|
||||||
python -m pip install --upgrade pip
|
|
||||||
pip install -U platformio
|
|
||||||
python -m pip install intelhex
|
|
||||||
|
|
||||||
- name: Build webUI
|
|
||||||
run: |
|
|
||||||
platformio run -e build_webUI
|
|
||||||
|
|
||||||
- name: Build modbus
|
|
||||||
run: |
|
|
||||||
platformio run -e build_modbus
|
|
||||||
|
|
||||||
- name: Build standalone
|
|
||||||
run: |
|
|
||||||
platformio run -e build_standalone
|
|
||||||
|
|
||||||
- name: Build all PIO target environments, from default_envs
|
|
||||||
run: |
|
|
||||||
platformio run
|
|
||||||
|
|
||||||
- name: Commit the generated files
|
|
||||||
uses: stefanzweifel/git-auto-commit-action@v5
|
|
||||||
with:
|
|
||||||
commit_message: "chore: update generated files for v${{steps.build_info.outputs.VERSION}}"
|
|
||||||
|
|
||||||
- name: Create GitHub Release
|
|
||||||
id: 'automatic_releases'
|
|
||||||
uses: emsesp/action-automatic-releases@v1.0.0
|
|
||||||
with:
|
|
||||||
repo_token: '${{ secrets.GITHUB_TOKEN }}'
|
|
||||||
title: Development Build v${{steps.build_info.outputs.VERSION}}
|
|
||||||
automatic_release_tag: 'latest'
|
|
||||||
prerelease: true
|
|
||||||
files: |
|
|
||||||
CHANGELOG_LATEST.md
|
|
||||||
./build/firmware/*.*
|
|
||||||
@@ -1,17 +1,16 @@
|
|||||||
name: 'Publish releases to discord'
|
name: 'github-releases-to-discord'
|
||||||
|
|
||||||
on:
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
release:
|
release:
|
||||||
types: [published]
|
types: [published]
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
github-releases-to-discord:
|
github-releases-to-discord:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
uses: actions/checkout@v5
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: GitHub Releases To Discord
|
- name: GitHub Releases To Discord
|
||||||
uses: SethCohen/github-releases-to-discord@v1.13.1
|
uses: SethCohen/github-releases-to-discord@v1.13.1
|
||||||
|
|||||||
27
.github/workflows/pr_check.yml
vendored
27
.github/workflows/pr_check.yml
vendored
@@ -1,32 +1,37 @@
|
|||||||
name: 'Pre-check on PR'
|
name: 'pr_check'
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
|
|
||||||
on:
|
on:
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
pull_request:
|
pull_request:
|
||||||
branches: dev
|
branches: dev
|
||||||
paths:
|
paths:
|
||||||
- 'src/**'
|
- '**.c'
|
||||||
|
- '**.cpp'
|
||||||
|
- '**.h'
|
||||||
|
- '**.hpp'
|
||||||
|
- '**.json'
|
||||||
|
- '**.py'
|
||||||
|
- '**.md'
|
||||||
|
- '.github/workflows/pr_check.yml'
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
pre-release:
|
pre-release:
|
||||||
name: 'Automatic pre-release build'
|
name: 'Automatic pre-release build'
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v5
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
- name: Install python 3.13
|
- name: Install python 3.11
|
||||||
uses: actions/setup-python@v6
|
uses: actions/setup-python@v5
|
||||||
with:
|
with:
|
||||||
python-version: '3.13'
|
python-version: '3.11'
|
||||||
|
|
||||||
- name: Install PlatformIO
|
- name: Install PlatformIO
|
||||||
run: |
|
run: |
|
||||||
pip install wheel
|
pip install wheel
|
||||||
pip install -U platformio
|
pip install -U platformio
|
||||||
|
|
||||||
- name: Run unit tests
|
- name: Build native
|
||||||
run: |
|
run: |
|
||||||
platformio run -e native-test -t exec
|
platformio run -e native
|
||||||
|
|||||||
66
.github/workflows/pre_release.yml
vendored
Normal file
66
.github/workflows/pre_release.yml
vendored
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
name: 'pre-release'
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- 'dev'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
pre-release:
|
||||||
|
name: 'Automatic pre-release build'
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Enable Corepack
|
||||||
|
run: corepack enable
|
||||||
|
|
||||||
|
- name: Install python 3.11
|
||||||
|
uses: actions/setup-python@v5
|
||||||
|
with:
|
||||||
|
python-version: '3.11'
|
||||||
|
|
||||||
|
- name: Install Node.js 20
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: '20.x'
|
||||||
|
|
||||||
|
- name: Get EMS-ESP version
|
||||||
|
id: build_info
|
||||||
|
run: |
|
||||||
|
version=`grep -E '^#define EMSESP_APP_VERSION' ./src/version.h | awk -F'"' '{print $2}'`
|
||||||
|
echo "VERSION=$version" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
|
- name: Install PlatformIO
|
||||||
|
run: |
|
||||||
|
python -m pip install --upgrade pip
|
||||||
|
pip install -U platformio
|
||||||
|
|
||||||
|
- name: Build WebUI
|
||||||
|
run: |
|
||||||
|
cd interface
|
||||||
|
yarn install
|
||||||
|
yarn typesafe-i18n --no-watch
|
||||||
|
sed -i "s/= 'pl'/= 'en'/" ./src/i18n/i18n-util.ts
|
||||||
|
yarn build
|
||||||
|
yarn webUI
|
||||||
|
|
||||||
|
- name: Build all PIO target environments from default_envs
|
||||||
|
run: |
|
||||||
|
platformio run
|
||||||
|
env:
|
||||||
|
NO_BUILD_WEBUI: true
|
||||||
|
|
||||||
|
- name: Create GitHub Release
|
||||||
|
id: 'automatic_releases'
|
||||||
|
uses: emsesp/action-automatic-releases@v1.0.0
|
||||||
|
with:
|
||||||
|
repo_token: '${{ secrets.GITHUB_TOKEN }}'
|
||||||
|
title: Development Build v${{steps.build_info.outputs.VERSION}}
|
||||||
|
automatic_release_tag: 'latest'
|
||||||
|
prerelease: true
|
||||||
|
files: |
|
||||||
|
CHANGELOG_LATEST.md
|
||||||
|
./build/firmware/*.*
|
||||||
27
.github/workflows/sonar_check.yml
vendored
27
.github/workflows/sonar_check.yml
vendored
@@ -1,14 +1,12 @@
|
|||||||
# see https://github.com/marketplace/actions/sonarcloud-scan-for-c-and-c#usage
|
# see https://github.com/marketplace/actions/sonarcloud-scan-for-c-and-c#usage
|
||||||
name: Sonar Check
|
name: Sonar Check
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- dev
|
- dev
|
||||||
paths:
|
# pull_request:
|
||||||
- 'src/**'
|
# types: [opened, synchronize, reopened]
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
@@ -19,15 +17,18 @@ jobs:
|
|||||||
BUILD_WRAPPER_OUT_DIR: bw-output
|
BUILD_WRAPPER_OUT_DIR: bw-output
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v5
|
uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
- name: Install Build Wrapper
|
|
||||||
uses: SonarSource/sonarqube-scan-action/install-build-wrapper@master
|
- name: Install sonar-scanner and build-wrapper
|
||||||
- name: Run Build Wrapper
|
uses: SonarSource/sonarcloud-github-c-cpp@v2
|
||||||
run: build-wrapper-linux-x86-64 --out-dir ${{ env.BUILD_WRAPPER_OUT_DIR }} make all
|
|
||||||
- name: SonarQube Scan
|
- name: Run build-wrapper
|
||||||
uses: SonarSource/sonarqube-scan-action@master
|
run: build-wrapper-linux-x86-64 --out-dir ${{ env.BUILD_WRAPPER_OUT_DIR }} make all
|
||||||
|
|
||||||
|
- name: Run sonar-scanner
|
||||||
env:
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }}
|
SONAR_TOKEN: ${{ secrets.SONAR_TOKEN }}
|
||||||
|
run: sonar-scanner --define sonar.cfamily.compile-commands="${{ env.BUILD_WRAPPER_OUT_DIR }}/compile_commands.json"
|
||||||
|
|||||||
63
.github/workflows/stable_release.yml
vendored
63
.github/workflows/stable_release.yml
vendored
@@ -1,63 +0,0 @@
|
|||||||
name: 'Build stable release'
|
|
||||||
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
|
|
||||||
on:
|
|
||||||
workflow_dispatch:
|
|
||||||
push:
|
|
||||||
tags:
|
|
||||||
- 'v*'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
tagged-release:
|
|
||||||
name: 'Build Stable Release'
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
|
|
||||||
- name: Install python 3.13
|
|
||||||
uses: actions/setup-python@v6
|
|
||||||
with:
|
|
||||||
python-version: '3.13'
|
|
||||||
|
|
||||||
- name: Install Node.js 24
|
|
||||||
uses: actions/setup-node@v6
|
|
||||||
with:
|
|
||||||
node-version: 24
|
|
||||||
|
|
||||||
- name: Checkout repository
|
|
||||||
uses: actions/checkout@v5
|
|
||||||
|
|
||||||
- name: Enable Corepack
|
|
||||||
run: corepack enable pnpm
|
|
||||||
|
|
||||||
- name: Install PlatformIO
|
|
||||||
run: |
|
|
||||||
python -m pip install --upgrade pip
|
|
||||||
pip install -U platformio
|
|
||||||
python -m pip install intelhex
|
|
||||||
|
|
||||||
- name: Build webUI
|
|
||||||
run: |
|
|
||||||
platformio run -e build_webUI
|
|
||||||
|
|
||||||
- name: Build modbus
|
|
||||||
run: |
|
|
||||||
platformio run -e build_modbus
|
|
||||||
|
|
||||||
- name: Build standalone
|
|
||||||
run: |
|
|
||||||
platformio run -e build_standalone
|
|
||||||
|
|
||||||
- name: Build all PIO target environments, from default_envs
|
|
||||||
run: |
|
|
||||||
platformio run
|
|
||||||
|
|
||||||
- name: Create GitHub Release
|
|
||||||
uses: emsesp/action-automatic-releases@v1.0.0
|
|
||||||
with:
|
|
||||||
repo_token: '${{ secrets.GITHUB_TOKEN }}'
|
|
||||||
prerelease: false
|
|
||||||
files: |
|
|
||||||
CHANGELOG.md
|
|
||||||
./build/firmware/*.*
|
|
||||||
27
.github/workflows/stale_issues.yml
vendored
27
.github/workflows/stale_issues.yml
vendored
@@ -1,27 +0,0 @@
|
|||||||
name: "Mark or close stale issues and PRs"
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
issues: write
|
|
||||||
pull-requests: write
|
|
||||||
|
|
||||||
on:
|
|
||||||
schedule:
|
|
||||||
- cron: "30 1 * * *"
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
stale:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- uses: actions/stale@v10
|
|
||||||
with:
|
|
||||||
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
days-before-stale: 30
|
|
||||||
days-before-close: 5
|
|
||||||
stale-issue-message: "This issue is stale because it has been open 30 days with no activity. Remove stale label or comment otherwise this will be closed in 5 days."
|
|
||||||
stale-pr-message: "This PR has been automatically marked as stale because there has been no activity in last 30 days. It will be closed if no further activity occurs. Thank you for your contributions."
|
|
||||||
close-issue-message: "This issue was closed because it has been stalled for 5 days with no activity."
|
|
||||||
close-pr-message: "This PR was automatically closed because of being stale."
|
|
||||||
stale-pr-label: "stale"
|
|
||||||
stale-issue-label: "stale"
|
|
||||||
exempt-issue-labels: "bug,enhancement,pinned,security"
|
|
||||||
exempt-pr-labels: "bug,enhancement,pinned,security"
|
|
||||||
57
.github/workflows/tagged_release.yml
vendored
Normal file
57
.github/workflows/tagged_release.yml
vendored
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
name: 'tagged-release'
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
push:
|
||||||
|
tags:
|
||||||
|
- 'v*'
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
tagged-release:
|
||||||
|
name: 'Tagged Release'
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Enable Corepack
|
||||||
|
run: corepack enable
|
||||||
|
|
||||||
|
- name: Install python 3.11
|
||||||
|
uses: actions/setup-python@v5
|
||||||
|
with:
|
||||||
|
python-version: '3.11'
|
||||||
|
|
||||||
|
- name: Install Node.js 20
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: '20.x'
|
||||||
|
|
||||||
|
- name: Install PlatformIO
|
||||||
|
run: |
|
||||||
|
python -m pip install --upgrade pip
|
||||||
|
pip install -U platformio
|
||||||
|
|
||||||
|
- name: Build WebUI
|
||||||
|
run: |
|
||||||
|
cd interface
|
||||||
|
yarn install
|
||||||
|
yarn typesafe-i18n --no-watch
|
||||||
|
sed -i "s/= 'pl'/= 'en'/" ./src/i18n/i18n-util.ts
|
||||||
|
yarn build
|
||||||
|
yarn webUI
|
||||||
|
|
||||||
|
- name: Build all PIO target environments from default_envs
|
||||||
|
run: |
|
||||||
|
platformio run
|
||||||
|
env:
|
||||||
|
NO_BUILD_WEBUI: true
|
||||||
|
|
||||||
|
- name: Create GitHub Release
|
||||||
|
uses: emsesp/action-automatic-releases@v1.0.0
|
||||||
|
with:
|
||||||
|
repo_token: '${{ secrets.GITHUB_TOKEN }}'
|
||||||
|
prerelease: false
|
||||||
|
files: |
|
||||||
|
CHANGELOG.md
|
||||||
|
./build/firmware/*.*
|
||||||
62
.github/workflows/test_release.yml
vendored
62
.github/workflows/test_release.yml
vendored
@@ -1,65 +1,55 @@
|
|||||||
name: 'Build test release'
|
name: 'test-release'
|
||||||
|
|
||||||
on:
|
on:
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
push:
|
push:
|
||||||
branches:
|
branches:
|
||||||
- 'test'
|
- 'dev2'
|
||||||
|
|
||||||
permissions:
|
|
||||||
contents: read
|
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
pre-release:
|
pre-release:
|
||||||
name: 'Build Test Release'
|
name: 'Automatic test-release build'
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
steps:
|
steps:
|
||||||
|
- uses: actions/checkout@v4
|
||||||
- name: Install python 3.13
|
|
||||||
uses: actions/setup-python@v6
|
|
||||||
with:
|
|
||||||
python-version: '3.13'
|
|
||||||
|
|
||||||
- name: Install Node.js 24
|
|
||||||
uses: actions/setup-node@v6
|
|
||||||
with:
|
|
||||||
node-version: 24
|
|
||||||
|
|
||||||
- name: Checkout repository
|
|
||||||
uses: actions/checkout@v5
|
|
||||||
|
|
||||||
- name: Enable Corepack
|
- name: Enable Corepack
|
||||||
run: corepack enable pnpm
|
run: corepack enable
|
||||||
|
|
||||||
- name: Get the EMS-ESP version
|
- uses: actions/setup-python@v5
|
||||||
|
with:
|
||||||
|
python-version: '3.11'
|
||||||
|
|
||||||
|
- name: Use Node.js 20.x
|
||||||
|
uses: actions/setup-node@v4
|
||||||
|
with:
|
||||||
|
node-version: '20.x'
|
||||||
|
|
||||||
|
- name: Get EMS-ESP source code and version
|
||||||
id: build_info
|
id: build_info
|
||||||
run: |
|
run: |
|
||||||
version=`grep -E '^#define EMSESP_APP_VERSION' ./src/emsesp_version.h | awk -F'"' '{print $2}'`
|
version=`grep -E '^#define EMSESP_APP_VERSION' ./src/version.h | awk -F'"' '{print $2}'`
|
||||||
echo "VERSION=$version" >> $GITHUB_OUTPUT
|
echo "VERSION=$version" >> $GITHUB_OUTPUT
|
||||||
|
|
||||||
- name: Install PlatformIO
|
- name: Install PlatformIO
|
||||||
run: |
|
run: |
|
||||||
python -m pip install --upgrade pip
|
python -m pip install --upgrade pip
|
||||||
pip install -U platformio
|
pip install -U platformio
|
||||||
python -m pip install intelhex
|
|
||||||
|
|
||||||
- name: Build webUI
|
- name: Build WebUI
|
||||||
run: |
|
run: |
|
||||||
platformio run -e build_webUI
|
cd interface
|
||||||
|
yarn install
|
||||||
|
yarn typesafe-i18n --no-watch
|
||||||
|
sed -i "s/= 'pl'/= 'en'/" ./src/i18n/i18n-util.ts
|
||||||
|
yarn build
|
||||||
|
yarn webUI
|
||||||
|
|
||||||
- name: Build modbus
|
- name: Build all target environments from default_envs
|
||||||
run: |
|
|
||||||
platformio run -e build_modbus
|
|
||||||
|
|
||||||
- name: Build standalone
|
|
||||||
run: |
|
|
||||||
platformio run -e build_standalone
|
|
||||||
|
|
||||||
- name: Build all PIO target environments, from default_envs
|
|
||||||
run: |
|
run: |
|
||||||
platformio run
|
platformio run
|
||||||
|
env:
|
||||||
|
NO_BUILD_WEBUI: true
|
||||||
|
|
||||||
- name: Create GitHub Release
|
- name: Create GitHub Release
|
||||||
id: 'automatic_releases'
|
id: 'automatic_releases'
|
||||||
|
|||||||
9
.gitignore
vendored
9
.gitignore
vendored
@@ -2,6 +2,7 @@
|
|||||||
.vscode/c_cpp_properties.json
|
.vscode/c_cpp_properties.json
|
||||||
.vscode/extensions.json
|
.vscode/extensions.json
|
||||||
.vscode/launch.json
|
.vscode/launch.json
|
||||||
|
.vscode/settings.json
|
||||||
|
|
||||||
# c++ compiling
|
# c++ compiling
|
||||||
.clang_complete
|
.clang_complete
|
||||||
@@ -27,10 +28,14 @@ stats.html
|
|||||||
*.sln
|
*.sln
|
||||||
*.sw?
|
*.sw?
|
||||||
.pnp.*
|
.pnp.*
|
||||||
|
*/.yarn/cache/*
|
||||||
|
*/.yarn/install-state.gz
|
||||||
analyse.html
|
analyse.html
|
||||||
interface/vite.config.ts.timestamp*
|
interface/vite.config.ts.timestamp*
|
||||||
*.local
|
*.local
|
||||||
src/ESP32React/WWWData.h
|
src/ESP32React/WWWData.h
|
||||||
|
.yarn/*
|
||||||
|
.yarnrc.yml
|
||||||
|
|
||||||
# i18n generated files
|
# i18n generated files
|
||||||
interface/src/i18n/i18n-react.tsx
|
interface/src/i18n/i18n-react.tsx
|
||||||
@@ -71,7 +76,3 @@ CMakeLists.txt
|
|||||||
logs/*
|
logs/*
|
||||||
sdkconfig.*
|
sdkconfig.*
|
||||||
sdkconfig_tasmota_esp32
|
sdkconfig_tasmota_esp32
|
||||||
pnpm-lock.yaml
|
|
||||||
.cache/
|
|
||||||
interface/.tsbuildinfo
|
|
||||||
test/test_api/package-lock.json
|
|
||||||
|
|||||||
118
CHANGELOG.md
118
CHANGELOG.md
@@ -5,120 +5,6 @@ All notable changes to this project will be documented in this file.
|
|||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
## [3.8.1] 11 January 2026
|
|
||||||
|
|
||||||
## Added
|
|
||||||
|
|
||||||
- update time saved in nvs
|
|
||||||
- heatpump entities [#2883](https://github.com/emsesp/EMS-ESP32/issues/2883)
|
|
||||||
- HA input number format (mode) selectable box/slider (slider for max range 100) [#2900](https://github.com/emsesp/EMS-ESP32/discussions/2900)
|
|
||||||
|
|
||||||
## Fixed
|
|
||||||
|
|
||||||
- fix EMS bus disconnected errors on some systems [#2881](https://github.com/emsesp/EMS-ESP32/issues/2881)
|
|
||||||
- selflowtemp fix [#2876](https://github.com/emsesp/EMS-ESP32/issues/2876)
|
|
||||||
- updated valid GPIOs for ESP32S2, ESP32S3 and ESP32 that caused custom systems to block gpios [#2887](https://github.com/emsesp/EMS-ESP32/issues/2887)
|
|
||||||
- Junkers wwcharge offset [#2860](https://github.com/emsesp/EMS-ESP32/issues/2860)
|
|
||||||
- fixed minflowtemp [#2890](https://github.com/emsesp/EMS-ESP32/issues/2890)
|
|
||||||
- don't add HA uom/classes for bool values [#2885](https://github.com/emsesp/EMS-ESP32/issues/2885)
|
|
||||||
- fixed missing progress bar on web firmware uploads
|
|
||||||
|
|
||||||
## Changed
|
|
||||||
|
|
||||||
- snapshot gpios stored in temporary ram
|
|
||||||
- GPIOs stored along with the name and reported in log if conflicting
|
|
||||||
- free GPIOs depend on board profile [#2901](https://github.com/emsesp/EMS-ESP32/issues/2901)
|
|
||||||
- prefer PSram for mqtt queue [#2889](https://github.com/emsesp/EMS-ESP32/issues/2889)
|
|
||||||
- day schedule defult to all days, no day selected is not allowed
|
|
||||||
- board profile `CUSTOM` can only be selected in developer mode
|
|
||||||
- mqtt sends round values without decimals (`28` instead of `28.0`)
|
|
||||||
|
|
||||||
|
|
||||||
## [3.8.0] 31 December 2025
|
|
||||||
|
|
||||||
## Added
|
|
||||||
|
|
||||||
- analogsensor types: NTC and RGB-Led
|
|
||||||
- Flag for HMC310 [#2465](https://github.com/emsesp/EMS-ESP32/issues/2465)
|
|
||||||
- boiler auxheatersource [#2489](https://github.com/emsesp/EMS-ESP32/discussions/2489)
|
|
||||||
- thermostat last error for RC100/300 [#2501](https://github.com/emsesp/EMS-ESP32/issues/2501)
|
|
||||||
- boiler 0xC6 telegram [#1963](https://github.com/emsesp/EMS-ESP32/issues/1963)
|
|
||||||
- CS6800i changes [#2448](https://github.com/emsesp/EMS-ESP32/issues/2448), [#2449](https://github.com/emsesp/EMS-ESP32/issues/2449)
|
|
||||||
- charging pump [#2544](https://github.com/emsesp/EMS-ESP32/issues/2544)
|
|
||||||
- hybrid CSH5800iG [#2569](https://github.com/emsesp/EMS-ESP32/issues/2569)
|
|
||||||
- added EMS Device details to Home Assistant MQTT Discovery
|
|
||||||
- disinfection command [#2601](https://github.com/emsesp/EMS-ESP32/issues/2601)
|
|
||||||
- added new board profile for upcoming BBQKees E32V2.2
|
|
||||||
- set differential pressure entity in Mixer device
|
|
||||||
- set set climate action cooling/heating in HA [#2583](https://github.com/emsesp/EMS-ESP32/issues/2583)
|
|
||||||
- Internal sensors of E32V2_2
|
|
||||||
- FW200 display options [#2610](https://github.com/emsesp/EMS-ESP32/discussions/2610)
|
|
||||||
- CR11 mode settings OFF/MANUAL depends on selTemp [#2437](https://github.com/emsesp/EMS-ESP32/issues/2437)
|
|
||||||
- implemented eFuse settings for BBQKees boards to store model type and ESP chipset
|
|
||||||
- analogsensors for pulse output [#2624](https://github.com/emsesp/EMS-ESP32/discussions/2624)
|
|
||||||
- analogsensors frequency input [#2631](https://github.com/emsesp/EMS-ESP32/discussions/2631)
|
|
||||||
- SRC plus thermostats [#2636](https://github.com/emsesp/EMS-ESP32/issues/2636)
|
|
||||||
- Greenstar 2000 [#2645](https://github.com/emsesp/EMS-ESP32/issues/2645)
|
|
||||||
- RC3xx `dhw modetype` [#2659](https://github.com/emsesp/EMS-ESP32/discussions/2659)
|
|
||||||
- new boiler entities VR0,VR1, compressor speed [#2669](https://github.com/emsesp/EMS-ESP32/issues/2669)
|
|
||||||
- solar temperature TS16 [#2690](https://github.com/emsesp/EMS-ESP32/issues/2690)
|
|
||||||
- pumpmode enum for HT3 boilers, add commands for manual defrost, chimneysweeper [#2727](https://github.com/emsesp/EMS-ESP32/issues/2727)
|
|
||||||
- pid settings [#2735](https://github.com/emsesp/EMS-ESP32/issues/2735)
|
|
||||||
- refresh MQTT button added to MQTT Settings page
|
|
||||||
- heating assistance, rounding custum settings [#2763](https://github.com/emsesp/EMS-ESP32/discussions/2763)
|
|
||||||
- added counter 0..2 for short pulses, high frequency [#2758](https://github.com/emsesp/EMS-ESP32/issues/2758)
|
|
||||||
- added LWT (Last Will and Testament) to MQTT entities in Home Assistant
|
|
||||||
- added api/metrics endpoint for prometheus integration by @gr3enk [#2774](https://github.com/emsesp/EMS-ESP32/pull/2774)
|
|
||||||
- added RTL8201 to eth phy list [#2800](https://github.com/emsesp/EMS-ESP32/issues/2800)
|
|
||||||
- added partitions to Web UI Version page, so previous firmware versions can be installed [#2837](https://github.com/emsesp/EMS-ESP32/issues/2837)
|
|
||||||
- button pressures show LED. On a long press (10 seconds) the LED flashes for 5 seconds to indicate a factory reset is about to happen. [#2848](https://github.com/emsesp/EMS-ESP32/issues/2848)
|
|
||||||
- added `txpause` command to pause the TX, by setting Txmode to 0 (disabled) [#2850](https://github.com/emsesp/EMS-ESP32/issues/2850)
|
|
||||||
|
|
||||||
## Fixed
|
|
||||||
|
|
||||||
- dhw/switchtime [#2490](https://github.com/emsesp/EMS-ESP32/issues/2490)
|
|
||||||
- switch to secure mqtt [#2492](https://github.com/emsesp/EMS-ESP32/issues/2492)
|
|
||||||
- update link buttons [#2497](https://github.com/emsesp/EMS-ESP32/issues/2497)
|
|
||||||
- refresh scheduler states [#2502](https://github.com/emsesp/EMS-ESP32/discussions/2502)
|
|
||||||
- also rebuild HA config on mqtt connect for scheduler, custom and shower
|
|
||||||
- FB100 controls the hc, not the master [#2510](https://github.com/emsesp/EMS-ESP32/issues/2510)
|
|
||||||
- IPM DHW module, [#2524](https://github.com/emsesp/EMS-ESP32/issues/2524)
|
|
||||||
- charge optimization [#2543](https://github.com/emsesp/EMS-ESP32/issues/2543)
|
|
||||||
- shower active state retained, shows correctly in HA
|
|
||||||
- MQTT Command Topic with slashes [#2571](https://github.com/emsesp/EMS-ESP32/issues/2571)
|
|
||||||
- Add pulsed water meter input to V1.3 gateway with Lilygo S3 [#2550](https://github.com/emsesp/EMS-ESP32/issues/2550)
|
|
||||||
- fix missing long 10-second press of Button to perform a factory reset
|
|
||||||
- fix wwMaxPower on Junkers ZBS14 [#2609](https://github.com/emsesp/EMS-ESP32/issues/2609)
|
|
||||||
- ventilation bypass state from telegram 0x55C [#1197](https://github.com/emsesp/EMS-ESP32/issues/1197)
|
|
||||||
- set selflowtemp for ems+ boilers [#2641](https://github.com/emsesp/EMS-ESP32/discussions/2641)
|
|
||||||
- syslog timestamp [#2704](https://github.com/emsesp/EMS-ESP32/issues/2704)
|
|
||||||
- fixed FS format command [#2720](https://github.com/emsesp/EMS-ESP32/discussions/2720)
|
|
||||||
- dhw priority setting to boiler and mixer, telegrams 0x2CC, 0x2CD, etc.
|
|
||||||
- check for valid GPIOs when board profile is changed [#2841](https://github.com/emsesp/EMS-ESP32/issues/2841)
|
|
||||||
|
|
||||||
## Changed
|
|
||||||
|
|
||||||
- show console log with ISO date/time [#2533](https://github.com/emsesp/EMS-ESP32/discussions/2533)
|
|
||||||
- removed ESP32 CPU temperature
|
|
||||||
- updated core libraries like AsyncTCP, AsyncWebServer and Modbus
|
|
||||||
- remove command `scan deep`
|
|
||||||
- ignore repeated `forceheatingoff` commands [#2641](https://github.com/emsesp/EMS-ESP32/discussions/2641)
|
|
||||||
- optimized web for better performance by adding lazy loading and caching
|
|
||||||
- internal system analog sensors (core_voltage, supply_voltage and gateway_temperature) cannot be accidentally removed
|
|
||||||
- double click button reconnects EMS-ESP to AP
|
|
||||||
- place system message command in side scheduler loop to reduce stack memory usage by 2KB
|
|
||||||
- syslog mark interval set to 1 hour
|
|
||||||
- handle process_telegram in oneloop
|
|
||||||
- improved GPIO validation for Analog Sensors and System GPIOs
|
|
||||||
- entities with no values are greyed out in the Web UI in the Customization page
|
|
||||||
- added System Status to Web Status page
|
|
||||||
- show number on entities and supported languages in log on boot
|
|
||||||
- on tx read fail delay the 3rd. retry 2 sec
|
|
||||||
- move vectors and lists to PSRAM
|
|
||||||
- removed unused last topic/payload echo-check
|
|
||||||
- added Home Assistant device details to MQTT Discovery for all devices
|
|
||||||
- device_class and state_class changes for HA MQTT Discovery [#2825](https://github.com/emsesp/EMS-ESP32/issues/2825)
|
|
||||||
|
|
||||||
## [3.7.2] 22 March 2025
|
## [3.7.2] 22 March 2025
|
||||||
|
|
||||||
## Added
|
## Added
|
||||||
@@ -199,7 +85,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||||||
- The automatically generated temperature sensor ID has replaced dashes (`-`) with underscores (`_`) to be compatible with Home Assistant.
|
- The automatically generated temperature sensor ID has replaced dashes (`-`) with underscores (`_`) to be compatible with Home Assistant.
|
||||||
- `api/system/info` has it's JSON key names changed to camelCase syntax.
|
- `api/system/info` has it's JSON key names changed to camelCase syntax.
|
||||||
|
|
||||||
For more details go to [emsesp.org](https://emsesp.org/).
|
For more details go to [docs.emsesp.org](https://docs.emsesp.org/).
|
||||||
|
|
||||||
## Added
|
## Added
|
||||||
|
|
||||||
@@ -331,7 +217,7 @@ For more details go to [emsesp.org](https://emsesp.org/).
|
|||||||
|
|
||||||
## **IMPORTANT! BREAKING CHANGES**
|
## **IMPORTANT! BREAKING CHANGES**
|
||||||
|
|
||||||
Writeable Text entities have moved from type `sensor` to `text` in Home Assistant to make them also editable within an HA dashboard. Examples are `datetime`, `holidays`, `switchtime`, `vacations`, `maintenancedate`... You will need to manually remove any old discovery topics from your MQTT broker using an application like MQTT Explorer.
|
Writeable Text entities have moved from type `sensor` to `text` in Home Assistant to make them also editable within an HA dashboard. Examples are `datetime`, `holidays`, `switchtime`, `vacations`, `maintenancedate`. You will need to manually remove any old discovery topics from your MQTT broker using an application like MQTT Explorer.
|
||||||
|
|
||||||
## Added
|
## Added
|
||||||
|
|
||||||
|
|||||||
@@ -1,33 +1 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
For more details go to [emsesp.org](https://emsesp.org/).
|
|
||||||
|
|
||||||
## [3.8.2]
|
|
||||||
|
|
||||||
## Added
|
|
||||||
|
|
||||||
- comfortpoint for BC400 [#2935](https://github.com/emsesp/EMS-ESP32/issues/2935)
|
|
||||||
- customize device brand [#2784](https://github.com/emsesp/EMS-ESP32/issues/2784)
|
|
||||||
- set model for ems-esp devices temperature, analog, etc. [#2958](https://github.com/emsesp/EMS-ESP32/discussions/2958)
|
|
||||||
- prometheus metrics for temperature/analog/scheduler/custom [#2962](https://github.com/emsesp/EMS-ESP32/issues/2962)
|
|
||||||
- boiler pumpkick [#2965](https://github.com/emsesp/EMS-ESP32/discussions/2965)
|
|
||||||
- heatpump reset [#2933](https://github.com/emsesp/EMS-ESP32/issues/2933)
|
|
||||||
- e-mail notification using ReadyMail Client
|
|
||||||
- 2.nd freshwater module (dhw4) [#2991](https://github.com/emsesp/EMS-ESP32/issues/2991)
|
|
||||||
|
|
||||||
## Fixed
|
|
||||||
|
|
||||||
- SRC climate creation [#2936](https://github.com/emsesp/EMS-ESP32/issues/2936) and [#2960](https://github.com/emsesp/EMS-ESP32/issues/2960)
|
|
||||||
|
|
||||||
## Changed
|
|
||||||
|
|
||||||
- weblogbuffer up to 1000 messages with PSRAM, mentioned in [#2933](https://github.com/emsesp/EMS-ESP32/issues/2933)
|
|
||||||
- validate custom entity writes, [#2931](https://github.com/emsesp/EMS-ESP32/issues/2931)
|
|
||||||
- remove wrong burnMinPower [#2918](https://github.com/emsesp/EMS-ESP32/issues/2918)
|
|
||||||
- store scheduler active state to nvs [#2946](https://github.com/emsesp/EMS-ESP32/discussions/2946)
|
|
||||||
- translated modes `heat` and `eco` for HA-climate mode-str-tpl
|
|
||||||
- support `minflowtemp` and `baseflowtemp` [#2969](https://github.com/emsesp/EMS-ESP32/discussions/2969)
|
|
||||||
- update version if it is 00.00 in first read [#2981](https://github.com/emsesp/EMS-ESP32/issues/2981)
|
|
||||||
- device class for % values [#2980](https://github.com/emsesp/EMS-ESP32/issues/2980)
|
|
||||||
- use tasmota core 2026.03.30
|
|
||||||
- secure mqtt uses ESP_SSLClient
|
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ Everybody is welcome and invited to contribute to the EMS-ESP Project by:
|
|||||||
|
|
||||||
- providing Pull Requests (Features, Fixes, suggestions)
|
- providing Pull Requests (Features, Fixes, suggestions)
|
||||||
- testing new released features and report issues on your EMS equipment
|
- testing new released features and report issues on your EMS equipment
|
||||||
- contributing to missing [documentation](https://emsesp.org)
|
- contributing to missing [documentation](https://docs.emsesp.org)
|
||||||
|
|
||||||
This document describes rules that are in effect for this repository, meant for handling issues by contributors in the issue tracker and PRs.
|
This document describes rules that are in effect for this repository, meant for handling issues by contributors in the issue tracker and PRs.
|
||||||
|
|
||||||
@@ -69,7 +69,7 @@ Format: `<type>(<scope>): <subject>`
|
|||||||
|
|
||||||
## Example
|
## Example
|
||||||
|
|
||||||
```text
|
```
|
||||||
feat: add hat wobble
|
feat: add hat wobble
|
||||||
^--^ ^------------^
|
^--^ ^------------^
|
||||||
| |
|
| |
|
||||||
@@ -96,7 +96,7 @@ References:
|
|||||||
|
|
||||||
## Contributor License Agreement (CLA)
|
## Contributor License Agreement (CLA)
|
||||||
|
|
||||||
```text
|
```
|
||||||
By making a contribution to this project, I certify that:
|
By making a contribution to this project, I certify that:
|
||||||
|
|
||||||
(a) The contribution was created in whole or in part by me and I
|
(a) The contribution was created in whole or in part by me and I
|
||||||
|
|||||||
78
Makefile
78
Makefile
@@ -19,20 +19,17 @@ C = $(words $N)$(eval N := x $N)
|
|||||||
ECHO = python3 $(I)/scripts/echo_progress.py --stepno=$C --nsteps=$T
|
ECHO = python3 $(I)/scripts/echo_progress.py --stepno=$C --nsteps=$T
|
||||||
endif
|
endif
|
||||||
|
|
||||||
# Optimize parallel build configuration
|
# determine number of parallel compiles based on OS
|
||||||
UNAME_S := $(shell uname -s)
|
UNAME_S := $(shell uname -s)
|
||||||
JOBS ?= 1
|
|
||||||
ifeq ($(UNAME_S),Linux)
|
ifeq ($(UNAME_S),Linux)
|
||||||
EXTRA_CPPFLAGS = -D LINUX
|
EXTRA_CPPFLAGS = -D LINUX
|
||||||
JOBS := $(shell nproc)
|
JOBS ?= $(shell nproc)
|
||||||
endif
|
endif
|
||||||
ifeq ($(UNAME_S),Darwin)
|
ifeq ($(UNAME_S),Darwin)
|
||||||
EXTRA_CPPFLAGS = -D OSX -Wno-tautological-constant-out-of-range-compare
|
EXTRA_CPPFLAGS = -D OSX -Wno-tautological-constant-out-of-range-compare
|
||||||
JOBS := $(shell sysctl -n hw.ncpu)
|
JOBS ?= $(shell sysctl -n hw.ncpu)
|
||||||
endif
|
endif
|
||||||
|
MAKEFLAGS += -j $(JOBS) -l $(JOBS)
|
||||||
# Set optimal parallel build settings
|
|
||||||
MAKEFLAGS += -j$(JOBS) -l$(shell echo $$(($(JOBS) * 2)))
|
|
||||||
|
|
||||||
# $(info Number of jobs: $(JOBS))
|
# $(info Number of jobs: $(JOBS))
|
||||||
|
|
||||||
@@ -47,8 +44,8 @@ MAKEFLAGS += -j$(JOBS) -l$(shell echo $$(($(JOBS) * 2)))
|
|||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
TARGET := emsesp
|
TARGET := emsesp
|
||||||
BUILD := build
|
BUILD := build
|
||||||
SOURCES := src/core src/devices src/web src/test lib_standalone lib/semver lib/espMqttClient/src lib/espMqttClient/src/* lib/ArduinoJson/src lib/uuid-common/src lib/uuid-console/src lib/uuid-log/src lib/PButton
|
SOURCES := src/core src/devices src/web src/test lib_standalone lib/semver lib/espMqttClient/src lib/espMqttClient/src/* lib/ArduinoJson/src lib/uuid-common/src lib/uuid-console/src lib/uuid-log/src lib/PButton
|
||||||
INCLUDES := src/core src/devices src/web src/test lib_standalone lib/* lib/semver lib/espMqttClient/src lib/espMqttClient/src/Transport lib/ArduinoJson/src lib/uuid-common/src lib/uuid-console/src lib/uuid-log/src lib/uuid-telnet/src lib/uuid-syslog/src
|
INCLUDES := src/core src/devices src/web src/test lib/* lib_standalone lib/semver lib/espMqttClient/src lib/espMqttClient/src/Transport lib/ArduinoJson/src lib/uuid-common/src lib/uuid-console/src lib/uuid-log/src lib/uuid-telnet/src lib/uuid-syslog/src
|
||||||
LIBRARIES :=
|
LIBRARIES :=
|
||||||
|
|
||||||
CPPCHECK = cppcheck
|
CPPCHECK = cppcheck
|
||||||
@@ -63,12 +60,11 @@ CXX_STANDARD := -std=gnu++17
|
|||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
# Defined Symbols
|
# Defined Symbols
|
||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
DEFINES += -DARDUINOJSON_ENABLE -DARDUINOJSON_ENABLE_ARDUINO_STRING -DARDUINOJSON_USE_DOUBLE=0
|
DEFINES += -DARDUINOJSON_ENABLE -DARDUINOJSON_ENABLE_ARDUINO_STRING -DARDUINOJSON_USE_DOUBLE=0
|
||||||
DEFINES += -DEMSESP_STANDALONE -DEMSESP_TEST -DEMSESP_DEBUG -DEMC_RX_BUFFER_SIZE=1500
|
DEFINES += -DEMSESP_STANDALONE -DEMSESP_TEST -DEMSESP_DEBUG -DEMC_RX_BUFFER_SIZE=1500
|
||||||
DEFINES += -DNO_TLS_SUPPORT
|
|
||||||
DEFINES += $(ARGS)
|
DEFINES += $(ARGS)
|
||||||
|
|
||||||
DEFAULTS = -DEMSESP_DEFAULT_LOCALE=\"en\" -DEMSESP_DEFAULT_BOARD_PROFILE=\"S32S3\"
|
DEFAULTS = -DEMSESP_DEFAULT_LOCALE=\"en\" -DEMSESP_DEFAULT_TX_MODE=8 -DEMSESP_DEFAULT_VERSION=\"3.7.3-dev\" -DEMSESP_DEFAULT_BOARD_PROFILE=\"S32S3\"
|
||||||
|
|
||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
# Sources & Files
|
# Sources & Files
|
||||||
@@ -76,25 +72,16 @@ DEFAULTS = -DEMSESP_DEFAULT_LOCALE=\"en\" -DEMSESP_DEFAULT_BOARD_PROFILE=\"S32S3
|
|||||||
OUTPUT := $(CURDIR)/$(TARGET)
|
OUTPUT := $(CURDIR)/$(TARGET)
|
||||||
SYMBOLS := $(CURDIR)/$(BUILD)/$(TARGET).out
|
SYMBOLS := $(CURDIR)/$(BUILD)/$(TARGET).out
|
||||||
|
|
||||||
# Optimize source discovery - use shell find for better performance
|
CSOURCES := $(foreach dir,$(SOURCES),$(wildcard $(dir)/*.c))
|
||||||
CSOURCES := $(shell find $(SOURCES) -name "*.c" 2>/dev/null)
|
CXXSOURCES := $(foreach dir,$(SOURCES),$(wildcard $(dir)/*.cpp))
|
||||||
CXXSOURCES := $(shell find $(SOURCES) -name "*.cpp" 2>/dev/null)
|
|
||||||
|
|
||||||
# Exclude files not needed for standalone build, if they exist
|
OBJS := $(patsubst %,$(BUILD)/%.o,$(basename $(CSOURCES)) $(basename $(CXXSOURCES)) )
|
||||||
CSOURCES := $(filter-out src/core/ModuleLibrary.c,$(CSOURCES))
|
DEPS := $(patsubst %,$(BUILD)/%.d,$(basename $(CSOURCES)) $(basename $(CXXSOURCES)) )
|
||||||
CXXSOURCES := $(filter-out src/core/ModuleLibrary.cpp,$(CXXSOURCES))
|
|
||||||
|
|
||||||
OBJS := $(patsubst %,$(BUILD)/%.o,$(basename $(CSOURCES)) $(basename $(CXXSOURCES)))
|
INCLUDE += $(addprefix -I,$(foreach dir,$(INCLUDES), $(wildcard $(dir))))
|
||||||
DEPS := $(patsubst %,$(BUILD)/%.d,$(basename $(CSOURCES)) $(basename $(CXXSOURCES)))
|
INCLUDE += $(addprefix -I,$(foreach dir,$(LIBRARIES),$(wildcard $(dir)/include)))
|
||||||
|
|
||||||
# Optimize include path discovery
|
LDLIBS += $(addprefix -L,$(foreach dir,$(LIBRARIES),$(wildcard $(dir)/lib)))
|
||||||
INCLUDE_DIRS := $(shell find $(INCLUDES) -type d 2>/dev/null)
|
|
||||||
LIBRARY_INCLUDES := $(shell find $(LIBRARIES) -name "include" -type d 2>/dev/null)
|
|
||||||
INCLUDE += $(addprefix -I,$(INCLUDE_DIRS) $(LIBRARY_INCLUDES))
|
|
||||||
|
|
||||||
# Optimize library path discovery
|
|
||||||
LIBRARY_DIRS := $(shell find $(LIBRARIES) -name "lib" -type d 2>/dev/null)
|
|
||||||
LDLIBS += $(addprefix -L,$(LIBRARY_DIRS))
|
|
||||||
|
|
||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
# Compiler & Linker
|
# Compiler & Linker
|
||||||
@@ -111,12 +98,13 @@ CXX := /usr/bin/g++
|
|||||||
# LDFLAGS Linker Flags
|
# LDFLAGS Linker Flags
|
||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
CPPFLAGS += $(DEFINES) $(DEFAULTS) $(INCLUDE)
|
CPPFLAGS += $(DEFINES) $(DEFAULTS) $(INCLUDE)
|
||||||
CPPFLAGS += -ggdb -g3 -MMD
|
CPPFLAGS += -ggdb -g3 -O3
|
||||||
CPPFLAGS += -flto=auto
|
CPPFLAGS += -MMD
|
||||||
CPPFLAGS += -Wall -Wextra -Werror -Wswitch-enum
|
CPPFLAGS += -flto=auto -fno-lto
|
||||||
CPPFLAGS += -Wno-unused-parameter -Wno-missing-braces -Wno-vla-cxx-extension
|
CPPFLAGS += -Wall -Wextra -Werror
|
||||||
CPPFLAGS += -ffunction-sections -fdata-sections -fno-exceptions -fno-rtti -fno-threadsafe-statics
|
CPPFLAGS += -Wswitch-enum
|
||||||
CPPFLAGS += -Os -DNDEBUG
|
CPPFLAGS += -Wno-unused-parameter
|
||||||
|
CPPFLAGS += -Wno-missing-braces
|
||||||
|
|
||||||
CPPFLAGS += $(EXTRA_CPPFLAGS)
|
CPPFLAGS += $(EXTRA_CPPFLAGS)
|
||||||
|
|
||||||
@@ -137,13 +125,11 @@ else
|
|||||||
LD := $(CXX)
|
LD := $(CXX)
|
||||||
endif
|
endif
|
||||||
|
|
||||||
# Dependency file generation
|
#DEPFLAGS += -MF $(BUILD)/$*.d
|
||||||
DEPFLAGS += -MF $(BUILD)/$*.d -MT $@
|
|
||||||
|
|
||||||
LINK.o = $(LD) $(LDFLAGS) $(LDLIBS) $^ -o $@
|
LINK.o = $(LD) $(LDFLAGS) $(LDLIBS) $^ -o $@
|
||||||
COMPILE.c = $(CC) $(C_STANDARD) $(CFLAGS) $(DEPFLAGS) -c $< -o $@
|
COMPILE.c = $(CC) $(C_STANDARD) $(CFLAGS) $(DEPFLAGS) -c $< -o $@
|
||||||
COMPILE.cpp = $(CXX) $(CXX_STANDARD) $(CXXFLAGS) $(DEPFLAGS) -c $< -o $@
|
COMPILE.cpp = $(CXX) $(CXX_STANDARD) $(CXXFLAGS) $(DEPFLAGS) -c $< -o $@
|
||||||
COMPILE.s = $(CC) $(CFLAGS) $(DEPFLAGS) -c $< -o $@
|
|
||||||
|
|
||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
# Special Built-in Target
|
# Special Built-in Target
|
||||||
@@ -156,10 +142,7 @@ COMPILE.s = $(CC) $(CFLAGS) $(DEPFLAGS) -c $< -o $@
|
|||||||
.SUFFIXES:
|
.SUFFIXES:
|
||||||
.INTERMEDIATE:
|
.INTERMEDIATE:
|
||||||
.PRECIOUS: $(OBJS) $(DEPS)
|
.PRECIOUS: $(OBJS) $(DEPS)
|
||||||
.PHONY: all clean help cppcheck run
|
.PHONY: all clean help
|
||||||
|
|
||||||
# Enable second expansion for more flexible rules
|
|
||||||
.SECONDEXPANSION:
|
|
||||||
|
|
||||||
#----------------------------------------------------------------------
|
#----------------------------------------------------------------------
|
||||||
# Targets
|
# Targets
|
||||||
@@ -174,6 +157,7 @@ $(OUTPUT): $(OBJS)
|
|||||||
@mkdir -p $(@D)
|
@mkdir -p $(@D)
|
||||||
@$(ECHO) Linking $@
|
@$(ECHO) Linking $@
|
||||||
$(LINK.o)
|
$(LINK.o)
|
||||||
|
$(SYMBOLS.out)
|
||||||
|
|
||||||
$(BUILD)/%.o: %.c
|
$(BUILD)/%.o: %.c
|
||||||
@mkdir -p $(@D)
|
@mkdir -p $(@D)
|
||||||
@@ -187,7 +171,6 @@ $(BUILD)/%.o: %.cpp
|
|||||||
|
|
||||||
$(BUILD)/%.o: %.s
|
$(BUILD)/%.o: %.s
|
||||||
@mkdir -p $(@D)
|
@mkdir -p $(@D)
|
||||||
@$(ECHO) Compiling $@
|
|
||||||
@$(COMPILE.s)
|
@$(COMPILE.s)
|
||||||
|
|
||||||
cppcheck: $(SOURCES)
|
cppcheck: $(SOURCES)
|
||||||
@@ -202,15 +185,8 @@ clean:
|
|||||||
@$(RM) -rf $(BUILD) $(OUTPUT)
|
@$(RM) -rf $(BUILD) $(OUTPUT)
|
||||||
|
|
||||||
help:
|
help:
|
||||||
@echo "Available targets:"
|
@echo available targets: all run clean
|
||||||
@echo " all - Build the project (default)"
|
@echo $(OUTPUT)
|
||||||
@echo " run - Build and run the executable"
|
|
||||||
@echo " clean - Remove build artifacts"
|
|
||||||
@echo " cppcheck - Run static analysis"
|
|
||||||
@echo " help - Show this help message"
|
|
||||||
@echo ""
|
|
||||||
@echo "Output: $(OUTPUT)"
|
|
||||||
@echo "Jobs: $(JOBS)"
|
|
||||||
|
|
||||||
-include $(DEPS)
|
-include $(DEPS)
|
||||||
|
|
||||||
|
|||||||
32
README.md
32
README.md
@@ -15,10 +15,10 @@
|
|||||||
<a href="https://github.com/emsesp/EMS-ESP32/blob/dev/CONTRIBUTING.md">
|
<a href="https://github.com/emsesp/EMS-ESP32/blob/dev/CONTRIBUTING.md">
|
||||||
<img src="https://img.shields.io/badge/Contribute-ff4785?style=for-the-badge&logo=git&logoColor=white" alt="Contribute" />
|
<img src="https://img.shields.io/badge/Contribute-ff4785?style=for-the-badge&logo=git&logoColor=white" alt="Contribute" />
|
||||||
</a>
|
</a>
|
||||||
<a href="https://emsesp.org">
|
<a href="https://docs.emsesp.org">
|
||||||
<img src="https://img.shields.io/badge/Documentation-0077b5?style=for-the-badge&logo=googledocs&logoColor=white" alt="Guides" />
|
<img src="https://img.shields.io/badge/Documentation-0077b5?style=for-the-badge&logo=googledocs&logoColor=white" alt="Guides" />
|
||||||
</a>
|
</a>
|
||||||
<a href="https://discord.gg/GP9DPSgeJq">
|
<a href="https://discord.gg/3J3GgnzpyT">
|
||||||
<img src="https://img.shields.io/badge/Discord-7289da?style=for-the-badge&logo=discord&logoColor=white" alt="Discord" />
|
<img src="https://img.shields.io/badge/Discord-7289da?style=for-the-badge&logo=discord&logoColor=white" alt="Discord" />
|
||||||
</a>
|
</a>
|
||||||
<a href="https://github.com/emsesp/EMS-ESP32/blob/main/CHANGELOG.md">
|
<a href="https://github.com/emsesp/EMS-ESP32/blob/main/CHANGELOG.md">
|
||||||
@@ -32,17 +32,15 @@
|
|||||||
[](https://sonarcloud.io/summary/new_code?id=emsesp_EMS-ESP32)
|
[](https://sonarcloud.io/summary/new_code?id=emsesp_EMS-ESP32)
|
||||||
[](https://app.codacy.com/gh/emsesp/EMS-ESP32/dashboard?utm_source=gh&utm_medium=referral&utm_content=&utm_campaign=Badge_grade)
|
[](https://app.codacy.com/gh/emsesp/EMS-ESP32/dashboard?utm_source=gh&utm_medium=referral&utm_content=&utm_campaign=Badge_grade)
|
||||||
[](https://github.com/emsesp/EMS-ESP32/releases)
|
[](https://github.com/emsesp/EMS-ESP32/releases)
|
||||||
[](https://discord.gg/GP9DPSgeJq)
|
[](https://discord.gg/3J3GgnzpyT)
|
||||||
[](https://deepwiki.com/emsesp/EMS-ESP32)
|
|
||||||
|
|
||||||
[](https://github.com/emsesp/EMS-ESP32/stargazers)
|
[](https://github.com/emsesp/EMS-ESP32/stargazers)
|
||||||
[](https://github.com/emsesp/EMS-ESP32/network)
|
[](https://github.com/emsesp/EMS-ES32P/network)
|
||||||
[](https://www.paypal.com/paypalme/prderbyshire/2)
|
[](https://www.paypal.com/paypalme/prderbyshire/2)
|
||||||
|
|
||||||
**EMS-ESP** is an open-source firmware for the Espressif ESP32 microcontroller to communicate with **EMS** (Energy Management System) compatible equipment from manufacturers such as Bosch, Buderus, Nefit, Junkers, Worcester, Sieger, elm.leblanc and iVT.
|
**EMS-ESP** is an open-source firmware for the Espressif ESP32 microcontroller to communicate with **EMS** (Energy Management System) compatible equipment from manufacturers such as Bosch, Buderus, Nefit, Junkers, Worcester, Sieger, elm.leblanc and iVT.
|
||||||
|
|
||||||
It requires a small circuit to interface with the EMS bus which can be purchased from <https://bbqkees-electronics.nl>. These gateways are tested thoroughly and certified to work with EMS-ESP.
|
It requires a small circuit to interface with the EMS bus which can be purchased from <https://bbqkees-electronics.nl> or custom built.
|
||||||
|
|
||||||
|
|
||||||
## 📦 **Key Features**
|
## 📦 **Key Features**
|
||||||
|
|
||||||
@@ -62,39 +60,35 @@ It requires a small circuit to interface with the EMS bus which can be purchased
|
|||||||
|
|
||||||
## 🚀 **Installing**
|
## 🚀 **Installing**
|
||||||
|
|
||||||
Head over to the [Installation Guide](https://emsesp.org/Installing) section of the documentation for instructions on how to install EMS-ESP.
|
Head over to [download.emsesp.org](https://download.emsesp.org) for instructions on how to install EMS-ESP. There is also further details on which boards are supported in [this section](https://docs.emsesp.org/Getting-Started/#first-time-install) of the documentation.
|
||||||
|
|
||||||
## 📋 **Documentation**
|
## 📋 **Documentation**
|
||||||
|
|
||||||
Visit [emsesp.org](https://emsesp.org) for more details on how to setup and configure EMS-ESP. You'll also find more a collection of example configuarations, Frequently Asked Questions and Troubleshooting tips.
|
Visit [emsesp.org](https://docs.emsesp.org) for more details on how to install and configure EMS-ESP. There is also a collection of Frequently Asked Questions and Troubleshooting tips with example customizations from the community.
|
||||||
|
|
||||||
## 💬 **Getting Support**
|
## 💬 **Getting Support**
|
||||||
|
|
||||||
To chat with the community reach out on our [Discord Server](https://discord.gg/GP9DPSgeJq).
|
To chat with the community reach out on our [Discord Server](https://discord.gg/3J3GgnzpyT).
|
||||||
|
|
||||||
If you find an issue or have a request, see the [Getting Support](https://emsesp.org/Support/) section of the documentation. Note if you are using a non-BBQKees EMS gateway, you may need to contact the manufacturer for support.
|
If you find an issue or have a request, see [here](https://docs.emsesp.org/Support/) on how to submit a bug report or feature request.
|
||||||
|
|
||||||
## 🎥 **Live Demo**
|
## 🎥 **Live Demo**
|
||||||
|
|
||||||
To see a live demo go to [demo.emsesp.org](https://demo.emsesp.org). Pick a language and use any username and password to log in. Note whast you're seeing is static example data so not all features are operational.
|
For a live demo go to [demo.emsesp.org](https://demo.emsesp.org). Pick a language from the sign on page and log in with any username or password. Note not all features are operational as it's based on static data.
|
||||||
|
|
||||||
## 💖 **Contributors**
|
## 💖 **Contributors**
|
||||||
|
|
||||||
EMS-ESP is a project originally created by [proddy](https://github.com/proddy) and maintained by the ems-esp community.
|
EMS-ESP is a project created by [proddy](https://github.com/proddy) and owned and maintained by both [proddy](https://github.com/proddy) and [MichaelDvP](https://github.com/MichaelDvP) with support from [BBQKees Electronics](https://bbqkees-electronics.nl).
|
||||||
|
|
||||||
If you like **EMS-ESP**, please give it a ✨ on GitHub, or even better fork it and contribute. You can also offer a small donation. This is an open-source project maintained by volunteers, and your support is greatly appreciated.
|
If you like **EMS-ESP**, please give it a ✨ on GitHub, or even better fork it and contribute. You can also offer a small donation. This is an open-source project maintained by volunteers, and your support is greatly appreciated.
|
||||||
|
|
||||||
## 📦 **Building**
|
|
||||||
|
|
||||||
See the [Building the firmware](https://emsesp.org/Building) guide in the documentation for instructions on how to build EMS-ESP from this source code.
|
|
||||||
|
|
||||||
## 📢 **Libraries used**
|
## 📢 **Libraries used**
|
||||||
|
|
||||||
- [esp8266-react](https://github.com/rjwats/esp8266-react) originally by @rjwats for the core framework that provides the Web UI, which has been heavily modified
|
- [esp8266-react](https://github.com/rjwats/esp8266-react) by @rjwats for the core framework that provides the Web UI, which has been heavily modified
|
||||||
- [uuid-\*](https://github.com/nomis/mcu-uuid-console) from @nomis. The console, syslog, telnet and logging are based off these awesome open source libraries
|
- [uuid-\*](https://github.com/nomis/mcu-uuid-console) from @nomis. The console, syslog, telnet and logging are based off these awesome open source libraries
|
||||||
- [ArduinoJson](https://github.com/bblanchon/ArduinoJson) for all the JSON processing
|
- [ArduinoJson](https://github.com/bblanchon/ArduinoJson) for all the JSON processing
|
||||||
- [espMqttClient](https://github.com/bertmelis/espMqttClient) for the MQTT client
|
- [espMqttClient](https://github.com/bertmelis/espMqttClient) for the MQTT client
|
||||||
- [ESPAsyncWebServer](https://github.com/ESP32Async/ESPAsyncWebServer) and [AsyncTCP](https://github.com/ESP32Async/AsyncTCP) for the Web server
|
- ESPAsyncWebServer and AsyncTCP for the Web server and TCP backends, with custom modifications for performance
|
||||||
|
|
||||||
## 📜 **License**
|
## 📜 **License**
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +0,0 @@
|
|||||||
# Security Policy
|
|
||||||
|
|
||||||
## Reporting a Vulnerability
|
|
||||||
|
|
||||||
Please report any security vulnerabilities using the [Contact Form](https://emsesp.org/About/#-contact).
|
|
||||||
@@ -5,7 +5,7 @@
|
|||||||
},
|
},
|
||||||
"core": "esp32",
|
"core": "esp32",
|
||||||
"extra_flags": [
|
"extra_flags": [
|
||||||
"-DNO_TLS_SUPPORT",
|
"-DTASMOTA_SDK",
|
||||||
"-DARDUINO_LOLIN_C3_MINI",
|
"-DARDUINO_LOLIN_C3_MINI",
|
||||||
"-DARDUINO_USB_MODE=1",
|
"-DARDUINO_USB_MODE=1",
|
||||||
"-DARDUINO_USB_CDC_ON_BOOT=1"
|
"-DARDUINO_USB_CDC_ON_BOOT=1"
|
||||||
|
|||||||
@@ -6,7 +6,7 @@
|
|||||||
"core": "esp32",
|
"core": "esp32",
|
||||||
"extra_flags": [
|
"extra_flags": [
|
||||||
"-DBOARD_HAS_PSRAM",
|
"-DBOARD_HAS_PSRAM",
|
||||||
"-DNO_TLS_SUPPORT",
|
"-DTASMOTA_SDK",
|
||||||
"-DARDUINO_USB_CDC_ON_BOOT=1",
|
"-DARDUINO_USB_CDC_ON_BOOT=1",
|
||||||
"-DARDUINO_USB_MODE=0"
|
"-DARDUINO_USB_MODE=0"
|
||||||
],
|
],
|
||||||
@@ -37,8 +37,8 @@
|
|||||||
"flash_size": "4MB",
|
"flash_size": "4MB",
|
||||||
"maximum_ram_size": 327680,
|
"maximum_ram_size": 327680,
|
||||||
"maximum_size": 4194304,
|
"maximum_size": 4194304,
|
||||||
"use_1200bps_touch": false,
|
"use_1200bps_touch": true,
|
||||||
"wait_for_upload_port": false,
|
"wait_for_upload_port": true,
|
||||||
"require_upload_port": true,
|
"require_upload_port": true,
|
||||||
"speed": 921600
|
"speed": 921600
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -21,11 +21,11 @@
|
|||||||
"arduino",
|
"arduino",
|
||||||
"espidf"
|
"espidf"
|
||||||
],
|
],
|
||||||
"name": "Tasmota ESP32-S3 32M Flash OPI PSRAM, 4608KB Code/OTA, 2MB FS",
|
"name": "Espressif ESP32-S3 32M Flash OPI PSRAM, 4608KB Code/OTA, 2MB FS",
|
||||||
"upload": {
|
"upload": {
|
||||||
"flash_size": "32MB",
|
"flash_size": "32MB",
|
||||||
"maximum_ram_size": 327680,
|
"maximum_ram_size": 327680,
|
||||||
"maximum_size": 33554432,
|
"maximum_size": 16777216,
|
||||||
"require_upload_port": true,
|
"require_upload_port": true,
|
||||||
"speed": 460800
|
"speed": 460800
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"build": {
|
"build": {
|
||||||
"core": "esp32",
|
"core": "esp32",
|
||||||
"extra_flags": "-DNO_TLS_SUPPORT",
|
"extra_flags": "-DTASMOTA_SDK",
|
||||||
"f_cpu": "240000000L",
|
"f_cpu": "240000000L",
|
||||||
"f_flash": "40000000L",
|
"f_flash": "40000000L",
|
||||||
"flash_mode": "dio",
|
"flash_mode": "dio",
|
||||||
@@ -19,7 +19,7 @@
|
|||||||
"arduino",
|
"arduino",
|
||||||
"espidf"
|
"espidf"
|
||||||
],
|
],
|
||||||
"name": "Tasmota ESP32 16M Flash, 4608KB Code/OTA, 2MB FS",
|
"name": "Espressif ESP32 16M Flash, 4608KB Code/OTA, 2MB FS",
|
||||||
"upload": {
|
"upload": {
|
||||||
"flash_size": "16MB",
|
"flash_size": "16MB",
|
||||||
"maximum_ram_size": 327680,
|
"maximum_ram_size": 327680,
|
||||||
|
|||||||
@@ -19,7 +19,7 @@
|
|||||||
"arduino",
|
"arduino",
|
||||||
"espidf"
|
"espidf"
|
||||||
],
|
],
|
||||||
"name": "Tasmota ESP32 16M Flash DIO PSRAM, 4608KB Code/OTA, 2MB FS",
|
"name": "Espressif ESP32 16M Flash DIO PSRAM, 4608KB Code/OTA, 2MB FS",
|
||||||
"upload": {
|
"upload": {
|
||||||
"flash_size": "16MB",
|
"flash_size": "16MB",
|
||||||
"maximum_ram_size": 327680,
|
"maximum_ram_size": 327680,
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"build": {
|
"build": {
|
||||||
"core": "esp32",
|
"core": "esp32",
|
||||||
"extra_flags": "-DNO_TLS_SUPPORT",
|
"extra_flags": "-DTASMOTA_SDK",
|
||||||
"f_cpu": "240000000L",
|
"f_cpu": "240000000L",
|
||||||
"f_flash": "40000000L",
|
"f_flash": "40000000L",
|
||||||
"flash_mode": "dio",
|
"flash_mode": "dio",
|
||||||
|
|||||||
@@ -1,49 +0,0 @@
|
|||||||
{
|
|
||||||
"build": {
|
|
||||||
"core": "esp32",
|
|
||||||
"extra_flags": [
|
|
||||||
"-DNO_TLS_SUPPORT",
|
|
||||||
"-DARDUINO_XIAO_ESP32C6",
|
|
||||||
"-DARDUINO_USB_MODE=1",
|
|
||||||
"-DARDUINO_USB_CDC_ON_BOOT=1"
|
|
||||||
],
|
|
||||||
"f_cpu": "160000000L",
|
|
||||||
"f_flash": "80000000L",
|
|
||||||
"flash_mode": "qio",
|
|
||||||
"hwids": [
|
|
||||||
[
|
|
||||||
"0x2886",
|
|
||||||
"0x0046"
|
|
||||||
],
|
|
||||||
[
|
|
||||||
"0x303a",
|
|
||||||
"0x1001"
|
|
||||||
]
|
|
||||||
],
|
|
||||||
"mcu": "esp32c6",
|
|
||||||
"variant": "XIAO_ESP32C6"
|
|
||||||
},
|
|
||||||
"connectivity": [
|
|
||||||
"wifi",
|
|
||||||
"bluetooth",
|
|
||||||
"zigbee",
|
|
||||||
"thread"
|
|
||||||
],
|
|
||||||
"debug": {
|
|
||||||
"openocd_target": "esp32c6.cfg"
|
|
||||||
},
|
|
||||||
"frameworks": [
|
|
||||||
"arduino",
|
|
||||||
"espidf"
|
|
||||||
],
|
|
||||||
"name": "Seeed Studio XIAO ESP32C6",
|
|
||||||
"upload": {
|
|
||||||
"flash_size": "4MB",
|
|
||||||
"maximum_ram_size": 327680,
|
|
||||||
"maximum_size": 4194304,
|
|
||||||
"require_upload_port": true,
|
|
||||||
"speed": 460800
|
|
||||||
},
|
|
||||||
"url": "https://wiki.seeedstudio.com/XIAO_ESP32C6_Getting_Started/",
|
|
||||||
"vendor": "Seeed Studio"
|
|
||||||
}
|
|
||||||
@@ -9,7 +9,6 @@
|
|||||||
}
|
}
|
||||||
],
|
],
|
||||||
"dictionaries": ["project-words"],
|
"dictionaries": ["project-words"],
|
||||||
"caseSensitive": false,
|
|
||||||
"ignorePaths": [
|
"ignorePaths": [
|
||||||
"node_modules",
|
"node_modules",
|
||||||
"compile_commands.json",
|
"compile_commands.json",
|
||||||
@@ -33,12 +32,6 @@
|
|||||||
"**/*.json",
|
"**/*.json",
|
||||||
"src/core/modbus_entity_parameters.hpp",
|
"src/core/modbus_entity_parameters.hpp",
|
||||||
"sdkconfig.*",
|
"sdkconfig.*",
|
||||||
"managed_components/**",
|
"managed_components/**"
|
||||||
"pnpm-*.yaml",
|
|
||||||
"vite.config.ts",
|
|
||||||
"lib/esp32-psram/**",
|
|
||||||
"test/test_api/test_api.h",
|
|
||||||
"lib_standalone/**",
|
|
||||||
"lib/mbedtls_ssl/**"
|
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
File diff suppressed because it is too large
Load Diff
11457
docs/dump_entities.csv
11457
docs/dump_entities.csv
File diff suppressed because it is too large
Load Diff
@@ -1,233 +1,227 @@
|
|||||||
telegram_type_id,name,is_fetched
|
telegram_type_id,name,is_fetched
|
||||||
0x04,UBAFactory,fetched
|
0x04,UBAFactory,fetched
|
||||||
0x06,RCTime,
|
0x06,RCTime,
|
||||||
0x0A,EasyMonitor,fetched
|
0x0A,EasyMonitor,fetched
|
||||||
0x10,UBAErrorMessage1,
|
0x10,UBAErrorMessage1,
|
||||||
0x11,UBAErrorMessage2,
|
0x11,UBAErrorMessage2,
|
||||||
0x12,RCErrorMessage,
|
0x12,RCErrorMessage,
|
||||||
0x13,RCErrorMessage2,
|
0x13,RCErrorMessage2,
|
||||||
0x14,UBATotalUptime,fetched
|
0x14,UBATotalUptime,fetched
|
||||||
0x15,UBAMaintenanceData,
|
0x15,UBAMaintenanceData,
|
||||||
0x16,UBAParameters,fetched
|
0x16,UBAParameters,fetched
|
||||||
0x18,UBAMonitorFast,
|
0x18,UBAMonitorFast,
|
||||||
0x19,UBAMonitorSlow,
|
0x19,UBAMonitorSlow,
|
||||||
0x1A,UBASetPoints,
|
0x1A,UBASetPoints,
|
||||||
0x1C,UBAMaintenanceStatus,
|
0x1C,UBAMaintenanceStatus,
|
||||||
0x1E,HydrTemp,
|
0x1E,WM10TempMessage,
|
||||||
0x23,JunkersSetMixer,fetched
|
0x23,JunkersSetMixer,fetched
|
||||||
0x27,UBASettingsWW,fetched
|
0x26,UBASettingsWW,fetched
|
||||||
0x28,WeatherComp,fetched
|
0x28,WeatherComp,fetched
|
||||||
0x2A,MC110Status,
|
0x2A,MC110Status,
|
||||||
0x2E,Meters,
|
0x2E,Meters,
|
||||||
0x33,UBAParameterWW,fetched
|
0x33,UBAParameterWW,fetched
|
||||||
0x34,UBAMonitorWW,
|
0x34,UBAMonitorWW,
|
||||||
0x35,UBAFlags,
|
0x35,UBAFlags,
|
||||||
0x37,WWSettings,fetched
|
0x37,WWSettings,fetched
|
||||||
0x38,WWTimer,fetched
|
0x38,WWTimer,fetched
|
||||||
0x39,WWCircTimer,fetched
|
0x39,WWCircTimer,fetched
|
||||||
0x3A,RC30WWSettings,fetched
|
0x3A,RC30WWSettings,fetched
|
||||||
0x3B,Energy,
|
0x3B,Energy,
|
||||||
0x3D,RC35Set,
|
0x3D,RC35Set,
|
||||||
0x3E,RC35Monitor,
|
0x3E,RC35Monitor,
|
||||||
0x3F,RC35Timer,
|
0x3F,RC35Timer,
|
||||||
0x40,RC30Temp,
|
0x40,RC30Temp,
|
||||||
0x41,RC30Monitor,
|
0x41,RC30Monitor,
|
||||||
0x42,RC35Timer2,
|
0x42,RC35Timer2,
|
||||||
0x47,RC35Set,
|
0x47,RC35Set,
|
||||||
0x48,RC35Monitor,
|
0x48,RC35Monitor,
|
||||||
0x49,RC35Timer,
|
0x49,RC35Timer,
|
||||||
0x4C,RC35Timer2,
|
0x4C,RC35Timer2,
|
||||||
0x51,RC35Set,
|
0x51,RC35Set,
|
||||||
0x52,RC35Monitor,
|
0x52,RC35Monitor,
|
||||||
0x53,RC35Timer,
|
0x53,RC35Timer,
|
||||||
0x56,RC35Timer2,
|
0x56,RC35Timer2,
|
||||||
0x5B,RC35Set,
|
0x5B,RC35Set,
|
||||||
0x5C,RC35Monitor,
|
0x5C,RC35Monitor,
|
||||||
0x5D,RC35Timer,
|
0x5D,RC35Timer,
|
||||||
0x60,RC35Timer2,
|
0x60,RC35Timer2,
|
||||||
0x96,SM10Config,fetched
|
0x96,SM10Config,fetched
|
||||||
0x97,SM10Monitor,
|
0x97,SM10Monitor,
|
||||||
0x9C,WM10MonitorMessage,
|
0x9C,WM10MonitorMessage,
|
||||||
0x9D,WM10SetMessage,
|
0x9D,WM10SetMessage,
|
||||||
0xA2,RCError,
|
0xA2,RCError,
|
||||||
0xA3,RCOutdoorTemp,
|
0xA3,RCOutdoorTemp,
|
||||||
0xA5,IBASettings,fetched
|
0xA5,IBASettings,fetched
|
||||||
0xA7,RC30Set,
|
0xA7,RC30Set,
|
||||||
0xA9,RC30Vacation,fetched
|
0xA9,RC30Vacation,fetched
|
||||||
0xAA,MMConfigMessage,fetched
|
0xAA,MMConfigMessage,fetched
|
||||||
0xAB,MMStatusMessage,
|
0xAB,MMStatusMessage,
|
||||||
0xAC,MMSetMessage,
|
0xAC,MMSetMessage,
|
||||||
0xAF,RC20Remote,
|
0xAF,RC20Remote,
|
||||||
0xB0,RC10Set,
|
0xB0,RC10Set,
|
||||||
0xB1,RC10Monitor,
|
0xB1,RC10Monitor,
|
||||||
0xBB,HybridSettings,fetched
|
0xBB,HybridSettings,fetched
|
||||||
0xBF,ErrorMessage,
|
0xBF,ErrorMessage,
|
||||||
0xC0,RCErrorMessage,
|
0xC2,UBAErrorMessage3,
|
||||||
0xC2,UBAErrorMessage3,
|
0xD1,UBAOutdoorTemp,
|
||||||
0xC6,UBAErrorMessage3,
|
0xE3,UBAMonitorSlowPlus2,
|
||||||
0xD1,UBAOutdoorTemp,
|
0xE4,UBAMonitorFastPlus,
|
||||||
0xE3,UBAMonitorSlowPlus2,
|
0xE5,UBAMonitorSlowPlus,
|
||||||
0xE4,UBAMonitorFastPlus,
|
0xE6,UBAParametersPlus,fetched
|
||||||
0xE5,UBAMonitorSlowPlus,
|
0xE9,UBAMonitorWWPlus,
|
||||||
0xE6,UBAParametersPlus,fetched
|
0xEA,UBAParameterWWPlus,fetched
|
||||||
0xE9,UBAMonitorWWPlus,
|
0x0101,ISM1Set,fetched
|
||||||
0xEA,UBAParameterWWPlus,fetched
|
0x0103,ISM1StatusMessage,fetched
|
||||||
0xEB,PumpKick,fetched
|
0x0104,ISM2StatusMessage,
|
||||||
0x0101,ISM1Set,fetched
|
0x010C,IPMStatusMessage,
|
||||||
0x0103,ISM1StatusMessage,fetched
|
0x011E,IPMTempMessage,
|
||||||
0x0104,ISM2StatusMessage,
|
0x012E,HPEnergy1,
|
||||||
0x010C,IPMStatusMessage,
|
0x013B,HPEnergy2,
|
||||||
0x011E,IPMTempMessage,
|
0x0165,JunkersSet,
|
||||||
0x012E,HPEnergy1,
|
0x0166,JunkersSet,
|
||||||
0x013B,HPEnergy2,
|
0x0167,JunkersSet,
|
||||||
0x0165,JunkersSet,
|
0x0168,JunkersSet,
|
||||||
0x0166,JunkersSet,
|
0x016E,Absent,fetched
|
||||||
0x0167,JunkersSet,
|
0x016F,JunkersMonitor,
|
||||||
0x0168,JunkersSet,
|
0x0170,JunkersMonitor,
|
||||||
0x016E,Absent,fetched
|
0x0171,JunkersMonitor,
|
||||||
0x016F,JunkersMonitor,
|
0x0172,JunkersMonitor,
|
||||||
0x0170,JunkersMonitor,
|
0x0179,JunkersSet,
|
||||||
0x0171,JunkersMonitor,
|
0x017A,JunkersSet,
|
||||||
0x0172,JunkersMonitor,
|
0x017B,JunkersSet,
|
||||||
0x0179,JunkersSet,
|
0x017C,JunkersSet,
|
||||||
0x017A,JunkersSet,
|
0x01D3,JunkersDhw,fetched
|
||||||
0x017B,JunkersSet,
|
0x023A,RC300OutdoorTemp,fetched
|
||||||
0x017C,JunkersSet,
|
0x023E,PVSettings,fetched
|
||||||
0x01D3,JunkersDhw,fetched
|
0x0240,RC300Settings,fetched
|
||||||
0x023A,RC300OutdoorTemp,fetched
|
0x0241,RC300Settings,fetched
|
||||||
0x023E,PVSettings,fetched
|
0x0267,RC300Floordry,
|
||||||
0x0240,RC300Settings,fetched
|
0x0269,RC300Holiday,fetched
|
||||||
0x0241,RC300Settings,fetched
|
0x0291,HPMode,fetched
|
||||||
0x0267,RC300Floordry,
|
0x0292,HPMode,fetched
|
||||||
0x0269,RC300Holiday,fetched
|
0x0293,HPMode,fetched
|
||||||
0x0291,HPMode,fetched
|
0x0294,HPMode,fetched
|
||||||
0x0292,HPMode,fetched
|
0x029B,RC300Curves,
|
||||||
0x0293,HPMode,fetched
|
0x029C,RC300Curves,
|
||||||
0x0294,HPMode,fetched
|
0x029D,RC300Curves,
|
||||||
0x029B,RC300Curves,
|
0x029E,RC300Curves,
|
||||||
0x029C,RC300Curves,
|
0x029F,RC300Curves,
|
||||||
0x029D,RC300Curves,
|
0x02A0,RC300Curves,
|
||||||
0x029E,RC300Curves,
|
0x02A1,RC300Curves,
|
||||||
0x029F,RC300Curves,
|
0x02A2,RC300Curves,
|
||||||
0x02A0,RC300Curves,
|
0x02A5,RC300Monitor,fetched
|
||||||
0x02A1,RC300Curves,
|
0x02A6,RC300Monitor,
|
||||||
0x02A2,RC300Curves,
|
0x02A7,CRFMonitor,
|
||||||
0x02A5,RC300Monitor,fetched
|
0x02A8,RC300Monitor,
|
||||||
0x02A6,CRFMonitor,
|
0x02A9,RC300Monitor,
|
||||||
0x02A7,RC300Monitor,
|
0x02AA,RC300Monitor,
|
||||||
0x02A8,CRFMonitor,
|
0x02AB,RC300Monitor,
|
||||||
0x02A9,RC300Monitor,
|
0x02AC,RC300Monitor,
|
||||||
0x02AA,RC300Monitor,
|
0x02AF,RC300Summer,
|
||||||
0x02AB,RC300Monitor,
|
0x02B0,RC300Summer,
|
||||||
0x02AC,RC300Monitor,
|
0x02B1,RC300Summer,
|
||||||
0x02AF,RC300Summer,
|
0x02B2,RC300Summer,
|
||||||
0x02B0,RC300Summer,
|
0x02B3,RC300Summer,
|
||||||
0x02B1,RC300Summer,
|
0x02B4,RC300Summer,
|
||||||
0x02B2,RC300Summer,
|
0x02B5,RC300Summer,
|
||||||
0x02B3,RC300Summer,
|
0x02B6,RC300Summer,
|
||||||
0x02B4,RC300Summer,
|
0x02B9,RC300Set,
|
||||||
0x02B5,RC300Summer,
|
0x02BA,RC300Set,
|
||||||
0x02B6,RC300Summer,
|
0x02BB,RC300Set,
|
||||||
0x02B9,RC300Set,
|
0x02BC,RC300Set,
|
||||||
0x02BA,RC300Set,
|
0x02BD,RC300Set,
|
||||||
0x02BB,RC300Set,
|
0x02BE,RC300Set,
|
||||||
0x02BC,RC300Set,
|
0x02BF,RC300Set,
|
||||||
0x02BD,RC300Set,
|
0x02C0,RC300Set,
|
||||||
0x02BE,RC300Set,
|
0x02CC,HPPressure,fetched
|
||||||
0x02BF,RC300Set,
|
0x02CD,MMPLUSConfigMessage,fetched
|
||||||
0x02C0,RC300Set,
|
0x02CE,RC300Set2,
|
||||||
0x02CC,HPPressure,fetched
|
0x02D0,RC300Set2,
|
||||||
0x02CD,MMPLUSConfigMessage,
|
0x02D2,RC300Set2,
|
||||||
0x02D6,HPPump2,fetched
|
0x02D6,HPPump2,fetched
|
||||||
0x02D7,MMPLUSStatusMessage,
|
0x02D7,MMPLUSStatusMessage,
|
||||||
0x02E0,UBASetPoints,
|
0x02F5,RC300WWmode,fetched
|
||||||
0x02F5,RC300WWmode,fetched
|
0x02F6,RC300WW2mode,fetched
|
||||||
0x02F6,RC300WW2mode,fetched
|
0x0313,MMPLUSConfigMessage_WWC,fetched
|
||||||
0x0313,MMPLUSConfigMessage_WWC,fetched
|
0x031B,RC300WWtemp,fetched
|
||||||
0x031B,RC300WWtemp,fetched
|
0x031D,RC300WWmode2,
|
||||||
0x031D,RC300WWmode2,
|
0x031E,RC300WWmode2,
|
||||||
0x031E,RC300WWmode2,
|
0x0331,MMPLUSStatusMessage_WWC,
|
||||||
0x0331,MMPLUSStatusMessage_WWC,
|
0x0358,SM100SystemConfig,fetched
|
||||||
0x0358,SM100SystemConfig,fetched
|
0x035A,SM100CircuitConfig,fetched
|
||||||
0x035A,SM100CircuitConfig,fetched
|
0x035C,SM100HeatAssist,fetched
|
||||||
0x035C,SM100HeatAssist,fetched
|
0x035D,SM100Circuit2Config,fetched
|
||||||
0x035D,SM100Circuit2Config,fetched
|
0x035F,SM100Config1,fetched
|
||||||
0x035F,SM100Config1,fetched
|
0x0361,SM100Differential,fetched
|
||||||
0x0361,SM100Differential,fetched
|
0x0362,SM100Monitor,
|
||||||
0x0362,SM100Monitor,
|
0x0363,SM100Monitor2,
|
||||||
0x0363,SM100Monitor2,
|
0x0364,SM100Status,
|
||||||
0x0364,SM100Status,
|
0x0366,SM100Config,
|
||||||
0x0366,SM100Config,
|
0x036A,SM100Status2,
|
||||||
0x036A,SM100Status2,
|
0x0380,SM100CollectorConfig,fetched
|
||||||
0x0380,SM100CollectorConfig,fetched
|
0x038E,SM100Energy,fetched
|
||||||
0x038E,SM100Energy,fetched
|
0x0391,SM100Time,fetched
|
||||||
0x0391,SM100Time,fetched
|
0x043F,CRHolidays,fetched
|
||||||
0x0421,RC300Set2,
|
0x0467,HPSet,
|
||||||
0x0422,RC300Set2,
|
0x0468,HPSet,
|
||||||
0x0423,RC300Set2,
|
0x0469,HPSet,
|
||||||
0x0424,RC300Set2,
|
0x046A,HPSet,
|
||||||
0x043F,CRHolidays,fetched
|
0x0471,RC300Summer2,
|
||||||
0x0467,HPSet,
|
0x0472,RC300Summer2,
|
||||||
0x0468,HPSet,
|
0x0473,RC300Summer2,
|
||||||
0x0469,HPSet,
|
0x0474,RC300Summer2,
|
||||||
0x046A,HPSet,
|
0x0475,RC300Summer2,
|
||||||
0x0471,RC300Summer2,
|
0x0476,RC300Summer2,
|
||||||
0x0472,RC300Summer2,
|
0x0477,RC300Summer2,
|
||||||
0x0473,RC300Summer2,
|
0x0478,RC300Summer2,
|
||||||
0x0474,RC300Summer2,
|
0x047B,HP2,
|
||||||
0x0475,RC300Summer2,
|
0x0484,HPSilentMode,fetched
|
||||||
0x0476,RC300Summer2,
|
0x0485,HpCooling,fetched
|
||||||
0x0477,RC300Summer2,
|
0x0486,HpInConfig,fetched
|
||||||
0x0478,RC300Summer2,
|
0x0488,HPValve,fetched
|
||||||
0x047B,HP2,
|
0x048A,HpPool,fetched
|
||||||
0x0484,HPSilentMode,fetched
|
0x048B,HPPumps,fetched
|
||||||
0x0485,HpCooling,fetched
|
0x048D,HpPower,fetched
|
||||||
0x0486,HpInConfig,fetched
|
0x048F,HpTemperatures,
|
||||||
0x0488,HPValve,fetched
|
0x0491,HPAdditionalHeater,fetched
|
||||||
0x048A,HpPool,fetched
|
0x0492,HpHeaterConfig,fetched
|
||||||
0x048B,HPPumps,fetched
|
0x0494,UBAEnergySupplied,
|
||||||
0x048D,HpPower,fetched
|
0x0495,UBAInformation,
|
||||||
0x048F,HpTemperatures,
|
0x0499,HPDhwSettings,fetched
|
||||||
0x0491,HPAdditionalHeater,fetched
|
0x049C,HPSettings2,fetched
|
||||||
0x0492,HpHeaterConfig,fetched
|
0x049D,HPSettings3,fetched
|
||||||
0x0494,UBAEnergySupplied,
|
0x04A2,HpInput,fetched
|
||||||
0x0495,UBAInformation,
|
0x04A5,HPFan,fetched
|
||||||
0x0499,HPDhwSettings,fetched
|
0x04A7,HPPowerLimit,fetched
|
||||||
0x049C,HPSettings2,fetched
|
0x04AA,HPPower2,fetched
|
||||||
0x049D,HPSettings3,fetched
|
0x04AE,HPEnergy,fetched
|
||||||
0x04A2,HpInput,fetched
|
0x04AF,HPMeters,fetched
|
||||||
0x04A5,HPFan,fetched
|
0x056B,VentilationMode,fetched
|
||||||
0x04A7,HPPowerLimit,fetched
|
0x0583,VentilationMonitor,
|
||||||
0x04AA,HPPower,
|
0x0585,Blowerspeed,
|
||||||
0x04AE,HPEnergy,fetched
|
0x0587,Bypass,
|
||||||
0x04AF,HPMeters,fetched
|
0x05BA,HpPoolStatus,fetched
|
||||||
0x055C,VentilationSet,fetched
|
0x05D9,Airquality,
|
||||||
0x056B,VentilationMode,fetched
|
0x0772,HIUSettings,
|
||||||
0x0583,VentilationMonitor,
|
0x0779,HIUMonitor,
|
||||||
0x0585,Blowerspeed,
|
0x07A5,SM100wwCirc,fetched
|
||||||
0x0587,Bypass,
|
0x07A6,SM100wwParam,fetched
|
||||||
0x05BA,HpPoolStatus,fetched
|
0x07AA,SM100wwStatus,
|
||||||
0x05D9,Airquality,
|
0x07AB,SM100wwCommand,
|
||||||
0x0772,HIUSettings,
|
0x07AC,SM100wwParam1,
|
||||||
0x0779,HIUMonitor,
|
0x07AD,SM100ValveStatus,
|
||||||
0x07A5,SM100wwCirc,fetched
|
0x07AE,SM100wwKeepWarm,fetched
|
||||||
0x07A6,SM100wwParam,fetched
|
0x07D6,SM100wwTemperature,
|
||||||
0x07AA,SM100wwStatus,
|
0x07E0,SM100wwStatus2,fetched
|
||||||
0x07AB,SM100wwCommand,
|
0x0935,EM100SetMessage,fetched
|
||||||
0x07AC,SM100wwParam1,
|
0x0936,EM100OutMessage,
|
||||||
0x07AD,SM100ValveStatus,
|
0x0937,EM100TempMessage,
|
||||||
0x07AE,SM100wwKeepWarm,fetched
|
0x0938,EM100InputMessage,
|
||||||
0x07D6,SM100wwTemperature,
|
0x0939,EM100MonitorMessage,
|
||||||
0x07E0,SM100wwStatus2,fetched
|
0x093A,EM100ConfigMessage,
|
||||||
0x0935,EM100SetMessage,fetched
|
0x0998,HPSettings,fetched
|
||||||
0x0936,EM100OutMessage,
|
0x0999,HPFunctionTest,fetched
|
||||||
0x0937,EM100TempMessage,
|
0x099A,HPStarts,
|
||||||
0x0938,EM100InputMessage,
|
0x099B,HPFlowTemp,
|
||||||
0x0939,EM100MonitorMessage,
|
0x099C,HPComp,
|
||||||
0x093A,EM100ConfigMessage,
|
0x09A0,HPTemperature,
|
||||||
0x0998,HPSettings,fetched
|
|
||||||
0x0999,HPFunctionTest,fetched
|
|
||||||
0x099A,HPStarts,
|
|
||||||
0x099B,HPFlowTemp,
|
|
||||||
0x099C,HPComp,
|
|
||||||
0x09A0,HPTemperature,
|
|
||||||
|
|||||||
|
4
interface/.gitattributes
vendored
Normal file
4
interface/.gitattributes
vendored
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
/.yarn/** linguist-vendored
|
||||||
|
/.yarn/releases/* binary
|
||||||
|
/.yarn/plugins/**/* binary
|
||||||
|
/.pnp.* binary linguist-generated
|
||||||
@@ -4,4 +4,5 @@ dist/
|
|||||||
src/i18n/*
|
src/i18n/*
|
||||||
|
|
||||||
.prettierrc
|
.prettierrc
|
||||||
.typesafe-i18n.json
|
.yarn/
|
||||||
|
.typesafe-i18n.json
|
||||||
@@ -1,5 +1,5 @@
|
|||||||
{
|
{
|
||||||
"adapter": "react",
|
"adapter": "react",
|
||||||
"baseLocale": "pl",
|
"baseLocale": "pl",
|
||||||
"$schema": "https://unpkg.com/typesafe-i18n@5.27.1/schema/typesafe-i18n.json"
|
"$schema": "https://unpkg.com/typesafe-i18n@5.26.2/schema/typesafe-i18n.json"
|
||||||
}
|
}
|
||||||
935
interface/.yarn/releases/yarn-4.7.0.cjs
vendored
Executable file
935
interface/.yarn/releases/yarn-4.7.0.cjs
vendored
Executable file
File diff suppressed because one or more lines are too long
3
interface/.yarnrc.yml
Normal file
3
interface/.yarnrc.yml
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
nodeLinker: node-modules
|
||||||
|
|
||||||
|
yarnPath: .yarn/releases/yarn-4.7.0.cjs
|
||||||
@@ -1,17 +1,17 @@
|
|||||||
// @ts-check
|
// @ts-check
|
||||||
import eslint from '@eslint/js';
|
import eslint from '@eslint/js';
|
||||||
import prettierConfig from 'eslint-config-prettier';
|
import prettierConfig from 'eslint-config-prettier';
|
||||||
import { defineConfig } from 'eslint/config';
|
|
||||||
import tseslint from 'typescript-eslint';
|
import tseslint from 'typescript-eslint';
|
||||||
|
|
||||||
export default defineConfig(
|
export default tseslint.config(
|
||||||
eslint.configs.recommended,
|
eslint.configs.recommended,
|
||||||
...tseslint.configs.recommendedTypeChecked,
|
...tseslint.configs.recommendedTypeChecked,
|
||||||
prettierConfig,
|
prettierConfig,
|
||||||
{
|
{
|
||||||
languageOptions: {
|
languageOptions: {
|
||||||
parserOptions: {
|
parserOptions: {
|
||||||
project: true
|
project: true,
|
||||||
|
tsconfigRootDir: import.meta.dirname
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "EMS-ESP",
|
"name": "EMS-ESP",
|
||||||
"version": "3.8.0",
|
"version": "3.7.2",
|
||||||
"description": "EMS-ESP WebUI",
|
"description": "EMS-ESP WebUI",
|
||||||
"homepage": "https://emsesp.org",
|
"homepage": "https://emsesp.org",
|
||||||
"author": "proddy, emsesp.org",
|
"author": "proddy, emsesp.org",
|
||||||
@@ -8,63 +8,59 @@
|
|||||||
"private": true,
|
"private": true,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"preinstall": "npx only-allow pnpm",
|
|
||||||
"dev": "vite dev",
|
"dev": "vite dev",
|
||||||
"build": "vite build",
|
"build": "vite build",
|
||||||
"preview": "vite preview",
|
"preview": "vite preview",
|
||||||
"build-hosted": "typesafe-i18n --no-watch && vite build --mode hosted",
|
"build-hosted": "typesafe-i18n --no-watch && vite build --mode hosted",
|
||||||
"mock-rest": "bun --watch ../mock-api/restServer.ts",
|
"preview-standalone": "typesafe-i18n --no-watch && vite build && concurrently -c \"auto\" \"yarn:mock-rest\" \"vite preview\"",
|
||||||
"preview-standalone": "typesafe-i18n --no-watch && vite build && concurrently -c \"auto\" \"pnpm:mock-rest\" \"vite preview\"",
|
"mock-rest": "bun --watch ../mock-api/rest_server.ts",
|
||||||
"standalone": "concurrently -c \"auto\" \"typesafe-i18n\" \"pnpm:mock-rest\" \"vite dev\"",
|
"standalone": "concurrently -c \"auto\" \"typesafe-i18n\" \"yarn:mock-rest\" \"vite\"",
|
||||||
"typesafe-i18n": "typesafe-i18n --no-watch",
|
"typesafe-i18n": "typesafe-i18n --no-watch",
|
||||||
"build_webUI": "typesafe-i18n --no-watch && vite build && node progmem-generator.js",
|
"webUI": "node progmem-generator.js",
|
||||||
"format": "prettier -l -w '**/*.{ts,tsx,js,css,json,md}'",
|
"format": "prettier -l -w '**/*.{ts,tsx,js,css,json,md}'",
|
||||||
"lint": "eslint . --fix",
|
"lint": "eslint . --fix"
|
||||||
"standalone-devcontainer": "concurrently -c \"auto\" \"typesafe-i18n\" \"pnpm:mock-rest\" \"vite --host\""
|
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@alova/adapter-xhr": "2.3.1",
|
"@alova/adapter-xhr": "2.1.1",
|
||||||
"@emotion/react": "^11.14.0",
|
"@emotion/react": "^11.14.0",
|
||||||
"@emotion/styled": "^11.14.1",
|
"@emotion/styled": "^11.14.0",
|
||||||
"@mui/icons-material": "^7.3.9",
|
"@mui/icons-material": "^6.4.8",
|
||||||
"@mui/material": "^7.3.9",
|
"@mui/material": "^6.4.8",
|
||||||
"@preact/compat": "^18.3.2",
|
"@table-library/react-table-library": "4.1.12",
|
||||||
"@table-library/react-table-library": "4.1.15",
|
"alova": "3.2.10",
|
||||||
"alova": "^3.5.1",
|
|
||||||
"async-validator": "^4.2.5",
|
"async-validator": "^4.2.5",
|
||||||
"etag": "^1.8.1",
|
|
||||||
"formidable": "^3.5.4",
|
|
||||||
"jwt-decode": "^4.0.0",
|
"jwt-decode": "^4.0.0",
|
||||||
"magic-string": "^0.30.21",
|
"mime-types": "^2.1.35",
|
||||||
"mime-types": "^3.0.2",
|
"preact": "^10.26.4",
|
||||||
"preact": "^10.29.0",
|
"react": "^19.0.0",
|
||||||
"react": "^19.2.4",
|
"react-dom": "^19.0.0",
|
||||||
"react-dom": "^19.2.4",
|
"react-icons": "^5.5.0",
|
||||||
"react-icons": "^5.6.0",
|
"react-router": "^7.4.0",
|
||||||
"react-router": "^7.13.1",
|
|
||||||
"react-toastify": "^11.0.5",
|
"react-toastify": "^11.0.5",
|
||||||
"typesafe-i18n": "^5.27.1",
|
"typesafe-i18n": "^5.26.2",
|
||||||
"typescript": "^5.9.3"
|
"typescript": "^5.8.2"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "^7.29.0",
|
"@babel/core": "^7.26.10",
|
||||||
"@eslint/js": "^10.0.1",
|
"@eslint/js": "^9.23.0",
|
||||||
"@preact/compat": "^18.3.2",
|
"@preact/compat": "^18.3.1",
|
||||||
"@preact/preset-vite": "^2.10.5",
|
"@preact/preset-vite": "^2.10.1",
|
||||||
"@trivago/prettier-plugin-sort-imports": "^6.0.2",
|
"@trivago/prettier-plugin-sort-imports": "^5.2.2",
|
||||||
"@types/node": "^25.5.0",
|
"@types/formidable": "^3",
|
||||||
"@types/react": "^19.2.14",
|
"@types/node": "^22.13.11",
|
||||||
"@types/react-dom": "^19.2.3",
|
"@types/react": "^19.0.12",
|
||||||
"axe-core": "^4.11.1",
|
"@types/react-dom": "^19.0.4",
|
||||||
"concurrently": "^9.2.1",
|
"concurrently": "^9.1.2",
|
||||||
"eslint": "^10.1.0",
|
"eslint": "^9.23.0",
|
||||||
"eslint-config-prettier": "^10.1.8",
|
"eslint-config-prettier": "^10.1.1",
|
||||||
"prettier": "^3.8.1",
|
"formidable": "^3.5.2",
|
||||||
"rollup-plugin-visualizer": "^7.0.1",
|
"prettier": "^3.5.3",
|
||||||
"terser": "^5.46.1",
|
"rollup-plugin-visualizer": "^5.14.0",
|
||||||
"typescript-eslint": "^8.57.1",
|
"terser": "^5.39.0",
|
||||||
"vite": "^8.0.1",
|
"typescript-eslint": "8.27.0",
|
||||||
"vite-plugin-imagemin": "^0.6.1"
|
"vite": "^6.2.2",
|
||||||
|
"vite-plugin-imagemin": "^0.6.1",
|
||||||
|
"vite-tsconfig-paths": "^5.1.4"
|
||||||
},
|
},
|
||||||
"packageManager": "pnpm@10.32.1+sha512.a706938f0e89ac1456b6563eab4edf1d1faf3368d1191fc5c59790e96dc918e4456ab2e67d613de1043d2e8c81f87303e6b40d4ffeca9df15ef1ad567348f2be"
|
"packageManager": "yarn@4.7.0"
|
||||||
}
|
}
|
||||||
|
|||||||
6569
interface/pnpm-lock.yaml
generated
6569
interface/pnpm-lock.yaml
generated
File diff suppressed because it is too large
Load Diff
@@ -1,8 +0,0 @@
|
|||||||
onlyBuiltDependencies:
|
|
||||||
- cwebp-bin
|
|
||||||
- esbuild
|
|
||||||
- gifsicle
|
|
||||||
- jpegtran-bin
|
|
||||||
- mozjpeg
|
|
||||||
- optipng-bin
|
|
||||||
- pngquant-bin
|
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import etag from 'etag';
|
import crypto from 'crypto';
|
||||||
import {
|
import {
|
||||||
createWriteStream,
|
createWriteStream,
|
||||||
existsSync,
|
existsSync,
|
||||||
@@ -15,79 +15,66 @@ const INDENT = ' ';
|
|||||||
const outputPath = '../src/ESP32React/WWWData.h';
|
const outputPath = '../src/ESP32React/WWWData.h';
|
||||||
const sourcePath = './dist';
|
const sourcePath = './dist';
|
||||||
const bytesPerLine = 20;
|
const bytesPerLine = 20;
|
||||||
let totalSize = 0;
|
var totalSize = 0;
|
||||||
let bundleStats = {
|
|
||||||
js: { count: 0, uncompressed: 0, compressed: 0 },
|
|
||||||
css: { count: 0, uncompressed: 0, compressed: 0 },
|
|
||||||
html: { count: 0, uncompressed: 0, compressed: 0 },
|
|
||||||
svg: { count: 0, uncompressed: 0, compressed: 0 },
|
|
||||||
other: { count: 0, uncompressed: 0, compressed: 0 }
|
|
||||||
};
|
|
||||||
|
|
||||||
const generateWWWClass =
|
const generateWWWClass = () =>
|
||||||
() => `typedef std::function<void(const char * uri, const String & contentType, const uint8_t * content, size_t len, const String & hash)> RouteRegistrationHandler;
|
`typedef std::function<void(const char * uri, const String & contentType, const uint8_t * content, size_t len, const String & hash)> RouteRegistrationHandler;
|
||||||
// Bundle Statistics:
|
// Total size is ${totalSize} bytes
|
||||||
// - Total compressed size: ${(totalSize / 1000).toFixed(1)} KB
|
|
||||||
// - Total uncompressed size: ${(Object.values(bundleStats).reduce((sum, stat) => sum + stat.uncompressed, 0) / 1000).toFixed(1)} KB
|
|
||||||
// - Compression ratio: ${(((Object.values(bundleStats).reduce((sum, stat) => sum + stat.uncompressed, 0) - totalSize) / Object.values(bundleStats).reduce((sum, stat) => sum + stat.uncompressed, 0)) * 100).toFixed(1)}%
|
|
||||||
// - Generated on: ${new Date().toISOString()}
|
|
||||||
|
|
||||||
class WWWData {
|
class WWWData {
|
||||||
${INDENT}public:
|
${indent}public:
|
||||||
${INDENT.repeat(2)}static void registerRoutes(RouteRegistrationHandler handler) {
|
${indent.repeat(2)}static void registerRoutes(RouteRegistrationHandler handler) {
|
||||||
${fileInfo.map((f) => `${INDENT.repeat(3)}handler("${f.uri}", "${f.mimeType}", ${f.variable}, ${f.size}, ${f.hash});`).join('\n')}
|
${fileInfo.map((file) => `${indent.repeat(3)}handler("${file.uri}", "${file.mimeType}", ${file.variable}, ${file.size}, "${file.hash}");`).join('\n')}
|
||||||
${INDENT.repeat(2)}}
|
${indent.repeat(2)}}
|
||||||
};
|
};
|
||||||
`;
|
`;
|
||||||
|
|
||||||
const getFilesSync = (dir, files = []) => {
|
function getFilesSync(dir, files = []) {
|
||||||
readdirSync(dir, { withFileTypes: true }).forEach((entry) => {
|
readdirSync(dir, { withFileTypes: true }).forEach((entry) => {
|
||||||
const entryPath = resolve(dir, entry.name);
|
const entryPath = resolve(dir, entry.name);
|
||||||
entry.isDirectory() ? getFilesSync(entryPath, files) : files.push(entryPath);
|
if (entry.isDirectory()) {
|
||||||
|
getFilesSync(entryPath, files);
|
||||||
|
} else {
|
||||||
|
files.push(entryPath);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
return files;
|
return files;
|
||||||
};
|
}
|
||||||
|
|
||||||
const cleanAndOpen = (path) => {
|
function cleanAndOpen(path) {
|
||||||
existsSync(path) && unlinkSync(path);
|
if (existsSync(path)) {
|
||||||
|
unlinkSync(path);
|
||||||
|
}
|
||||||
return createWriteStream(path, { flags: 'w+' });
|
return createWriteStream(path, { flags: 'w+' });
|
||||||
};
|
}
|
||||||
|
|
||||||
const getFileType = (filePath) => {
|
|
||||||
const ext = filePath.split('.').pop().toLowerCase();
|
|
||||||
if (ext === 'js') return 'js';
|
|
||||||
if (ext === 'css') return 'css';
|
|
||||||
if (ext === 'html') return 'html';
|
|
||||||
if (ext === 'svg') return 'svg';
|
|
||||||
return 'other';
|
|
||||||
};
|
|
||||||
|
|
||||||
const writeFile = (relativeFilePath, buffer) => {
|
const writeFile = (relativeFilePath, buffer) => {
|
||||||
const variable = `ESP_REACT_DATA_${fileInfo.length}`;
|
const variable = 'ESP_REACT_DATA_' + fileInfo.length;
|
||||||
const mimeType = mime.lookup(relativeFilePath);
|
const mimeType = mime.lookup(relativeFilePath);
|
||||||
const fileType = getFileType(relativeFilePath);
|
var size = 0;
|
||||||
let size = 0;
|
writeStream.write('const uint8_t ' + variable + '[] = {');
|
||||||
writeStream.write(`const uint8_t ${variable}[] = {`);
|
// const zipBuffer = zlib.brotliCompressSync(buffer, { quality: 1 });
|
||||||
|
|
||||||
const zipBuffer = zlib.gzipSync(buffer, { level: 9 });
|
const zipBuffer = zlib.gzipSync(buffer, { level: 9 });
|
||||||
// const hash = crypto.createHash('sha256').update(zipBuffer).digest('hex');
|
|
||||||
const hash = etag(zipBuffer); // use smaller md5 instead of sha256
|
// create sha
|
||||||
|
const hashSum = crypto.createHash('sha256');
|
||||||
|
hashSum.update(zipBuffer);
|
||||||
|
const hash = hashSum.digest('hex');
|
||||||
|
|
||||||
zipBuffer.forEach((b) => {
|
zipBuffer.forEach((b) => {
|
||||||
if (!(size % bytesPerLine)) {
|
if (!(size % bytesPerLine)) {
|
||||||
writeStream.write('\n' + INDENT);
|
writeStream.write('\n');
|
||||||
|
writeStream.write(indent);
|
||||||
}
|
}
|
||||||
writeStream.write('0x' + b.toString(16).toUpperCase().padStart(2, '0') + ',');
|
writeStream.write('0x' + ('00' + b.toString(16).toUpperCase()).slice(-2) + ',');
|
||||||
size++;
|
size++;
|
||||||
});
|
});
|
||||||
|
|
||||||
size % bytesPerLine && writeStream.write('\n');
|
if (size % bytesPerLine) {
|
||||||
writeStream.write('};\n\n');
|
writeStream.write('\n');
|
||||||
|
}
|
||||||
|
|
||||||
// Update bundle statistics
|
writeStream.write('};\n\n');
|
||||||
bundleStats[fileType].count++;
|
|
||||||
bundleStats[fileType].uncompressed += buffer.length;
|
|
||||||
bundleStats[fileType].compressed += zipBuffer.length;
|
|
||||||
|
|
||||||
fileInfo.push({
|
fileInfo.push({
|
||||||
uri: '/' + relativeFilePath.replace(sep, '/'),
|
uri: '/' + relativeFilePath.replace(sep, '/'),
|
||||||
@@ -97,52 +84,32 @@ const writeFile = (relativeFilePath, buffer) => {
|
|||||||
hash
|
hash
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// console.log(relativeFilePath + ' (size ' + size + ' bytes)');
|
||||||
totalSize += size;
|
totalSize += size;
|
||||||
};
|
};
|
||||||
|
|
||||||
console.log(`Generating ${outputPath} from ${sourcePath}`);
|
// start
|
||||||
|
console.log('Generating ' + outputPath + ' from ' + sourcePath);
|
||||||
|
const includes = ARDUINO_INCLUDES;
|
||||||
|
const indent = INDENT;
|
||||||
const fileInfo = [];
|
const fileInfo = [];
|
||||||
const writeStream = cleanAndOpen(resolve(outputPath));
|
const writeStream = cleanAndOpen(resolve(outputPath));
|
||||||
|
|
||||||
writeStream.write(ARDUINO_INCLUDES);
|
// includes
|
||||||
|
writeStream.write(includes);
|
||||||
|
|
||||||
|
// process static files
|
||||||
const buildPath = resolve(sourcePath);
|
const buildPath = resolve(sourcePath);
|
||||||
for (const filePath of getFilesSync(buildPath)) {
|
for (const filePath of getFilesSync(buildPath)) {
|
||||||
writeFile(relative(buildPath, filePath), readFileSync(filePath));
|
const readStream = readFileSync(filePath);
|
||||||
|
const relativeFilePath = relative(buildPath, filePath);
|
||||||
|
writeFile(relativeFilePath, readStream);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// add class
|
||||||
writeStream.write(generateWWWClass());
|
writeStream.write(generateWWWClass());
|
||||||
|
|
||||||
|
// end
|
||||||
writeStream.end();
|
writeStream.end();
|
||||||
|
|
||||||
// Calculate and display bundle statistics
|
console.log('Total size: ' + totalSize / 1000 + ' KB');
|
||||||
const totalUncompressed = Object.values(bundleStats).reduce(
|
|
||||||
(sum, stat) => sum + stat.uncompressed,
|
|
||||||
0
|
|
||||||
);
|
|
||||||
const totalCompressed = Object.values(bundleStats).reduce(
|
|
||||||
(sum, stat) => sum + stat.compressed,
|
|
||||||
0
|
|
||||||
);
|
|
||||||
const compressionRatio = (
|
|
||||||
((totalUncompressed - totalCompressed) / totalUncompressed) *
|
|
||||||
100
|
|
||||||
).toFixed(1);
|
|
||||||
|
|
||||||
console.log('\n📊 Bundle Size Analysis:');
|
|
||||||
console.log('='.repeat(50));
|
|
||||||
console.log(`Total compressed size: ${(totalSize / 1000).toFixed(1)} KB`);
|
|
||||||
console.log(`Total uncompressed size: ${(totalUncompressed / 1000).toFixed(1)} KB`);
|
|
||||||
console.log(`Compression ratio: ${compressionRatio}%`);
|
|
||||||
console.log('\n📁 File Type Breakdown:');
|
|
||||||
Object.entries(bundleStats).forEach(([type, stats]) => {
|
|
||||||
if (stats.count > 0) {
|
|
||||||
const ratio = (
|
|
||||||
((stats.uncompressed - stats.compressed) / stats.uncompressed) *
|
|
||||||
100
|
|
||||||
).toFixed(1);
|
|
||||||
console.log(
|
|
||||||
`${type.toUpperCase().padEnd(4)}: ${stats.count} files, ${(stats.uncompressed / 1000).toFixed(1)} KB → ${(stats.compressed / 1000).toFixed(1)} KB (${ratio}% compression)`
|
|
||||||
);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
console.log('='.repeat(50));
|
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useCallback, useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
import { ToastContainer, Zoom } from 'react-toastify';
|
import { ToastContainer, Zoom } from 'react-toastify';
|
||||||
|
|
||||||
import AppRouting from 'AppRouting';
|
import AppRouting from 'AppRouting';
|
||||||
@@ -8,7 +8,7 @@ import type { Locales } from 'i18n/i18n-types';
|
|||||||
import { loadLocaleAsync } from 'i18n/i18n-util.async';
|
import { loadLocaleAsync } from 'i18n/i18n-util.async';
|
||||||
import { detectLocale, navigatorDetector } from 'typesafe-i18n/detectors';
|
import { detectLocale, navigatorDetector } from 'typesafe-i18n/detectors';
|
||||||
|
|
||||||
const AVAILABLE_LOCALES = [
|
const availableLocales = [
|
||||||
'de',
|
'de',
|
||||||
'en',
|
'en',
|
||||||
'it',
|
'it',
|
||||||
@@ -20,56 +20,47 @@ const AVAILABLE_LOCALES = [
|
|||||||
'sv',
|
'sv',
|
||||||
'tr',
|
'tr',
|
||||||
'cz'
|
'cz'
|
||||||
] as Locales[];
|
];
|
||||||
|
|
||||||
// Static toast configuration - no need to recreate on every render
|
const App = () => {
|
||||||
const TOAST_CONTAINER_PROPS = {
|
|
||||||
position: 'bottom-left' as const,
|
|
||||||
autoClose: 3000,
|
|
||||||
hideProgressBar: false,
|
|
||||||
newestOnTop: false,
|
|
||||||
closeOnClick: true,
|
|
||||||
rtl: false,
|
|
||||||
pauseOnFocusLoss: true,
|
|
||||||
draggable: false,
|
|
||||||
pauseOnHover: false,
|
|
||||||
transition: Zoom,
|
|
||||||
closeButton: false,
|
|
||||||
theme: 'dark' as const,
|
|
||||||
toastStyle: {
|
|
||||||
border: '1px solid #177ac9',
|
|
||||||
width: 'fit-content'
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const App = memo(() => {
|
|
||||||
const [wasLoaded, setWasLoaded] = useState(false);
|
const [wasLoaded, setWasLoaded] = useState(false);
|
||||||
const [locale, setLocale] = useState<Locales>('en');
|
const [locale, setLocale] = useState<Locales>('en');
|
||||||
|
|
||||||
// Memoize locale initialization to prevent unnecessary re-runs
|
useEffect(() => {
|
||||||
const initializeLocale = useCallback(async () => {
|
// determine locale, take from session if set other default to browser language
|
||||||
const browserLocale = detectLocale('en', AVAILABLE_LOCALES, navigatorDetector);
|
const browserLocale = detectLocale('en', availableLocales, navigatorDetector);
|
||||||
const newLocale = (localStorage.getItem('lang') || browserLocale) as Locales;
|
const newLocale = (localStorage.getItem('lang') || browserLocale) as Locales;
|
||||||
localStorage.setItem('lang', newLocale);
|
localStorage.setItem('lang', newLocale);
|
||||||
setLocale(newLocale);
|
setLocale(newLocale);
|
||||||
await loadLocaleAsync(newLocale);
|
void loadLocaleAsync(newLocale).then(() => setWasLoaded(true));
|
||||||
setWasLoaded(true);
|
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
void initializeLocale();
|
|
||||||
}, [initializeLocale]);
|
|
||||||
|
|
||||||
if (!wasLoaded) return null;
|
if (!wasLoaded) return null;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<TypesafeI18n locale={locale}>
|
<TypesafeI18n locale={locale}>
|
||||||
<CustomTheme>
|
<CustomTheme>
|
||||||
<AppRouting />
|
<AppRouting />
|
||||||
<ToastContainer {...TOAST_CONTAINER_PROPS} />
|
<ToastContainer
|
||||||
|
position="bottom-left"
|
||||||
|
autoClose={3000}
|
||||||
|
hideProgressBar={false}
|
||||||
|
newestOnTop={false}
|
||||||
|
closeOnClick
|
||||||
|
rtl={false}
|
||||||
|
pauseOnFocusLoss
|
||||||
|
draggable={false}
|
||||||
|
pauseOnHover={false}
|
||||||
|
transition={Zoom}
|
||||||
|
closeButton={false}
|
||||||
|
theme="dark"
|
||||||
|
toastStyle={{
|
||||||
|
border: '1px solid #177ac9'
|
||||||
|
}}
|
||||||
|
/>
|
||||||
</CustomTheme>
|
</CustomTheme>
|
||||||
</TypesafeI18n>
|
</TypesafeI18n>
|
||||||
);
|
);
|
||||||
});
|
};
|
||||||
|
|
||||||
export default App;
|
export default App;
|
||||||
|
|||||||
@@ -1,80 +1,60 @@
|
|||||||
import { type FC, Suspense, lazy, memo, useContext, useEffect, useRef } from 'react';
|
import { useContext, useEffect } from 'react';
|
||||||
import { Navigate, Route, Routes } from 'react-router';
|
import { Navigate, Route, Routes } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import {
|
import AuthenticatedRouting from 'AuthenticatedRouting';
|
||||||
LoadingSpinner,
|
import SignIn from 'SignIn';
|
||||||
RequireAuthenticated,
|
import { RequireAuthenticated, RequireUnauthenticated } from 'components';
|
||||||
RequireUnauthenticated
|
|
||||||
} from 'components';
|
|
||||||
import { Authentication, AuthenticationContext } from 'contexts/authentication';
|
import { Authentication, AuthenticationContext } from 'contexts/authentication';
|
||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
|
|
||||||
// Lazy load route components for better code splitting
|
|
||||||
const SignIn = lazy(() => import('SignIn'));
|
|
||||||
const AuthenticatedRouting = lazy(() => import('AuthenticatedRouting'));
|
|
||||||
|
|
||||||
interface SecurityRedirectProps {
|
interface SecurityRedirectProps {
|
||||||
readonly message: string;
|
message: string;
|
||||||
readonly signOut?: boolean;
|
signOut?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
const RootRedirect: FC<SecurityRedirectProps> = memo(
|
const RootRedirect = ({ message, signOut }: SecurityRedirectProps) => {
|
||||||
({ message, signOut = false }) => {
|
const authenticationContext = useContext(AuthenticationContext);
|
||||||
const { signOut: contextSignOut } = useContext(AuthenticationContext);
|
useEffect(() => {
|
||||||
const hasShownToast = useRef(false);
|
signOut && authenticationContext.signOut(false);
|
||||||
|
toast.success(message);
|
||||||
|
}, [message, signOut, authenticationContext]);
|
||||||
|
return <Navigate to="/" />;
|
||||||
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
const AppRouting = () => {
|
||||||
// Prevent duplicate toasts on strict mode or re-renders
|
|
||||||
if (!hasShownToast.current) {
|
|
||||||
hasShownToast.current = true;
|
|
||||||
if (signOut) {
|
|
||||||
contextSignOut(false);
|
|
||||||
}
|
|
||||||
toast.success(message);
|
|
||||||
}
|
|
||||||
// Only run once on mount - using ref to track execution
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return <Navigate to="/" replace />;
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
const AppRouting: FC = memo(() => {
|
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Authentication>
|
<Authentication>
|
||||||
<Suspense fallback={<LoadingSpinner />}>
|
<Routes>
|
||||||
<Routes>
|
<Route
|
||||||
<Route
|
path="/unauthorized"
|
||||||
path="/unauthorized"
|
element={<RootRedirect message={LL.PLEASE_SIGNIN()} signOut />}
|
||||||
element={<RootRedirect message={LL.PLEASE_SIGNIN()} signOut />}
|
/>
|
||||||
/>
|
<Route
|
||||||
<Route
|
path="/fileUpdated"
|
||||||
path="/fileUpdated"
|
element={<RootRedirect message={LL.UPLOAD_SUCCESSFUL()} />}
|
||||||
element={<RootRedirect message={LL.UPLOAD_SUCCESSFUL()} />}
|
/>
|
||||||
/>
|
<Route
|
||||||
<Route
|
path="/"
|
||||||
path="/"
|
element={
|
||||||
element={
|
<RequireUnauthenticated>
|
||||||
<RequireUnauthenticated>
|
<SignIn />
|
||||||
<SignIn />
|
</RequireUnauthenticated>
|
||||||
</RequireUnauthenticated>
|
}
|
||||||
}
|
/>
|
||||||
/>
|
<Route
|
||||||
<Route
|
path="/*"
|
||||||
path="/*"
|
element={
|
||||||
element={
|
<RequireAuthenticated>
|
||||||
<RequireAuthenticated>
|
<AuthenticatedRouting />
|
||||||
<AuthenticatedRouting />
|
</RequireAuthenticated>
|
||||||
</RequireAuthenticated>
|
}
|
||||||
}
|
/>
|
||||||
/>
|
</Routes>
|
||||||
</Routes>
|
|
||||||
</Suspense>
|
|
||||||
</Authentication>
|
</Authentication>
|
||||||
);
|
);
|
||||||
});
|
};
|
||||||
|
|
||||||
export default AppRouting;
|
export default AppRouting;
|
||||||
|
|||||||
@@ -1,88 +1,76 @@
|
|||||||
import { Suspense, lazy, memo, useContext } from 'react';
|
import { useContext } from 'react';
|
||||||
import { Navigate, Route, Routes } from 'react-router';
|
import { Navigate, Route, Routes } from 'react-router';
|
||||||
|
|
||||||
import { Layout, LoadingSpinner } from 'components';
|
import CustomEntities from 'app/main/CustomEntities';
|
||||||
|
import Customizations from 'app/main/Customizations';
|
||||||
|
import Dashboard from 'app/main/Dashboard';
|
||||||
|
import Devices from 'app/main/Devices';
|
||||||
|
import Help from 'app/main/Help';
|
||||||
|
import Modules from 'app/main/Modules';
|
||||||
|
import Scheduler from 'app/main/Scheduler';
|
||||||
|
import Sensors from 'app/main/Sensors';
|
||||||
|
import APSettings from 'app/settings/APSettings';
|
||||||
|
import ApplicationSettings from 'app/settings/ApplicationSettings';
|
||||||
|
import DownloadUpload from 'app/settings/DownloadUpload';
|
||||||
|
import MqttSettings from 'app/settings/MqttSettings';
|
||||||
|
import NTPSettings from 'app/settings/NTPSettings';
|
||||||
|
import Settings from 'app/settings/Settings';
|
||||||
|
import Network from 'app/settings/network/Network';
|
||||||
|
import Security from 'app/settings/security/Security';
|
||||||
|
import APStatus from 'app/status/APStatus';
|
||||||
|
import Activity from 'app/status/Activity';
|
||||||
|
import HardwareStatus from 'app/status/HardwareStatus';
|
||||||
|
import MqttStatus from 'app/status/MqttStatus';
|
||||||
|
import NTPStatus from 'app/status/NTPStatus';
|
||||||
|
import NetworkStatus from 'app/status/NetworkStatus';
|
||||||
|
import Status from 'app/status/Status';
|
||||||
|
import SystemLog from 'app/status/SystemLog';
|
||||||
|
import Version from 'app/status/Version';
|
||||||
|
import { Layout } from 'components';
|
||||||
import { AuthenticatedContext } from 'contexts/authentication';
|
import { AuthenticatedContext } from 'contexts/authentication';
|
||||||
|
|
||||||
// Lazy load all route components for better code splitting
|
const AuthenticatedRouting = () => {
|
||||||
const Dashboard = lazy(() => import('app/main/Dashboard'));
|
|
||||||
const Devices = lazy(() => import('app/main/Devices'));
|
|
||||||
const Sensors = lazy(() => import('app/main/Sensors'));
|
|
||||||
const Help = lazy(() => import('app/main/Help'));
|
|
||||||
const Customizations = lazy(() => import('app/main/Customizations'));
|
|
||||||
const Scheduler = lazy(() => import('app/main/Scheduler'));
|
|
||||||
const CustomEntities = lazy(() => import('app/main/CustomEntities'));
|
|
||||||
const Modules = lazy(() => import('app/main/Modules'));
|
|
||||||
const UserProfile = lazy(() => import('app/main/UserProfile'));
|
|
||||||
|
|
||||||
const Status = lazy(() => import('app/status/Status'));
|
|
||||||
const HardwareStatus = lazy(() => import('app/status/HardwareStatus'));
|
|
||||||
const Activity = lazy(() => import('app/status/Activity'));
|
|
||||||
const SystemLog = lazy(() => import('app/status/SystemLog'));
|
|
||||||
const MqttStatus = lazy(() => import('app/status/MqttStatus'));
|
|
||||||
const NTPStatus = lazy(() => import('app/status/NTPStatus'));
|
|
||||||
const APStatus = lazy(() => import('app/status/APStatus'));
|
|
||||||
const NetworkStatus = lazy(() => import('app/status/NetworkStatus'));
|
|
||||||
const Version = lazy(() => import('app/status/Version'));
|
|
||||||
|
|
||||||
const Settings = lazy(() => import('app/settings/Settings'));
|
|
||||||
const ApplicationSettings = lazy(() => import('app/settings/ApplicationSettings'));
|
|
||||||
const MqttSettings = lazy(() => import('app/settings/MqttSettings'));
|
|
||||||
const NTPSettings = lazy(() => import('app/settings/NTPSettings'));
|
|
||||||
const APSettings = lazy(() => import('app/settings/APSettings'));
|
|
||||||
const DownloadUpload = lazy(() => import('app/settings/DownloadUpload'));
|
|
||||||
const Network = lazy(() => import('app/settings/network/Network'));
|
|
||||||
const Security = lazy(() => import('app/settings/security/Security'));
|
|
||||||
|
|
||||||
const AuthenticatedRouting = memo(() => {
|
|
||||||
const { me } = useContext(AuthenticatedContext);
|
const { me } = useContext(AuthenticatedContext);
|
||||||
return (
|
return (
|
||||||
<Layout>
|
<Layout>
|
||||||
<Suspense fallback={<LoadingSpinner />}>
|
<Routes>
|
||||||
<Routes>
|
<Route path="/dashboard/*" element={<Dashboard />} />
|
||||||
<Route path="/dashboard/*" element={<Dashboard />} />
|
<Route path="/devices/*" element={<Devices />} />
|
||||||
<Route path="/devices/*" element={<Devices />} />
|
<Route path="/sensors/*" element={<Sensors />} />
|
||||||
<Route path="/sensors/*" element={<Sensors />} />
|
<Route path="/status/*" element={<Status />} />
|
||||||
<Route path="/help/*" element={<Help />} />
|
<Route path="/help/*" element={<Help />} />
|
||||||
<Route path="/user/*" element={<UserProfile />} />
|
<Route path="/*" element={<Navigate to="/" />} />
|
||||||
|
|
||||||
<Route path="/status/*" element={<Status />} />
|
<Route path="/status/hardwarestatus/*" element={<HardwareStatus />} />
|
||||||
<Route path="/status/hardwarestatus/*" element={<HardwareStatus />} />
|
<Route path="/status/activity" element={<Activity />} />
|
||||||
<Route path="/status/activity" element={<Activity />} />
|
<Route path="/status/log" element={<SystemLog />} />
|
||||||
<Route path="/status/log" element={<SystemLog />} />
|
<Route path="/status/mqtt" element={<MqttStatus />} />
|
||||||
<Route path="/status/mqtt" element={<MqttStatus />} />
|
<Route path="/status/ntp" element={<NTPStatus />} />
|
||||||
<Route path="/status/ntp" element={<NTPStatus />} />
|
<Route path="/status/ap" element={<APStatus />} />
|
||||||
<Route path="/status/ap" element={<APStatus />} />
|
<Route path="/status/network" element={<NetworkStatus />} />
|
||||||
<Route path="/status/network" element={<NetworkStatus />} />
|
<Route path="/status/version" element={<Version />} />
|
||||||
<Route path="/status/version" element={<Version />} />
|
|
||||||
|
|
||||||
{me.admin && (
|
{me.admin && (
|
||||||
<>
|
<>
|
||||||
<Route path="/settings" element={<Settings />} />
|
<Route path="/settings" element={<Settings />} />
|
||||||
<Route
|
<Route path="/settings/application" element={<ApplicationSettings />} />
|
||||||
path="/settings/application"
|
<Route path="/settings/mqtt" element={<MqttSettings />} />
|
||||||
element={<ApplicationSettings />}
|
<Route path="/settings/ntp" element={<NTPSettings />} />
|
||||||
/>
|
<Route path="/settings/ap" element={<APSettings />} />
|
||||||
<Route path="/settings/mqtt" element={<MqttSettings />} />
|
<Route path="/settings/modules" element={<Modules />} />
|
||||||
<Route path="/settings/ntp" element={<NTPSettings />} />
|
<Route path="/settings/downloadUpload" element={<DownloadUpload />} />
|
||||||
<Route path="/settings/ap" element={<APSettings />} />
|
|
||||||
<Route path="/settings/modules" element={<Modules />} />
|
|
||||||
<Route path="/settings/downloadUpload" element={<DownloadUpload />} />
|
|
||||||
|
|
||||||
<Route path="/settings/network/*" element={<Network />} />
|
<Route path="/settings/network/*" element={<Network />} />
|
||||||
<Route path="/settings/security/*" element={<Security />} />
|
<Route path="/settings/security/*" element={<Security />} />
|
||||||
|
|
||||||
<Route path="/customizations" element={<Customizations />} />
|
<Route path="/customizations" element={<Customizations />} />
|
||||||
<Route path="/scheduler" element={<Scheduler />} />
|
<Route path="/scheduler" element={<Scheduler />} />
|
||||||
<Route path="/customentities" element={<CustomEntities />} />
|
<Route path="/customentities" element={<CustomEntities />} />
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
</Routes>
|
||||||
<Route path="/*" element={<Navigate to="/" />} />
|
|
||||||
</Routes>
|
|
||||||
</Suspense>
|
|
||||||
</Layout>
|
</Layout>
|
||||||
);
|
);
|
||||||
});
|
};
|
||||||
|
|
||||||
export default AuthenticatedRouting;
|
export default AuthenticatedRouting;
|
||||||
|
|||||||
@@ -1,12 +1,6 @@
|
|||||||
import { memo } from 'react';
|
|
||||||
import type { FC } from 'react';
|
import type { FC } from 'react';
|
||||||
|
|
||||||
import {
|
import { CssBaseline, ThemeProvider, responsiveFontSizes } from '@mui/material';
|
||||||
CssBaseline,
|
|
||||||
ThemeProvider,
|
|
||||||
responsiveFontSizes,
|
|
||||||
tooltipClasses
|
|
||||||
} from '@mui/material';
|
|
||||||
import { createTheme } from '@mui/material/styles';
|
import { createTheme } from '@mui/material/styles';
|
||||||
|
|
||||||
import type { RequiredChildrenProps } from 'utils';
|
import type { RequiredChildrenProps } from 'utils';
|
||||||
@@ -16,9 +10,9 @@ export const dialogStyle = {
|
|||||||
borderRadius: '8px',
|
borderRadius: '8px',
|
||||||
borderColor: '#565656',
|
borderColor: '#565656',
|
||||||
borderStyle: 'solid',
|
borderStyle: 'solid',
|
||||||
borderWidth: '2px'
|
borderWidth: '1px'
|
||||||
}
|
}
|
||||||
} as const;
|
};
|
||||||
|
|
||||||
const theme = responsiveFontSizes(
|
const theme = responsiveFontSizes(
|
||||||
createTheme({
|
createTheme({
|
||||||
@@ -36,45 +30,15 @@ const theme = responsiveFontSizes(
|
|||||||
text: {
|
text: {
|
||||||
disabled: '#eee' // white
|
disabled: '#eee' // white
|
||||||
}
|
}
|
||||||
},
|
|
||||||
components: {
|
|
||||||
MuiListItemText: {
|
|
||||||
styleOverrides: {
|
|
||||||
primary: {
|
|
||||||
fontSize: 14
|
|
||||||
},
|
|
||||||
secondary: {
|
|
||||||
color: '#9e9e9e' // grey[500]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
MuiTooltip: {
|
|
||||||
defaultProps: {
|
|
||||||
placement: 'top',
|
|
||||||
arrow: true
|
|
||||||
},
|
|
||||||
styleOverrides: {
|
|
||||||
tooltip: {
|
|
||||||
padding: '4px 8px',
|
|
||||||
fontSize: 10,
|
|
||||||
color: 'rgba(0, 0, 0, 0.87)',
|
|
||||||
backgroundColor: '#4caf50', // MUI success.main default color
|
|
||||||
boxShadow: '0px 2px 8px rgba(0, 0, 0, 0.15)',
|
|
||||||
[`& .${tooltipClasses.arrow}`]: {
|
|
||||||
color: '#4caf50'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
const CustomTheme: FC<RequiredChildrenProps> = memo(({ children }) => (
|
const CustomTheme: FC<RequiredChildrenProps> = ({ children }) => (
|
||||||
<ThemeProvider theme={theme}>
|
<ThemeProvider theme={theme}>
|
||||||
<CssBaseline />
|
<CssBaseline />
|
||||||
{children}
|
{children}
|
||||||
</ThemeProvider>
|
</ThemeProvider>
|
||||||
));
|
);
|
||||||
|
|
||||||
export default CustomTheme;
|
export default CustomTheme;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useCallback, useContext, useEffect, useMemo, useState } from 'react';
|
import { useContext, useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import ForwardIcon from '@mui/icons-material/Forward';
|
import ForwardIcon from '@mui/icons-material/Forward';
|
||||||
@@ -19,7 +19,7 @@ import type { SignInRequest } from 'types';
|
|||||||
import { onEnterCallback, updateValue } from 'utils';
|
import { onEnterCallback, updateValue } from 'utils';
|
||||||
import { SIGN_IN_REQUEST_VALIDATOR, validate } from 'validators';
|
import { SIGN_IN_REQUEST_VALIDATOR, validate } from 'validators';
|
||||||
|
|
||||||
const SignIn = memo(() => {
|
const SignIn = () => {
|
||||||
const authenticationContext = useContext(AuthenticationContext);
|
const authenticationContext = useContext(AuthenticationContext);
|
||||||
|
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
@@ -42,18 +42,9 @@ const SignIn = memo(() => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Memoize callback to prevent recreation on every render
|
const updateLoginRequestValue = updateValue(setSignInRequest);
|
||||||
const updateLoginRequestValue = useMemo(
|
|
||||||
() =>
|
|
||||||
updateValue((updater) =>
|
|
||||||
setSignInRequest(
|
|
||||||
updater as unknown as (prevState: SignInRequest) => SignInRequest
|
|
||||||
)
|
|
||||||
),
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
const signIn = useCallback(async () => {
|
const signIn = async () => {
|
||||||
await callSignIn(signInRequest).catch((event: Error) => {
|
await callSignIn(signInRequest).catch((event: Error) => {
|
||||||
if (event.message === 'Unauthorized') {
|
if (event.message === 'Unauthorized') {
|
||||||
toast.warning(LL.INVALID_LOGIN());
|
toast.warning(LL.INVALID_LOGIN());
|
||||||
@@ -62,9 +53,9 @@ const SignIn = memo(() => {
|
|||||||
}
|
}
|
||||||
setProcessing(false);
|
setProcessing(false);
|
||||||
});
|
});
|
||||||
}, [callSignIn, signInRequest, LL]);
|
};
|
||||||
|
|
||||||
const validateAndSignIn = useCallback(async () => {
|
const validateAndSignIn = async () => {
|
||||||
setProcessing(true);
|
setProcessing(true);
|
||||||
SIGN_IN_REQUEST_VALIDATOR.messages({
|
SIGN_IN_REQUEST_VALIDATOR.messages({
|
||||||
required: LL.IS_REQUIRED('%s')
|
required: LL.IS_REQUIRED('%s')
|
||||||
@@ -76,19 +67,9 @@ const SignIn = memo(() => {
|
|||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
setProcessing(false);
|
setProcessing(false);
|
||||||
}
|
}
|
||||||
}, [signInRequest, signIn, LL]);
|
};
|
||||||
|
|
||||||
// Memoize callback to prevent recreation on every render
|
const submitOnEnter = onEnterCallback(signIn);
|
||||||
const submitOnEnter = useMemo(() => onEnterCallback(signIn), [signIn]);
|
|
||||||
|
|
||||||
// get rid of scrollbar
|
|
||||||
useEffect(() => {
|
|
||||||
const originalOverflow = document.body.style.overflow;
|
|
||||||
document.body.style.overflow = 'hidden';
|
|
||||||
return () => {
|
|
||||||
document.body.style.overflow = originalOverflow;
|
|
||||||
};
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box
|
<Box
|
||||||
@@ -111,27 +92,23 @@ const SignIn = memo(() => {
|
|||||||
width: '100%'
|
width: '100%'
|
||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
<Typography mb={1} variant="h4">
|
<Typography variant="h4">{PROJECT_NAME}</Typography>
|
||||||
{PROJECT_NAME}
|
|
||||||
</Typography>
|
|
||||||
<LanguageSelector />
|
<LanguageSelector />
|
||||||
<Box
|
|
||||||
mt={1}
|
<Box display="flex" flexDirection="column" alignItems="center">
|
||||||
display="flex"
|
|
||||||
flexDirection="column"
|
|
||||||
gap={1}
|
|
||||||
alignItems="center"
|
|
||||||
>
|
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
disabled={processing}
|
disabled={processing}
|
||||||
sx={{
|
sx={{
|
||||||
width: '32ch'
|
width: 240
|
||||||
}}
|
}}
|
||||||
name="username"
|
name="username"
|
||||||
label={LL.USERNAME(0)}
|
label={LL.USERNAME(0)}
|
||||||
value={signInRequest.username}
|
value={signInRequest.username}
|
||||||
onChange={updateLoginRequestValue}
|
onChange={updateLoginRequestValue}
|
||||||
|
margin="normal"
|
||||||
|
variant="outlined"
|
||||||
slotProps={{
|
slotProps={{
|
||||||
input: {
|
input: {
|
||||||
autoCapitalize: 'none',
|
autoCapitalize: 'none',
|
||||||
@@ -140,16 +117,17 @@ const SignIn = memo(() => {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<ValidatedPasswordField
|
<ValidatedPasswordField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
disabled={processing}
|
disabled={processing}
|
||||||
sx={{
|
sx={{
|
||||||
width: '32ch'
|
width: 240
|
||||||
}}
|
}}
|
||||||
name="password"
|
name="password"
|
||||||
label={LL.PASSWORD()}
|
label={LL.PASSWORD()}
|
||||||
value={signInRequest.password}
|
value={signInRequest.password}
|
||||||
onChange={updateLoginRequestValue}
|
onChange={updateLoginRequestValue}
|
||||||
onKeyDown={submitOnEnter}
|
onKeyDown={submitOnEnter}
|
||||||
|
variant="outlined"
|
||||||
/>
|
/>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
@@ -166,6 +144,6 @@ const SignIn = memo(() => {
|
|||||||
</Paper>
|
</Paper>
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
});
|
};
|
||||||
|
|
||||||
export default SignIn;
|
export default SignIn;
|
||||||
|
|||||||
@@ -20,18 +20,19 @@ import type {
|
|||||||
WriteTemperatureSensor
|
WriteTemperatureSensor
|
||||||
} from '../app/main/types';
|
} from '../app/main/types';
|
||||||
|
|
||||||
const MSGPACK_CONFIG = { responseType: 'arraybuffer' as const };
|
|
||||||
|
|
||||||
// Dashboard
|
// Dashboard
|
||||||
export const readDashboard = () =>
|
export const readDashboard = () =>
|
||||||
alovaInstance.Get<DashboardData>('/rest/dashboardData', MSGPACK_CONFIG);
|
alovaInstance.Get<DashboardData>('/rest/dashboardData', {
|
||||||
|
responseType: 'arraybuffer' // uses msgpack
|
||||||
|
});
|
||||||
|
|
||||||
// Devices
|
// Devices
|
||||||
export const readCoreData = () => alovaInstance.Get<CoreData>('/rest/coreData');
|
export const readCoreData = () => alovaInstance.Get<CoreData>(`/rest/coreData`);
|
||||||
export const readDeviceData = (id: number) =>
|
export const readDeviceData = (id: number) =>
|
||||||
alovaInstance.Get<DeviceData>('/rest/deviceData', {
|
alovaInstance.Get<DeviceData>('/rest/deviceData', {
|
||||||
|
// alovaInstance.Get<DeviceData>(`/rest/deviceData/${id}`, {
|
||||||
params: { id },
|
params: { id },
|
||||||
...MSGPACK_CONFIG
|
responseType: 'arraybuffer' // uses msgpack
|
||||||
});
|
});
|
||||||
export const writeDeviceValue = (data: { id: number; c: string; v: unknown }) =>
|
export const writeDeviceValue = (data: { id: number; c: string; v: unknown }) =>
|
||||||
alovaInstance.Post('/rest/writeDeviceValue', data);
|
alovaInstance.Post('/rest/writeDeviceValue', data);
|
||||||
@@ -65,13 +66,12 @@ export const callAction = (action: Action) =>
|
|||||||
|
|
||||||
// SettingsCustomization
|
// SettingsCustomization
|
||||||
export const readDeviceEntities = (id: number) =>
|
export const readDeviceEntities = (id: number) =>
|
||||||
alovaInstance.Get<DeviceEntity[]>('/rest/deviceEntities', {
|
// alovaInstance.Get<DeviceEntity[]>(`/rest/deviceEntities/${id}`, {
|
||||||
|
alovaInstance.Get<DeviceEntity[]>(`/rest/deviceEntities`, {
|
||||||
params: { id },
|
params: { id },
|
||||||
...MSGPACK_CONFIG,
|
responseType: 'arraybuffer',
|
||||||
// @ts-expect-error - exactOptionalPropertyTypes compatibility issue
|
|
||||||
transform(data) {
|
transform(data) {
|
||||||
const entities = data as DeviceEntity[];
|
return (data as DeviceEntity[]).map((de: DeviceEntity) => ({
|
||||||
return entities.map((de) => ({
|
|
||||||
...de,
|
...de,
|
||||||
o_m: de.m,
|
o_m: de.m,
|
||||||
o_cn: de.cn,
|
o_cn: de.cn,
|
||||||
@@ -92,10 +92,8 @@ export const writeDeviceName = (data: { id: number; name: string }) =>
|
|||||||
// SettingsScheduler
|
// SettingsScheduler
|
||||||
export const readSchedule = () =>
|
export const readSchedule = () =>
|
||||||
alovaInstance.Get<ScheduleItem[]>('/rest/schedule', {
|
alovaInstance.Get<ScheduleItem[]>('/rest/schedule', {
|
||||||
// @ts-expect-error - exactOptionalPropertyTypes compatibility issue
|
|
||||||
transform(data) {
|
transform(data) {
|
||||||
const schedule = (data as Schedule).schedule;
|
return (data as Schedule).schedule.map((si: ScheduleItem) => ({
|
||||||
return schedule.map((si) => ({
|
|
||||||
...si,
|
...si,
|
||||||
o_id: si.id,
|
o_id: si.id,
|
||||||
o_active: si.active,
|
o_active: si.active,
|
||||||
@@ -115,8 +113,7 @@ export const writeSchedule = (data: Schedule) =>
|
|||||||
export const readModules = () =>
|
export const readModules = () =>
|
||||||
alovaInstance.Get<ModuleItem[]>('/rest/modules', {
|
alovaInstance.Get<ModuleItem[]>('/rest/modules', {
|
||||||
transform(data) {
|
transform(data) {
|
||||||
const modules = (data as Modules).modules;
|
return (data as Modules).modules.map((mi: ModuleItem) => ({
|
||||||
return modules.map((mi) => ({
|
|
||||||
...mi,
|
...mi,
|
||||||
o_enabled: mi.enabled,
|
o_enabled: mi.enabled,
|
||||||
o_license: mi.license
|
o_license: mi.license
|
||||||
@@ -132,10 +129,8 @@ export const writeModules = (data: {
|
|||||||
// CustomEntities
|
// CustomEntities
|
||||||
export const readCustomEntities = () =>
|
export const readCustomEntities = () =>
|
||||||
alovaInstance.Get<EntityItem[]>('/rest/customEntities', {
|
alovaInstance.Get<EntityItem[]>('/rest/customEntities', {
|
||||||
// @ts-expect-error - exactOptionalPropertyTypes compatibility issue
|
|
||||||
transform(data) {
|
transform(data) {
|
||||||
const entities = (data as Entities).entities;
|
return (data as Entities).entities.map((ei: EntityItem) => ({
|
||||||
return entities.map((ei) => ({
|
|
||||||
...ei,
|
...ei,
|
||||||
o_id: ei.id,
|
o_id: ei.id,
|
||||||
o_ram: ei.ram,
|
o_ram: ei.ram,
|
||||||
@@ -148,8 +143,7 @@ export const readCustomEntities = () =>
|
|||||||
o_name: ei.name,
|
o_name: ei.name,
|
||||||
o_writeable: ei.writeable,
|
o_writeable: ei.writeable,
|
||||||
o_value: ei.value,
|
o_value: ei.value,
|
||||||
o_deleted: ei.deleted,
|
o_deleted: ei.deleted
|
||||||
o_hide: ei.hide
|
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -4,57 +4,55 @@ import ReactHook from 'alova/react';
|
|||||||
|
|
||||||
import { unpack } from './unpack';
|
import { unpack } from './unpack';
|
||||||
|
|
||||||
export const ACCESS_TOKEN = 'access_token' as const;
|
export const ACCESS_TOKEN = 'access_token';
|
||||||
|
|
||||||
// Cached token to avoid repeated localStorage access
|
|
||||||
let cachedToken: string | null = null;
|
|
||||||
|
|
||||||
const getAccessToken = (): string | null => {
|
|
||||||
if (cachedToken === null) {
|
|
||||||
cachedToken = localStorage.getItem(ACCESS_TOKEN);
|
|
||||||
}
|
|
||||||
return cachedToken;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Clear token cache when needed (e.g., on logout)
|
|
||||||
export const clearTokenCache = (): void => {
|
|
||||||
cachedToken = null;
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleResponse = async (response: AlovaXHRResponse) => {
|
|
||||||
// Handle various HTTP status codes
|
|
||||||
if (response.status === 205) {
|
|
||||||
throw new Error('Reboot required');
|
|
||||||
}
|
|
||||||
if (response.status === 400) {
|
|
||||||
throw new Error('Request Failed');
|
|
||||||
}
|
|
||||||
if (response.status >= 400) {
|
|
||||||
throw new Error(response.statusText);
|
|
||||||
}
|
|
||||||
|
|
||||||
const data = (await response.data) as ArrayBuffer;
|
|
||||||
|
|
||||||
// Unpack MessagePack data if ArrayBuffer
|
|
||||||
if (data instanceof ArrayBuffer) {
|
|
||||||
return unpack(data) as ArrayBuffer;
|
|
||||||
}
|
|
||||||
|
|
||||||
return data;
|
|
||||||
};
|
|
||||||
|
|
||||||
export const alovaInstance = createAlova({
|
export const alovaInstance = createAlova({
|
||||||
statesHook: ReactHook,
|
statesHook: ReactHook,
|
||||||
|
// timeout: 3000, // 3 seconds before throwing a timeout error, default is 0 = none
|
||||||
cacheFor: null, // disable cache
|
cacheFor: null, // disable cache
|
||||||
|
// cacheFor: {
|
||||||
|
// GET: {
|
||||||
|
// mode: 'memory',
|
||||||
|
// expire: 60 * 10 * 1000 // 60 seconds in cache
|
||||||
|
// }
|
||||||
|
// },
|
||||||
requestAdapter: xhrRequestAdapter(),
|
requestAdapter: xhrRequestAdapter(),
|
||||||
beforeRequest(method) {
|
beforeRequest(method) {
|
||||||
const token = getAccessToken();
|
if (localStorage.getItem(ACCESS_TOKEN)) {
|
||||||
if (token) {
|
method.config.headers.Authorization =
|
||||||
method.config.headers.Authorization = `Bearer ${token}`;
|
'Bearer ' + localStorage.getItem(ACCESS_TOKEN);
|
||||||
}
|
}
|
||||||
|
// for simulating very slow networks
|
||||||
|
// return new Promise((resolve) => {
|
||||||
|
// const random = 3000 + Math.random() * 2000;
|
||||||
|
// setTimeout(resolve, Math.floor(random));
|
||||||
|
// });
|
||||||
},
|
},
|
||||||
|
|
||||||
responded: {
|
responded: {
|
||||||
onSuccess: handleResponse
|
onSuccess: async (response: AlovaXHRResponse) => {
|
||||||
|
// if (response.status === 202) {
|
||||||
|
// throw new Error('Wait'); // wifi scan in progress
|
||||||
|
// } else
|
||||||
|
if (response.status === 205) {
|
||||||
|
throw new Error('Reboot required');
|
||||||
|
} else if (response.status === 400) {
|
||||||
|
throw new Error('Request Failed');
|
||||||
|
} else if (response.status >= 400) {
|
||||||
|
throw new Error(response.statusText);
|
||||||
|
}
|
||||||
|
const data: ArrayBuffer = (await response.data) as ArrayBuffer;
|
||||||
|
if (response.data instanceof ArrayBuffer) {
|
||||||
|
return unpack(data) as ArrayBuffer;
|
||||||
|
}
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Interceptor for request failure. This interceptor will be entered when the request is wrong.
|
||||||
|
// http errors like 401 (unauthorized) are handled either in the methods or AuthenticatedRouting()
|
||||||
|
// onError: (error, method) => {
|
||||||
|
// alert(error.message);
|
||||||
|
// }
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -2,14 +2,12 @@ import type { NetworkSettingsType, NetworkStatusType, WiFiNetworkList } from 'ty
|
|||||||
|
|
||||||
import { alovaInstance } from './endpoints';
|
import { alovaInstance } from './endpoints';
|
||||||
|
|
||||||
const LIST_NETWORKS_TIMEOUT = 20000; // 20 seconds
|
|
||||||
|
|
||||||
export const readNetworkStatus = () =>
|
export const readNetworkStatus = () =>
|
||||||
alovaInstance.Get<NetworkStatusType>('/rest/networkStatus');
|
alovaInstance.Get<NetworkStatusType>('/rest/networkStatus');
|
||||||
export const scanNetworks = () => alovaInstance.Get('/rest/scanNetworks');
|
export const scanNetworks = () => alovaInstance.Get('/rest/scanNetworks');
|
||||||
export const listNetworks = () =>
|
export const listNetworks = () =>
|
||||||
alovaInstance.Get<WiFiNetworkList>('/rest/listNetworks', {
|
alovaInstance.Get<WiFiNetworkList>('/rest/listNetworks', {
|
||||||
timeout: LIST_NETWORKS_TIMEOUT
|
timeout: 20000 // 20 seconds
|
||||||
});
|
});
|
||||||
export const readNetworkSettings = () =>
|
export const readNetworkSettings = () =>
|
||||||
alovaInstance.Get<NetworkSettingsType>('/rest/networkSettings');
|
alovaInstance.Get<NetworkSettingsType>('/rest/networkSettings');
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ export const readNTPStatus = () =>
|
|||||||
alovaInstance.Get<NTPStatusType>('/rest/ntpStatus');
|
alovaInstance.Get<NTPStatusType>('/rest/ntpStatus');
|
||||||
|
|
||||||
export const readNTPSettings = () =>
|
export const readNTPSettings = () =>
|
||||||
alovaInstance.Get<NTPSettingsType>('/rest/ntpSettings');
|
alovaInstance.Get<NTPSettingsType>('/rest/ntpSettings', {});
|
||||||
export const updateNTPSettings = (data: NTPSettingsType) =>
|
export const updateNTPSettings = (data: NTPSettingsType) =>
|
||||||
alovaInstance.Post<NTPSettingsType>('/rest/ntpSettings', data);
|
alovaInstance.Post<NTPSettingsType>('/rest/ntpSettings', data);
|
||||||
|
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ export const readSystemStatus = () =>
|
|||||||
|
|
||||||
// SystemLog
|
// SystemLog
|
||||||
export const readLogSettings = () =>
|
export const readLogSettings = () =>
|
||||||
alovaInstance.Get<LogSettings>('/rest/logSettings');
|
alovaInstance.Get<LogSettings>(`/rest/logSettings`);
|
||||||
export const updateLogSettings = (data: LogSettings) =>
|
export const updateLogSettings = (data: LogSettings) =>
|
||||||
alovaInstance.Post('/rest/logSettings', data);
|
alovaInstance.Post('/rest/logSettings', data);
|
||||||
export const fetchLogES = () => alovaInstance.Get('/es/log');
|
export const fetchLogES = () => alovaInstance.Get('/es/log');
|
||||||
@@ -30,18 +30,16 @@ export const getDevVersion = () =>
|
|||||||
cacheFor: 60 * 10 * 1000,
|
cacheFor: 60 * 10 * 1000,
|
||||||
transform(response: { data: { name: string; published_at: string } }) {
|
transform(response: { data: { name: string; published_at: string } }) {
|
||||||
return {
|
return {
|
||||||
name: response.data.name.split(/\s+/).splice(-1)[0]?.substring(1) || '',
|
name: response.data.name.split(/\s+/).splice(-1)[0].substring(1),
|
||||||
published_at: response.data.published_at
|
published_at: response.data.published_at
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const UPLOAD_TIMEOUT = 60000; // 1 minute
|
|
||||||
|
|
||||||
export const uploadFile = (file: File) => {
|
export const uploadFile = (file: File) => {
|
||||||
const formData = new FormData();
|
const formData = new FormData();
|
||||||
formData.append('file', file);
|
formData.append('file', file);
|
||||||
return alovaInstance.Post('/rest/uploadFile', formData, {
|
return alovaInstance.Post('/rest/uploadFile', formData, {
|
||||||
timeout: UPLOAD_TIMEOUT
|
timeout: 60000 // override timeout for uploading firmware - 1 minute
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,37 +1,40 @@
|
|||||||
// @ts-nocheck - Optimized MessagePack unpacking library for EMS-ESP32
|
let decoder;
|
||||||
let decoder,
|
|
||||||
src,
|
|
||||||
srcEnd,
|
|
||||||
position = 0,
|
|
||||||
strings = [],
|
|
||||||
stringPosition = 0,
|
|
||||||
currentUnpackr = {},
|
|
||||||
currentStructures,
|
|
||||||
srcString,
|
|
||||||
srcStringStart = 0,
|
|
||||||
srcStringEnd = 0,
|
|
||||||
bundledStrings,
|
|
||||||
referenceMap,
|
|
||||||
dataView;
|
|
||||||
const EMPTY_ARRAY = [],
|
|
||||||
currentExtensions = [];
|
|
||||||
const defaultOptions = { useRecords: false, mapsAsObjects: true };
|
|
||||||
try {
|
try {
|
||||||
decoder = new TextDecoder();
|
decoder = new TextDecoder();
|
||||||
} catch (error) {}
|
} catch (error) {}
|
||||||
class C1Type {}
|
let src;
|
||||||
const C1 = new C1Type();
|
let srcEnd;
|
||||||
|
let position = 0;
|
||||||
|
const EMPTY_ARRAY = [];
|
||||||
|
let strings = EMPTY_ARRAY;
|
||||||
|
let stringPosition = 0;
|
||||||
|
let currentUnpackr = {};
|
||||||
|
let currentStructures;
|
||||||
|
let srcString;
|
||||||
|
let srcStringStart = 0;
|
||||||
|
let srcStringEnd = 0;
|
||||||
|
let bundledStrings;
|
||||||
|
let referenceMap;
|
||||||
|
const currentExtensions = [];
|
||||||
|
let dataView;
|
||||||
|
const defaultOptions = {
|
||||||
|
useRecords: false,
|
||||||
|
mapsAsObjects: true
|
||||||
|
};
|
||||||
|
export class C1Type {}
|
||||||
|
export const C1 = new C1Type();
|
||||||
C1.name = 'MessagePack 0xC1';
|
C1.name = 'MessagePack 0xC1';
|
||||||
let sequentialMode = false,
|
let sequentialMode = false;
|
||||||
inlineObjectReadThreshold = 2,
|
let inlineObjectReadThreshold = 2;
|
||||||
readStruct,
|
let readStruct, onLoadedStructures, onSaveState;
|
||||||
onLoadedStructures,
|
// no-eval build
|
||||||
onSaveState;
|
|
||||||
try {
|
try {
|
||||||
new Function('');
|
new Function('');
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
// if eval variants are not supported, do not create inline object readers ever
|
||||||
inlineObjectReadThreshold = Infinity;
|
inlineObjectReadThreshold = Infinity;
|
||||||
}
|
}
|
||||||
|
|
||||||
export class Unpackr {
|
export class Unpackr {
|
||||||
constructor(options) {
|
constructor(options) {
|
||||||
if (options) {
|
if (options) {
|
||||||
@@ -47,15 +50,19 @@ export class Unpackr {
|
|||||||
if (options.structures)
|
if (options.structures)
|
||||||
options.structures.sharedLength = options.structures.length;
|
options.structures.sharedLength = options.structures.length;
|
||||||
else if (options.getStructures) {
|
else if (options.getStructures) {
|
||||||
(options.structures = []).uninitialized = true;
|
(options.structures = []).uninitialized = true; // this is what we use to denote an uninitialized structures
|
||||||
options.structures.sharedLength = 0;
|
options.structures.sharedLength = 0;
|
||||||
}
|
}
|
||||||
if (options.int64AsNumber) options.int64AsType = 'number';
|
if (options.int64AsNumber) {
|
||||||
|
options.int64AsType = 'number';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
Object.assign(this, options);
|
Object.assign(this, options);
|
||||||
}
|
}
|
||||||
unpack(source, options?: { start?: number; end?: number; lazy?: boolean }) {
|
|
||||||
|
unpack(source, options?: any) {
|
||||||
if (src) {
|
if (src) {
|
||||||
|
// re-entrant execution, save the state and restore it after we do this unpack
|
||||||
return saveState(() => {
|
return saveState(() => {
|
||||||
clearSource();
|
clearSource();
|
||||||
return this
|
return this
|
||||||
@@ -79,6 +86,9 @@ export class Unpackr {
|
|||||||
strings = EMPTY_ARRAY;
|
strings = EMPTY_ARRAY;
|
||||||
bundledStrings = null;
|
bundledStrings = null;
|
||||||
src = source;
|
src = source;
|
||||||
|
// this provides cached access to the data view for a buffer if it is getting reused, which is a recommend
|
||||||
|
// technique for getting data from a database where it can be copied into an existing buffer instead of creating
|
||||||
|
// new ones
|
||||||
try {
|
try {
|
||||||
dataView =
|
dataView =
|
||||||
source.dataView ||
|
source.dataView ||
|
||||||
@@ -181,10 +191,10 @@ export class Unpackr {
|
|||||||
return this.unpack(source, end);
|
return this.unpack(source, end);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
function getPosition() {
|
export function getPosition() {
|
||||||
return position;
|
return position;
|
||||||
}
|
}
|
||||||
function checkedRead(options?: { lazy?: boolean }) {
|
export function checkedRead(options: any) {
|
||||||
try {
|
try {
|
||||||
if (!currentUnpackr.trusted && !sequentialMode) {
|
if (!currentUnpackr.trusted && !sequentialMode) {
|
||||||
const sharedLength = currentStructures.sharedLength || 0;
|
const sharedLength = currentStructures.sharedLength || 0;
|
||||||
@@ -254,7 +264,7 @@ function restoreStructures() {
|
|||||||
currentStructures.restoreStructures = null;
|
currentStructures.restoreStructures = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
function read() {
|
export function read() {
|
||||||
let token = src[position++];
|
let token = src[position++];
|
||||||
if (token < 0xa0) {
|
if (token < 0xa0) {
|
||||||
if (token < 0x80) {
|
if (token < 0x80) {
|
||||||
@@ -579,7 +589,7 @@ const createSecondByteReader = (firstId, read0) =>
|
|||||||
return structure.read();
|
return structure.read();
|
||||||
};
|
};
|
||||||
|
|
||||||
function loadStructures() {
|
export function loadStructures() {
|
||||||
const loadedStructures = saveState(() => {
|
const loadedStructures = saveState(() => {
|
||||||
// save the state in case getStructures modifies our buffer
|
// save the state in case getStructures modifies our buffer
|
||||||
src = null;
|
src = null;
|
||||||
@@ -595,8 +605,9 @@ var readFixedString = readStringJS;
|
|||||||
var readString8 = readStringJS;
|
var readString8 = readStringJS;
|
||||||
var readString16 = readStringJS;
|
var readString16 = readStringJS;
|
||||||
var readString32 = readStringJS;
|
var readString32 = readStringJS;
|
||||||
let isNativeAccelerationEnabled = false;
|
export let isNativeAccelerationEnabled = false;
|
||||||
function setExtractor(extractStrings) {
|
|
||||||
|
export function setExtractor(extractStrings) {
|
||||||
isNativeAccelerationEnabled = true;
|
isNativeAccelerationEnabled = true;
|
||||||
readFixedString = readString(1);
|
readFixedString = readString(1);
|
||||||
readString8 = readString(2);
|
readString8 = readString(2);
|
||||||
@@ -690,7 +701,7 @@ function readStringJS(length) {
|
|||||||
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
function readString(source, start, length) {
|
export function readString(source, start, length) {
|
||||||
const existingSrc = src;
|
const existingSrc = src;
|
||||||
src = source;
|
src = source;
|
||||||
position = start;
|
position = start;
|
||||||
@@ -1054,7 +1065,7 @@ currentExtensions[0x70] = (data) => {
|
|||||||
|
|
||||||
currentExtensions[0x73] = () => new Set(read());
|
currentExtensions[0x73] = () => new Set(read());
|
||||||
|
|
||||||
const typedArrays = [
|
export const typedArrays = [
|
||||||
'Int8',
|
'Int8',
|
||||||
'Uint8',
|
'Uint8',
|
||||||
'Uint8Clamped',
|
'Uint8Clamped',
|
||||||
@@ -1166,20 +1177,44 @@ function saveState(callback) {
|
|||||||
dataView = new DataView(src.buffer, src.byteOffset, src.byteLength);
|
dataView = new DataView(src.buffer, src.byteOffset, src.byteLength);
|
||||||
return value;
|
return value;
|
||||||
}
|
}
|
||||||
function clearSource() {
|
export function clearSource() {
|
||||||
src = null;
|
src = null;
|
||||||
referenceMap = null;
|
referenceMap = null;
|
||||||
currentStructures = null;
|
currentStructures = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
function addExtension(extension) {
|
export function addExtension(extension) {
|
||||||
if (extension.unpack) currentExtensions[extension.type] = extension.unpack;
|
if (extension.unpack) currentExtensions[extension.type] = extension.unpack;
|
||||||
else currentExtensions[extension.type] = extension;
|
else currentExtensions[extension.type] = extension;
|
||||||
}
|
}
|
||||||
|
|
||||||
const mult10 = new Array(147);
|
export const mult10 = new Array(147); // this is a table matching binary exponents to the multiplier to determine significant digit rounding
|
||||||
for (let i = 0; i < 256; i++) {
|
for (let i = 0; i < 256; i++) {
|
||||||
mult10[i] = +('1e' + Math.floor(45.15 - i * 0.30103));
|
mult10[i] = +('1e' + Math.floor(45.15 - i * 0.30103));
|
||||||
}
|
}
|
||||||
const defaultUnpackr = new Unpackr({ useRecords: false });
|
export const Decoder = Unpackr;
|
||||||
|
var defaultUnpackr = new Unpackr({ useRecords: false });
|
||||||
export const unpack = defaultUnpackr.unpack;
|
export const unpack = defaultUnpackr.unpack;
|
||||||
|
export const unpackMultiple = defaultUnpackr.unpackMultiple;
|
||||||
|
export const decode = defaultUnpackr.unpack;
|
||||||
|
export const FLOAT32_OPTIONS = {
|
||||||
|
NEVER: 0,
|
||||||
|
ALWAYS: 1,
|
||||||
|
DECIMAL_ROUND: 3,
|
||||||
|
DECIMAL_FIT: 4
|
||||||
|
};
|
||||||
|
const f32Array = new Float32Array(1);
|
||||||
|
const u8Array = new Uint8Array(f32Array.buffer, 0, 4);
|
||||||
|
export function roundFloat32(float32Number) {
|
||||||
|
f32Array[0] = float32Number;
|
||||||
|
const multiplier = mult10[((u8Array[3] & 0x7f) << 1) | (u8Array[2] >> 7)];
|
||||||
|
return (
|
||||||
|
((multiplier * float32Number + (float32Number > 0 ? 0.5 : -0.5)) >> 0) /
|
||||||
|
multiplier
|
||||||
|
);
|
||||||
|
}
|
||||||
|
export function setReadStruct(updatedReadStruct, loadedStructs, saveState) {
|
||||||
|
readStruct = updatedReadStruct;
|
||||||
|
onLoadedStructures = loadedStructs;
|
||||||
|
onSaveState = saveState;
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useMemo, useState } from 'react';
|
import { useCallback, useState } from 'react';
|
||||||
import { useBlocker } from 'react-router';
|
import { useBlocker } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
@@ -35,10 +35,6 @@ import { DeviceValueTypeNames, DeviceValueUOM_s } from './types';
|
|||||||
import type { Entities, EntityItem } from './types';
|
import type { Entities, EntityItem } from './types';
|
||||||
import { entityItemValidation } from './validators';
|
import { entityItemValidation } from './validators';
|
||||||
|
|
||||||
const MIN_ID = -100;
|
|
||||||
const MAX_ID = 100;
|
|
||||||
const ICON_SIZE = 12;
|
|
||||||
|
|
||||||
const CustomEntities = () => {
|
const CustomEntities = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [numChanges, setNumChanges] = useState<number>(0);
|
const [numChanges, setNumChanges] = useState<number>(0);
|
||||||
@@ -57,20 +53,18 @@ const CustomEntities = () => {
|
|||||||
initialData: []
|
initialData: []
|
||||||
});
|
});
|
||||||
|
|
||||||
const intervalCallback = useCallback(() => {
|
useInterval(() => {
|
||||||
if (!dialogOpen && !numChanges) {
|
if (!dialogOpen && !numChanges) {
|
||||||
void fetchEntities();
|
void fetchEntities();
|
||||||
}
|
}
|
||||||
}, [dialogOpen, numChanges, fetchEntities]);
|
});
|
||||||
|
|
||||||
useInterval(intervalCallback);
|
|
||||||
|
|
||||||
const { send: writeEntities } = useRequest(
|
const { send: writeEntities } = useRequest(
|
||||||
(data: Entities) => writeCustomEntities(data),
|
(data: Entities) => writeCustomEntities(data),
|
||||||
{ immediate: false }
|
{ immediate: false }
|
||||||
);
|
);
|
||||||
|
|
||||||
const hasEntityChanged = useCallback((ei: EntityItem) => {
|
function hasEntityChanged(ei: EntityItem) {
|
||||||
return (
|
return (
|
||||||
ei.id !== ei.o_id ||
|
ei.id !== ei.o_id ||
|
||||||
ei.ram !== ei.o_ram ||
|
ei.ram !== ei.o_ram ||
|
||||||
@@ -82,25 +76,22 @@ const CustomEntities = () => {
|
|||||||
ei.factor !== ei.o_factor ||
|
ei.factor !== ei.o_factor ||
|
||||||
ei.value_type !== ei.o_value_type ||
|
ei.value_type !== ei.o_value_type ||
|
||||||
ei.writeable !== ei.o_writeable ||
|
ei.writeable !== ei.o_writeable ||
|
||||||
ei.hide !== ei.o_hide ||
|
|
||||||
ei.deleted !== ei.o_deleted ||
|
ei.deleted !== ei.o_deleted ||
|
||||||
(ei.value || '') !== (ei.o_value || '')
|
(ei.value || '') !== (ei.o_value || '')
|
||||||
);
|
);
|
||||||
}, []);
|
}
|
||||||
|
|
||||||
const entity_theme = useMemo(
|
const entity_theme = useTheme({
|
||||||
() =>
|
Table: `
|
||||||
useTheme({
|
|
||||||
Table: `
|
|
||||||
--data-table-library_grid-template-columns: repeat(1, minmax(60px, 1fr)) minmax(80px, auto) 80px 80px 80px 120px;
|
--data-table-library_grid-template-columns: repeat(1, minmax(60px, 1fr)) minmax(80px, auto) 80px 80px 80px 120px;
|
||||||
`,
|
`,
|
||||||
BaseRow: `
|
BaseRow: `
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
.td {
|
.td {
|
||||||
height: 32px;
|
height: 32px;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
BaseCell: `
|
BaseCell: `
|
||||||
&:nth-of-type(1) {
|
&:nth-of-type(1) {
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
}
|
}
|
||||||
@@ -120,7 +111,7 @@ const CustomEntities = () => {
|
|||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
HeaderRow: `
|
HeaderRow: `
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
background-color: black;
|
background-color: black;
|
||||||
color: #90CAF9;
|
color: #90CAF9;
|
||||||
@@ -129,7 +120,7 @@ const CustomEntities = () => {
|
|||||||
height: 36px;
|
height: 36px;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
Row: `
|
Row: `
|
||||||
background-color: #1e1e1e;
|
background-color: #1e1e1e;
|
||||||
position: relative;
|
position: relative;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
@@ -140,15 +131,13 @@ const CustomEntities = () => {
|
|||||||
background-color: #177ac9;
|
background-color: #177ac9;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
}),
|
});
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
const saveEntities = useCallback(async () => {
|
const saveEntities = async () => {
|
||||||
await writeEntities({
|
await writeEntities({
|
||||||
entities: entities
|
entities: entities
|
||||||
.filter((ei: EntityItem) => !ei.deleted)
|
.filter((ei) => !ei.deleted)
|
||||||
.map((condensed_ei: EntityItem) => ({
|
.map((condensed_ei) => ({
|
||||||
id: condensed_ei.id,
|
id: condensed_ei.id,
|
||||||
ram: condensed_ei.ram,
|
ram: condensed_ei.ram,
|
||||||
name: condensed_ei.name,
|
name: condensed_ei.name,
|
||||||
@@ -158,7 +147,6 @@ const CustomEntities = () => {
|
|||||||
factor: condensed_ei.factor,
|
factor: condensed_ei.factor,
|
||||||
uom: condensed_ei.uom,
|
uom: condensed_ei.uom,
|
||||||
writeable: condensed_ei.writeable,
|
writeable: condensed_ei.writeable,
|
||||||
hide: condensed_ei.hide,
|
|
||||||
value_type: condensed_ei.value_type,
|
value_type: condensed_ei.value_type,
|
||||||
value: condensed_ei.value
|
value: condensed_ei.value
|
||||||
}))
|
}))
|
||||||
@@ -173,7 +161,7 @@ const CustomEntities = () => {
|
|||||||
await fetchEntities();
|
await fetchEntities();
|
||||||
setNumChanges(0);
|
setNumChanges(0);
|
||||||
});
|
});
|
||||||
}, [entities, writeEntities, LL, fetchEntities]);
|
};
|
||||||
|
|
||||||
const editEntityItem = useCallback((ei: EntityItem) => {
|
const editEntityItem = useCallback((ei: EntityItem) => {
|
||||||
setCreating(false);
|
setCreating(false);
|
||||||
@@ -181,39 +169,36 @@ const CustomEntities = () => {
|
|||||||
setDialogOpen(true);
|
setDialogOpen(true);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const onDialogClose = useCallback(() => {
|
const onDialogClose = () => {
|
||||||
setDialogOpen(false);
|
setDialogOpen(false);
|
||||||
}, []);
|
};
|
||||||
|
|
||||||
const onDialogCancel = useCallback(async () => {
|
const onDialogCancel = async () => {
|
||||||
await fetchEntities().then(() => {
|
await fetchEntities().then(() => {
|
||||||
setNumChanges(0);
|
setNumChanges(0);
|
||||||
});
|
});
|
||||||
}, [fetchEntities]);
|
};
|
||||||
|
|
||||||
const onDialogSave = useCallback(
|
const onDialogSave = (updatedItem: EntityItem) => {
|
||||||
(updatedItem: EntityItem) => {
|
setDialogOpen(false);
|
||||||
setDialogOpen(false);
|
void updateState(readCustomEntities(), (data: EntityItem[]) => {
|
||||||
void updateState(readCustomEntities(), (data: EntityItem[]) => {
|
const new_data = creating
|
||||||
const new_data = creating
|
? [
|
||||||
? [
|
...data.filter((ei) => creating || ei.o_id !== updatedItem.o_id),
|
||||||
...data.filter((ei) => creating || ei.o_id !== updatedItem.o_id),
|
updatedItem
|
||||||
updatedItem
|
]
|
||||||
]
|
: data.map((ei) =>
|
||||||
: data.map((ei) =>
|
ei.id === updatedItem.id ? { ...ei, ...updatedItem } : ei
|
||||||
ei.id === updatedItem.id ? { ...ei, ...updatedItem } : ei
|
);
|
||||||
);
|
setNumChanges(new_data.filter((ei) => hasEntityChanged(ei)).length);
|
||||||
setNumChanges(new_data.filter((ei) => hasEntityChanged(ei)).length);
|
return new_data;
|
||||||
return new_data;
|
});
|
||||||
});
|
};
|
||||||
},
|
|
||||||
[creating, hasEntityChanged]
|
|
||||||
);
|
|
||||||
|
|
||||||
const onDialogDup = useCallback((item: EntityItem) => {
|
const onDialogDup = (item: EntityItem) => {
|
||||||
setCreating(true);
|
setCreating(true);
|
||||||
setSelectedEntityItem({
|
setSelectedEntityItem({
|
||||||
id: Math.floor(Math.random() * (MAX_ID - MIN_ID) + MIN_ID),
|
id: Math.floor(Math.random() * (Math.floor(200) - 100) + 100),
|
||||||
name: item.name + '_',
|
name: item.name + '_',
|
||||||
ram: item.ram,
|
ram: item.ram,
|
||||||
device_id: item.device_id,
|
device_id: item.device_id,
|
||||||
@@ -224,16 +209,15 @@ const CustomEntities = () => {
|
|||||||
value_type: item.value_type,
|
value_type: item.value_type,
|
||||||
writeable: item.writeable,
|
writeable: item.writeable,
|
||||||
deleted: false,
|
deleted: false,
|
||||||
hide: item.hide,
|
|
||||||
value: item.value
|
value: item.value
|
||||||
});
|
});
|
||||||
setDialogOpen(true);
|
setDialogOpen(true);
|
||||||
}, []);
|
};
|
||||||
|
|
||||||
const addEntityItem = useCallback(() => {
|
const addEntityItem = () => {
|
||||||
setCreating(true);
|
setCreating(true);
|
||||||
setSelectedEntityItem({
|
setSelectedEntityItem({
|
||||||
id: Math.floor(Math.random() * (MAX_ID - MIN_ID) + MIN_ID),
|
id: Math.floor(Math.random() * (Math.floor(200) - 100) + 100),
|
||||||
name: '',
|
name: '',
|
||||||
ram: 0,
|
ram: 0,
|
||||||
device_id: '0',
|
device_id: '0',
|
||||||
@@ -244,44 +228,35 @@ const CustomEntities = () => {
|
|||||||
value_type: 0,
|
value_type: 0,
|
||||||
writeable: false,
|
writeable: false,
|
||||||
deleted: false,
|
deleted: false,
|
||||||
hide: false,
|
|
||||||
value: ''
|
value: ''
|
||||||
});
|
});
|
||||||
setDialogOpen(true);
|
setDialogOpen(true);
|
||||||
}, []);
|
};
|
||||||
|
|
||||||
const formatValue = useCallback((value: unknown, uom: number) => {
|
function formatValue(value: unknown, uom: number) {
|
||||||
return value === undefined
|
return value === undefined
|
||||||
? ''
|
? ''
|
||||||
: typeof value === 'number'
|
: typeof value === 'number'
|
||||||
? new Intl.NumberFormat().format(value) +
|
? new Intl.NumberFormat().format(value) +
|
||||||
(uom === 0 ? '' : ` ${DeviceValueUOM_s[uom]}`)
|
(uom === 0 ? '' : ' ' + DeviceValueUOM_s[uom])
|
||||||
: `${value as string}${uom === 0 ? '' : ` ${DeviceValueUOM_s[uom]}`}`;
|
: (value as string) + (uom === 0 ? '' : ' ' + DeviceValueUOM_s[uom]);
|
||||||
}, []);
|
}
|
||||||
|
|
||||||
const showHex = useCallback((value: number, digit: number) => {
|
function showHex(value: number, digit: number) {
|
||||||
return `0x${value.toString(16).toUpperCase().padStart(digit, '0')}`;
|
return '0x' + value.toString(16).toUpperCase().padStart(digit, '0');
|
||||||
}, []);
|
}
|
||||||
|
|
||||||
const filteredAndSortedEntities = useMemo(
|
const renderEntity = () => {
|
||||||
() =>
|
|
||||||
entities
|
|
||||||
?.filter((ei: EntityItem) => !ei.deleted)
|
|
||||||
.sort((a: EntityItem, b: EntityItem) => a.name.localeCompare(b.name)) ?? [],
|
|
||||||
[entities]
|
|
||||||
);
|
|
||||||
|
|
||||||
const renderEntity = useCallback(() => {
|
|
||||||
if (!entities) {
|
if (!entities) {
|
||||||
return (
|
return <FormLoader onRetry={fetchEntities} errorMessage={error?.message} />;
|
||||||
<FormLoader onRetry={fetchEntities} errorMessage={error?.message || ''} />
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Table
|
<Table
|
||||||
data={{
|
data={{
|
||||||
nodes: filteredAndSortedEntities
|
nodes: entities
|
||||||
|
.filter((ei) => !ei.deleted)
|
||||||
|
.sort((a, b) => a.name.localeCompare(b.name))
|
||||||
}}
|
}}
|
||||||
theme={entity_theme}
|
theme={entity_theme}
|
||||||
layout={{ custom: true }}
|
layout={{ custom: true }}
|
||||||
@@ -304,21 +279,16 @@ const CustomEntities = () => {
|
|||||||
<Cell>
|
<Cell>
|
||||||
{ei.name}
|
{ei.name}
|
||||||
{ei.writeable && (
|
{ei.writeable && (
|
||||||
<EditOutlinedIcon
|
<EditOutlinedIcon color="primary" sx={{ fontSize: 12 }} />
|
||||||
color="primary"
|
|
||||||
sx={{ fontSize: ICON_SIZE }}
|
|
||||||
/>
|
|
||||||
)}
|
)}
|
||||||
</Cell>
|
</Cell>
|
||||||
<Cell>{ei.ram > 0 ? '' : showHex(ei.device_id as number, 2)}</Cell>
|
|
||||||
<Cell>{ei.ram > 0 ? '' : showHex(ei.type_id as number, 3)}</Cell>
|
|
||||||
<Cell>{ei.ram > 0 ? '' : ei.offset}</Cell>
|
|
||||||
<Cell>
|
<Cell>
|
||||||
{ei.ram === 1
|
{ei.ram === 1 ? '' : showHex(ei.device_id as number, 2)}
|
||||||
? 'RAM'
|
</Cell>
|
||||||
: ei.ram === 2
|
<Cell>{ei.ram === 1 ? '' : showHex(ei.type_id as number, 3)}</Cell>
|
||||||
? 'NVS'
|
<Cell>{ei.ram === 1 ? '' : ei.offset}</Cell>
|
||||||
: DeviceValueTypeNames[ei.value_type]}
|
<Cell>
|
||||||
|
{ei.ram === 1 ? 'RAM' : DeviceValueTypeNames[ei.value_type]}
|
||||||
</Cell>
|
</Cell>
|
||||||
<Cell>{formatValue(ei.value, ei.uom)}</Cell>
|
<Cell>{formatValue(ei.value, ei.uom)}</Cell>
|
||||||
</Row>
|
</Row>
|
||||||
@@ -328,17 +298,7 @@ const CustomEntities = () => {
|
|||||||
)}
|
)}
|
||||||
</Table>
|
</Table>
|
||||||
);
|
);
|
||||||
}, [
|
};
|
||||||
entities,
|
|
||||||
error,
|
|
||||||
fetchEntities,
|
|
||||||
entity_theme,
|
|
||||||
editEntityItem,
|
|
||||||
LL,
|
|
||||||
filteredAndSortedEntities,
|
|
||||||
showHex,
|
|
||||||
formatValue
|
|
||||||
]);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
@@ -361,7 +321,7 @@ const CustomEntities = () => {
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Box mt={2} display="flex" flexWrap="wrap">
|
<Box mt={1} display="flex" flexWrap="wrap">
|
||||||
<Box flexGrow={1}>
|
<Box flexGrow={1}>
|
||||||
{numChanges > 0 && (
|
{numChanges > 0 && (
|
||||||
<ButtonRow>
|
<ButtonRow>
|
||||||
|
|||||||
@@ -1,12 +1,8 @@
|
|||||||
import { useCallback, useEffect, useMemo, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
import AddIcon from '@mui/icons-material/Add';
|
import AddIcon from '@mui/icons-material/Add';
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import CommentsDisabledOutlinedIcon from '@mui/icons-material/CommentsDisabledOutlined';
|
|
||||||
import DoneIcon from '@mui/icons-material/Done';
|
import DoneIcon from '@mui/icons-material/Done';
|
||||||
import EditOffOutlinedIcon from '@mui/icons-material/EditOffOutlined';
|
|
||||||
import EditOutlinedIcon from '@mui/icons-material/EditOutlined';
|
|
||||||
import InsertCommentOutlinedIcon from '@mui/icons-material/InsertCommentOutlined';
|
|
||||||
import RemoveIcon from '@mui/icons-material/RemoveCircleOutline';
|
import RemoveIcon from '@mui/icons-material/RemoveCircleOutline';
|
||||||
import {
|
import {
|
||||||
Box,
|
Box,
|
||||||
@@ -16,7 +12,7 @@ import {
|
|||||||
DialogActions,
|
DialogActions,
|
||||||
DialogContent,
|
DialogContent,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
Grid,
|
Grid2 as Grid,
|
||||||
InputAdornment,
|
InputAdornment,
|
||||||
MenuItem,
|
MenuItem,
|
||||||
TextField
|
TextField
|
||||||
@@ -33,19 +29,6 @@ import { validate } from 'validators';
|
|||||||
import { DeviceValueType, DeviceValueTypeNames, DeviceValueUOM_s } from './types';
|
import { DeviceValueType, DeviceValueTypeNames, DeviceValueUOM_s } from './types';
|
||||||
import type { EntityItem } from './types';
|
import type { EntityItem } from './types';
|
||||||
|
|
||||||
// Constant value type options for the dropdown
|
|
||||||
const VALUE_TYPE_OPTIONS = [
|
|
||||||
DeviceValueType.BOOL,
|
|
||||||
DeviceValueType.INT8,
|
|
||||||
DeviceValueType.UINT8,
|
|
||||||
DeviceValueType.INT16,
|
|
||||||
DeviceValueType.UINT16,
|
|
||||||
DeviceValueType.UINT24,
|
|
||||||
DeviceValueType.TIME,
|
|
||||||
DeviceValueType.UINT32,
|
|
||||||
DeviceValueType.STRING
|
|
||||||
] as const;
|
|
||||||
|
|
||||||
interface CustomEntitiesDialogProps {
|
interface CustomEntitiesDialogProps {
|
||||||
open: boolean;
|
open: boolean;
|
||||||
creating: boolean;
|
creating: boolean;
|
||||||
@@ -68,97 +51,61 @@ const CustomEntitiesDialog = ({
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [editItem, setEditItem] = useState<EntityItem>(selectedItem);
|
const [editItem, setEditItem] = useState<EntityItem>(selectedItem);
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
const updateFormValue = useMemo(
|
const updateFormValue = updateValue(setEditItem);
|
||||||
() =>
|
|
||||||
updateValue(
|
|
||||||
setEditItem as unknown as React.Dispatch<
|
|
||||||
React.SetStateAction<Record<string, unknown>>
|
|
||||||
>
|
|
||||||
),
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
// Convert to hex strings - combined into single setEditItem call
|
setEditItem(selectedItem);
|
||||||
const deviceIdHex =
|
// convert to hex strings straight away
|
||||||
typeof selectedItem.device_id === 'number'
|
|
||||||
? selectedItem.device_id.toString(16).toUpperCase()
|
|
||||||
: selectedItem.device_id;
|
|
||||||
const typeIdHex =
|
|
||||||
typeof selectedItem.type_id === 'number'
|
|
||||||
? selectedItem.type_id.toString(16).toUpperCase()
|
|
||||||
: selectedItem.type_id;
|
|
||||||
const factorValue =
|
|
||||||
selectedItem.value_type === DeviceValueType.BOOL &&
|
|
||||||
typeof selectedItem.factor === 'number'
|
|
||||||
? selectedItem.factor.toString(16).toUpperCase()
|
|
||||||
: selectedItem.factor;
|
|
||||||
|
|
||||||
setEditItem({
|
setEditItem({
|
||||||
...selectedItem,
|
...selectedItem,
|
||||||
device_id: deviceIdHex,
|
device_id: selectedItem.device_id.toString(16).toUpperCase(),
|
||||||
type_id: typeIdHex,
|
type_id: selectedItem.type_id.toString(16).toUpperCase(),
|
||||||
factor: factorValue
|
factor:
|
||||||
|
selectedItem.value_type === DeviceValueType.BOOL
|
||||||
|
? selectedItem.factor.toString(16).toUpperCase()
|
||||||
|
: selectedItem.factor
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const handleClose = useCallback(
|
const handleClose = (_event, reason: 'backdropClick' | 'escapeKeyDown') => {
|
||||||
(_event: React.SyntheticEvent, reason: 'backdropClick' | 'escapeKeyDown') => {
|
if (reason !== 'backdropClick') {
|
||||||
if (reason !== 'backdropClick') {
|
onClose();
|
||||||
onClose();
|
}
|
||||||
}
|
};
|
||||||
},
|
|
||||||
[onClose]
|
|
||||||
);
|
|
||||||
|
|
||||||
const save = useCallback(async () => {
|
const save = async () => {
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(validator, editItem);
|
await validate(validator, editItem);
|
||||||
|
if (typeof editItem.device_id === 'string') {
|
||||||
// Create a copy to avoid mutating the state directly
|
editItem.device_id = parseInt(editItem.device_id, 16);
|
||||||
const processedItem: EntityItem = { ...editItem };
|
|
||||||
|
|
||||||
if (typeof processedItem.device_id === 'string') {
|
|
||||||
processedItem.device_id = Number.parseInt(processedItem.device_id, 16);
|
|
||||||
}
|
}
|
||||||
if (typeof processedItem.type_id === 'string') {
|
if (typeof editItem.type_id === 'string') {
|
||||||
processedItem.type_id = Number.parseInt(processedItem.type_id, 16);
|
editItem.type_id = parseInt(editItem.type_id, 16);
|
||||||
}
|
}
|
||||||
if (
|
if (
|
||||||
processedItem.value_type === DeviceValueType.BOOL &&
|
editItem.value_type === DeviceValueType.BOOL &&
|
||||||
typeof processedItem.factor === 'string'
|
typeof editItem.factor === 'string'
|
||||||
) {
|
) {
|
||||||
processedItem.factor = Number.parseInt(processedItem.factor, 16);
|
editItem.factor = parseInt(editItem.factor, 16);
|
||||||
}
|
}
|
||||||
onSave(processedItem);
|
onSave(editItem);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
}, [validator, editItem, onSave]);
|
};
|
||||||
|
|
||||||
const remove = useCallback(() => {
|
const remove = () => {
|
||||||
const itemWithDeleted = { ...editItem, deleted: true };
|
editItem.deleted = true;
|
||||||
onSave(itemWithDeleted);
|
onSave(editItem);
|
||||||
}, [editItem, onSave]);
|
};
|
||||||
|
|
||||||
const dup = useCallback(() => {
|
const dup = () => {
|
||||||
onDup(editItem);
|
onDup(editItem);
|
||||||
}, [editItem, onDup]);
|
};
|
||||||
|
|
||||||
// Memoize UOM menu items to avoid recreating on every render
|
|
||||||
const uomMenuItems = useMemo(
|
|
||||||
() =>
|
|
||||||
DeviceValueUOM_s.map((val, i) => (
|
|
||||||
<MenuItem key={val} value={i}>
|
|
||||||
{val}
|
|
||||||
</MenuItem>
|
|
||||||
)),
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
||||||
@@ -166,10 +113,13 @@ const CustomEntitiesDialog = ({
|
|||||||
{creating ? LL.ADD(1) + ' ' + LL.NEW(1) : LL.EDIT()} {LL.ENTITY()}
|
{creating ? LL.ADD(1) + ' ' + LL.NEW(1) : LL.EDIT()} {LL.ENTITY()}
|
||||||
</DialogTitle>
|
</DialogTitle>
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
|
<Box display="flex" flexWrap="wrap" mb={1}>
|
||||||
|
<Box flexWrap="nowrap" whiteSpace="nowrap" />
|
||||||
|
</Box>
|
||||||
<Grid container spacing={2} rowSpacing={0}>
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
<Grid size={12}>
|
<Grid size={12}>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="name"
|
name="name"
|
||||||
label={LL.NAME(0)}
|
label={LL.NAME(0)}
|
||||||
value={editItem.name}
|
value={editItem.name}
|
||||||
@@ -178,20 +128,6 @@ const CustomEntitiesDialog = ({
|
|||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid mt={3}>
|
|
||||||
<BlockFormControlLabel
|
|
||||||
control={
|
|
||||||
<Checkbox
|
|
||||||
icon={<InsertCommentOutlinedIcon htmlColor="white" />}
|
|
||||||
checkedIcon={<CommentsDisabledOutlinedIcon color="primary" />}
|
|
||||||
checked={editItem.hide}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
name="hide"
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
label="API/MQTT"
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<TextField
|
||||||
name="ram"
|
name="ram"
|
||||||
@@ -205,10 +141,9 @@ const CustomEntitiesDialog = ({
|
|||||||
>
|
>
|
||||||
<MenuItem value={0}>EMS-{LL.VALUE(1)}</MenuItem>
|
<MenuItem value={0}>EMS-{LL.VALUE(1)}</MenuItem>
|
||||||
<MenuItem value={1}>RAM-{LL.VALUE(1)}</MenuItem>
|
<MenuItem value={1}>RAM-{LL.VALUE(1)}</MenuItem>
|
||||||
<MenuItem value={2}>NVS-{LL.VALUE(1)}</MenuItem>
|
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
{editItem.ram > 0 && (
|
{editItem.ram === 1 && (
|
||||||
<>
|
<>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<TextField
|
||||||
@@ -231,19 +166,21 @@ const CustomEntitiesDialog = ({
|
|||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
select
|
select
|
||||||
>
|
>
|
||||||
{uomMenuItems}
|
{DeviceValueUOM_s.map((val, i) => (
|
||||||
|
<MenuItem key={val} value={i}>
|
||||||
|
{val}
|
||||||
|
</MenuItem>
|
||||||
|
))}
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
{editItem.ram === 0 && (
|
{editItem.ram === 0 && (
|
||||||
<>
|
<>
|
||||||
<Grid mt={3}>
|
<Grid mt={3} size={9}>
|
||||||
<BlockFormControlLabel
|
<BlockFormControlLabel
|
||||||
control={
|
control={
|
||||||
<Checkbox
|
<Checkbox
|
||||||
icon={<EditOffOutlinedIcon color="primary" />}
|
|
||||||
checkedIcon={<EditOutlinedIcon htmlColor="white" />}
|
|
||||||
checked={editItem.writeable}
|
checked={editItem.writeable}
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
name="writeable"
|
name="writeable"
|
||||||
@@ -254,7 +191,7 @@ const CustomEntitiesDialog = ({
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="device_id"
|
name="device_id"
|
||||||
label={LL.ID_OF(LL.DEVICE())}
|
label={LL.ID_OF(LL.DEVICE())}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
@@ -274,7 +211,7 @@ const CustomEntitiesDialog = ({
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="type_id"
|
name="type_id"
|
||||||
label={LL.ID_OF(LL.TYPE(1))}
|
label={LL.ID_OF(LL.TYPE(1))}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
@@ -294,7 +231,7 @@ const CustomEntitiesDialog = ({
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="offset"
|
name="offset"
|
||||||
label={LL.OFFSET()}
|
label={LL.OFFSET()}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
@@ -315,11 +252,33 @@ const CustomEntitiesDialog = ({
|
|||||||
margin="normal"
|
margin="normal"
|
||||||
select
|
select
|
||||||
>
|
>
|
||||||
{VALUE_TYPE_OPTIONS.map((valueType) => (
|
<MenuItem value={DeviceValueType.BOOL}>
|
||||||
<MenuItem key={valueType} value={valueType}>
|
{DeviceValueTypeNames[DeviceValueType.BOOL]}
|
||||||
{DeviceValueTypeNames[valueType]}
|
</MenuItem>
|
||||||
</MenuItem>
|
<MenuItem value={DeviceValueType.INT8}>
|
||||||
))}
|
{DeviceValueTypeNames[DeviceValueType.INT8]}
|
||||||
|
</MenuItem>
|
||||||
|
<MenuItem value={DeviceValueType.UINT8}>
|
||||||
|
{DeviceValueTypeNames[DeviceValueType.UINT8]}
|
||||||
|
</MenuItem>
|
||||||
|
<MenuItem value={DeviceValueType.INT16}>
|
||||||
|
{DeviceValueTypeNames[DeviceValueType.INT16]}
|
||||||
|
</MenuItem>
|
||||||
|
<MenuItem value={DeviceValueType.UINT16}>
|
||||||
|
{DeviceValueTypeNames[DeviceValueType.UINT16]}
|
||||||
|
</MenuItem>
|
||||||
|
<MenuItem value={DeviceValueType.UINT24}>
|
||||||
|
{DeviceValueTypeNames[DeviceValueType.UINT24]}
|
||||||
|
</MenuItem>
|
||||||
|
<MenuItem value={DeviceValueType.TIME}>
|
||||||
|
{DeviceValueTypeNames[DeviceValueType.TIME]}
|
||||||
|
</MenuItem>
|
||||||
|
<MenuItem value={DeviceValueType.UINT32}>
|
||||||
|
{DeviceValueTypeNames[DeviceValueType.UINT32]}
|
||||||
|
</MenuItem>
|
||||||
|
<MenuItem value={DeviceValueType.STRING}>
|
||||||
|
{DeviceValueTypeNames[DeviceValueType.STRING]}
|
||||||
|
</MenuItem>
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
@@ -351,7 +310,11 @@ const CustomEntitiesDialog = ({
|
|||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
select
|
select
|
||||||
>
|
>
|
||||||
{uomMenuItems}
|
{DeviceValueUOM_s.map((val, i) => (
|
||||||
|
<MenuItem key={val} value={i}>
|
||||||
|
{val}
|
||||||
|
</MenuItem>
|
||||||
|
))}
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
</>
|
</>
|
||||||
@@ -360,7 +323,7 @@ const CustomEntitiesDialog = ({
|
|||||||
editItem.device_id !== '0' && (
|
editItem.device_id !== '0' && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="factor"
|
name="factor"
|
||||||
label={LL.BYTES()}
|
label={LL.BYTES()}
|
||||||
value={numberValue(editItem.factor as number)}
|
value={numberValue(editItem.factor as number)}
|
||||||
@@ -378,7 +341,7 @@ const CustomEntitiesDialog = ({
|
|||||||
{editItem.value_type === DeviceValueType.BOOL && (
|
{editItem.value_type === DeviceValueType.BOOL && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="factor"
|
name="factor"
|
||||||
label={LL.BITMASK()}
|
label={LL.BITMASK()}
|
||||||
value={editItem.factor as string}
|
value={editItem.factor as string}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useEffect, useMemo, useState } from 'react';
|
import { useCallback, useEffect, useState } from 'react';
|
||||||
import { useBlocker, useLocation } from 'react-router';
|
import { useBlocker, useLocation } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
@@ -16,7 +16,7 @@ import {
|
|||||||
DialogActions,
|
DialogActions,
|
||||||
DialogContent,
|
DialogContent,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
Grid,
|
Grid2 as Grid,
|
||||||
InputAdornment,
|
InputAdornment,
|
||||||
Link,
|
Link,
|
||||||
MenuItem,
|
MenuItem,
|
||||||
@@ -62,24 +62,7 @@ import OptionIcon from './OptionIcon';
|
|||||||
import { DeviceEntityMask } from './types';
|
import { DeviceEntityMask } from './types';
|
||||||
import type { APIcall, Device, DeviceEntity } from './types';
|
import type { APIcall, Device, DeviceEntity } from './types';
|
||||||
|
|
||||||
export const APIURL = `${window.location.origin}/api/`;
|
export const APIURL = window.location.origin + '/api/';
|
||||||
|
|
||||||
const MAX_BUFFER_SIZE = 2000;
|
|
||||||
|
|
||||||
// Helper function to create masked entity ID - extracted to avoid duplication
|
|
||||||
const createMaskedEntityId = (de: DeviceEntity): string => {
|
|
||||||
const maskHex = de.m.toString(16).padStart(2, '0');
|
|
||||||
const hasCustomizations = !!(de.cn || de.mi || de.ma);
|
|
||||||
const customizations = [
|
|
||||||
de.cn || '',
|
|
||||||
de.mi ? `>${de.mi}` : '',
|
|
||||||
de.ma ? `<${de.ma}` : ''
|
|
||||||
]
|
|
||||||
.filter(Boolean)
|
|
||||||
.join('');
|
|
||||||
|
|
||||||
return `${maskHex}${de.id}${hasCustomizations ? `|${customizations}` : ''}`;
|
|
||||||
};
|
|
||||||
|
|
||||||
const Customizations = () => {
|
const Customizations = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
@@ -111,14 +94,13 @@ const Customizations = () => {
|
|||||||
const [selectedDeviceTypeNameURL, setSelectedDeviceTypeNameURL] =
|
const [selectedDeviceTypeNameURL, setSelectedDeviceTypeNameURL] =
|
||||||
useState<string>(''); // needed for API URL
|
useState<string>(''); // needed for API URL
|
||||||
const [selectedDeviceName, setSelectedDeviceName] = useState<string>('');
|
const [selectedDeviceName, setSelectedDeviceName] = useState<string>('');
|
||||||
const [selectedDeviceBrand, setSelectedDeviceBrand] = useState<string>('');
|
|
||||||
|
|
||||||
const { send: sendResetCustomizations } = useRequest(resetCustomizations(), {
|
const { send: sendResetCustomizations } = useRequest(resetCustomizations(), {
|
||||||
immediate: false
|
immediate: false
|
||||||
});
|
});
|
||||||
|
|
||||||
const { send: sendDeviceName } = useRequest(
|
const { send: sendDeviceName } = useRequest(
|
||||||
(data: { id: number; name: string; brand: string }) => writeDeviceName(data),
|
(data: { id: number; name: string }) => writeDeviceName(data),
|
||||||
{
|
{
|
||||||
immediate: false
|
immediate: false
|
||||||
}
|
}
|
||||||
@@ -143,22 +125,13 @@ const Customizations = () => {
|
|||||||
|
|
||||||
const setOriginalSettings = (data: DeviceEntity[]) => {
|
const setOriginalSettings = (data: DeviceEntity[]) => {
|
||||||
setDeviceEntities(
|
setDeviceEntities(
|
||||||
data.map((de) => {
|
data.map((de) => ({
|
||||||
const result: DeviceEntity = {
|
...de,
|
||||||
...de,
|
o_m: de.m,
|
||||||
o_m: de.m
|
o_cn: de.cn,
|
||||||
};
|
o_mi: de.mi,
|
||||||
if (de.cn !== undefined) {
|
o_ma: de.ma
|
||||||
result.o_cn = de.cn;
|
}))
|
||||||
}
|
|
||||||
if (de.mi !== undefined) {
|
|
||||||
result.o_mi = de.mi;
|
|
||||||
}
|
|
||||||
if (de.ma !== undefined) {
|
|
||||||
result.o_ma = de.ma;
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
})
|
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -171,19 +144,17 @@ const Customizations = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const entities_theme = useMemo(
|
const entities_theme = useTheme({
|
||||||
() =>
|
Table: `
|
||||||
useTheme({
|
|
||||||
Table: `
|
|
||||||
--data-table-library_grid-template-columns: 156px repeat(1, minmax(80px, 1fr)) 45px minmax(45px, auto) minmax(120px, auto);
|
--data-table-library_grid-template-columns: 156px repeat(1, minmax(80px, 1fr)) 45px minmax(45px, auto) minmax(120px, auto);
|
||||||
`,
|
`,
|
||||||
BaseRow: `
|
BaseRow: `
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
.td {
|
.td {
|
||||||
height: 32px;
|
height: 32px;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
BaseCell: `
|
BaseCell: `
|
||||||
&:nth-of-type(3) {
|
&:nth-of-type(3) {
|
||||||
text-align: right;
|
text-align: right;
|
||||||
}
|
}
|
||||||
@@ -194,7 +165,7 @@ const Customizations = () => {
|
|||||||
text-align: right;
|
text-align: right;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
HeaderRow: `
|
HeaderRow: `
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
background-color: black;
|
background-color: black;
|
||||||
color: #90CAF9;
|
color: #90CAF9;
|
||||||
@@ -206,7 +177,7 @@ const Customizations = () => {
|
|||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
Row: `
|
Row: `
|
||||||
background-color: #1e1e1e;
|
background-color: #1e1e1e;
|
||||||
position: relative;
|
position: relative;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
@@ -222,7 +193,7 @@ const Customizations = () => {
|
|||||||
background-color: #177ac9;
|
background-color: #177ac9;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
Cell: `
|
Cell: `
|
||||||
&:nth-of-type(2) {
|
&:nth-of-type(2) {
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
}
|
}
|
||||||
@@ -236,9 +207,7 @@ const Customizations = () => {
|
|||||||
padding-right: 8px;
|
padding-right: 8px;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
}),
|
});
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
function hasEntityChanged(de: DeviceEntity) {
|
function hasEntityChanged(de: DeviceEntity) {
|
||||||
return (
|
return (
|
||||||
@@ -251,8 +220,19 @@ const Customizations = () => {
|
|||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (deviceEntities.length) {
|
if (deviceEntities.length) {
|
||||||
const changedEntities = deviceEntities.filter((de) => hasEntityChanged(de));
|
setNumChanges(
|
||||||
setNumChanges(changedEntities.length);
|
deviceEntities
|
||||||
|
.filter((de) => hasEntityChanged(de))
|
||||||
|
.map(
|
||||||
|
(new_de) =>
|
||||||
|
new_de.m.toString(16).padStart(2, '0') +
|
||||||
|
new_de.id +
|
||||||
|
(new_de.cn || new_de.mi || new_de.ma ? '|' : '') +
|
||||||
|
(new_de.cn ? new_de.cn : '') +
|
||||||
|
(new_de.mi ? '>' + new_de.mi : '') +
|
||||||
|
(new_de.ma ? '<' + new_de.ma : '')
|
||||||
|
).length
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}, [deviceEntities]);
|
}, [deviceEntities]);
|
||||||
|
|
||||||
@@ -264,12 +244,8 @@ const Customizations = () => {
|
|||||||
setSelectedDevice(-1);
|
setSelectedDevice(-1);
|
||||||
setSelectedDeviceTypeNameURL('');
|
setSelectedDeviceTypeNameURL('');
|
||||||
} else {
|
} else {
|
||||||
const device = devices.devices[index];
|
setSelectedDeviceTypeNameURL(devices.devices[index].url || '');
|
||||||
if (device) {
|
setSelectedDeviceName(devices.devices[index].n);
|
||||||
setSelectedDeviceTypeNameURL(device.url || '');
|
|
||||||
setSelectedDeviceName(device.n);
|
|
||||||
setSelectedDeviceBrand(device.b);
|
|
||||||
}
|
|
||||||
setNumChanges(0);
|
setNumChanges(0);
|
||||||
setRestartNeeded(false);
|
setRestartNeeded(false);
|
||||||
}
|
}
|
||||||
@@ -287,26 +263,18 @@ const Customizations = () => {
|
|||||||
return value as string;
|
return value as string;
|
||||||
}
|
}
|
||||||
|
|
||||||
const isCommand = useCallback((de: DeviceEntity) => {
|
const formatName = (de: DeviceEntity, withShortname: boolean) =>
|
||||||
return de.n && de.n[0] === '!';
|
(de.n && de.n[0] === '!'
|
||||||
}, []);
|
? de.t
|
||||||
|
? LL.COMMAND(1) + ': ' + de.t + ' ' + de.n.slice(1)
|
||||||
const formatName = useCallback(
|
: LL.COMMAND(1) + ': ' + de.n.slice(1)
|
||||||
(de: DeviceEntity, withShortname: boolean) => {
|
: de.cn && de.cn !== ''
|
||||||
let name: string;
|
? de.t
|
||||||
if (isCommand(de)) {
|
? de.t + ' ' + de.cn
|
||||||
name = de.t
|
: de.cn
|
||||||
? `${LL.COMMAND(1)}: ${de.t} ${de.n?.slice(1)}`
|
: de.t
|
||||||
: `${LL.COMMAND(1)}: ${de.n?.slice(1)}`;
|
? de.t + ' ' + de.n
|
||||||
} else if (de.cn && de.cn !== '') {
|
: de.n) + (withShortname ? ' ' + de.id : '');
|
||||||
name = de.t ? `${de.t} ${de.cn}` : de.cn;
|
|
||||||
} else {
|
|
||||||
name = de.t ? `${de.t} ${de.n}` : de.n || '';
|
|
||||||
}
|
|
||||||
return withShortname ? `${name} ${de.id}` : name;
|
|
||||||
},
|
|
||||||
[LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const getMaskNumber = (newMask: string[]) => {
|
const getMaskNumber = (newMask: string[]) => {
|
||||||
let new_mask = 0;
|
let new_mask = 0;
|
||||||
@@ -336,33 +304,34 @@ const Customizations = () => {
|
|||||||
return new_masks;
|
return new_masks;
|
||||||
};
|
};
|
||||||
|
|
||||||
const filter_entity = useCallback(
|
const filter_entity = (de: DeviceEntity) =>
|
||||||
(de: DeviceEntity) =>
|
(de.m & selectedFilters || !selectedFilters) &&
|
||||||
(de.m & selectedFilters || !selectedFilters) &&
|
formatName(de, true).includes(search);
|
||||||
formatName(de, true).toLowerCase().includes(search.toLowerCase()),
|
|
||||||
[selectedFilters, search, formatName]
|
|
||||||
);
|
|
||||||
|
|
||||||
const maskDisabled = useCallback(
|
const maskDisabled = (set: boolean) => {
|
||||||
(set: boolean) => {
|
setDeviceEntities(
|
||||||
setDeviceEntities((prev) =>
|
deviceEntities.map(function (de) {
|
||||||
prev.map((de) => {
|
if (filter_entity(de)) {
|
||||||
if (filter_entity(de)) {
|
return {
|
||||||
const excludeMask =
|
...de,
|
||||||
DeviceEntityMask.DV_API_MQTT_EXCLUDE | DeviceEntityMask.DV_WEB_EXCLUDE;
|
m: set
|
||||||
return {
|
? de.m |
|
||||||
...de,
|
(DeviceEntityMask.DV_API_MQTT_EXCLUDE |
|
||||||
m: set ? de.m | excludeMask : de.m & ~excludeMask
|
DeviceEntityMask.DV_WEB_EXCLUDE)
|
||||||
};
|
: de.m &
|
||||||
}
|
~(
|
||||||
|
DeviceEntityMask.DV_API_MQTT_EXCLUDE |
|
||||||
|
DeviceEntityMask.DV_WEB_EXCLUDE
|
||||||
|
)
|
||||||
|
};
|
||||||
|
} else {
|
||||||
return de;
|
return de;
|
||||||
})
|
}
|
||||||
);
|
})
|
||||||
},
|
);
|
||||||
[filter_entity]
|
};
|
||||||
);
|
|
||||||
|
|
||||||
const resetCustomization = useCallback(async () => {
|
const resetCustomization = async () => {
|
||||||
try {
|
try {
|
||||||
await sendResetCustomizations();
|
await sendResetCustomizations();
|
||||||
toast.info(LL.CUSTOMIZATIONS_RESTART());
|
toast.info(LL.CUSTOMIZATIONS_RESTART());
|
||||||
@@ -370,30 +339,25 @@ const Customizations = () => {
|
|||||||
toast.error((error as Error).message);
|
toast.error((error as Error).message);
|
||||||
} finally {
|
} finally {
|
||||||
setConfirmReset(false);
|
setConfirmReset(false);
|
||||||
setRestarting(true);
|
|
||||||
}
|
}
|
||||||
}, [sendResetCustomizations, LL]);
|
};
|
||||||
|
|
||||||
const onDialogClose = () => {
|
const onDialogClose = () => {
|
||||||
setDialogOpen(false);
|
setDialogOpen(false);
|
||||||
};
|
};
|
||||||
|
|
||||||
const updateDeviceEntity = useCallback((updatedItem: DeviceEntity) => {
|
const updateDeviceEntity = (updatedItem: DeviceEntity) => {
|
||||||
setDeviceEntities(
|
setDeviceEntities(
|
||||||
(prev) =>
|
deviceEntities?.map((de) =>
|
||||||
prev?.map((de) =>
|
de.id === updatedItem.id ? { ...de, ...updatedItem } : de
|
||||||
de.id === updatedItem.id ? { ...de, ...updatedItem } : de
|
)
|
||||||
) ?? []
|
|
||||||
);
|
);
|
||||||
}, []);
|
};
|
||||||
|
|
||||||
const onDialogSave = useCallback(
|
const onDialogSave = (updatedItem: DeviceEntity) => {
|
||||||
(updatedItem: DeviceEntity) => {
|
setDialogOpen(false);
|
||||||
setDialogOpen(false);
|
updateDeviceEntity(updatedItem);
|
||||||
updateDeviceEntity(updatedItem);
|
};
|
||||||
},
|
|
||||||
[updateDeviceEntity]
|
|
||||||
);
|
|
||||||
|
|
||||||
const editDeviceEntity = useCallback((de: DeviceEntity) => {
|
const editDeviceEntity = useCallback((de: DeviceEntity) => {
|
||||||
if (de.n === undefined || (de.n && de.n[0] === '!')) {
|
if (de.n === undefined || (de.n && de.n[0] === '!')) {
|
||||||
@@ -408,65 +372,54 @@ const Customizations = () => {
|
|||||||
setDialogOpen(true);
|
setDialogOpen(true);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const saveCustomization = useCallback(async () => {
|
const saveCustomization = async () => {
|
||||||
if (!devices || !deviceEntities || selectedDevice === -1) {
|
if (devices && deviceEntities && selectedDevice !== -1) {
|
||||||
return;
|
const masked_entities = deviceEntities
|
||||||
}
|
.filter((de: DeviceEntity) => hasEntityChanged(de))
|
||||||
|
.map(
|
||||||
|
(new_de) =>
|
||||||
|
new_de.m.toString(16).padStart(2, '0') +
|
||||||
|
new_de.id +
|
||||||
|
(new_de.cn || new_de.mi || new_de.ma ? '|' : '') +
|
||||||
|
(new_de.cn ? new_de.cn : '') +
|
||||||
|
(new_de.mi ? '>' + new_de.mi : '') +
|
||||||
|
(new_de.ma ? '<' + new_de.ma : '')
|
||||||
|
);
|
||||||
|
|
||||||
const masked_entities = deviceEntities
|
// check size in bytes to match buffer in CPP, which is 2048
|
||||||
.filter((de: DeviceEntity) => hasEntityChanged(de))
|
const bytes = new TextEncoder().encode(JSON.stringify(masked_entities)).length;
|
||||||
.map((new_de) => createMaskedEntityId(new_de));
|
if (bytes > 2000) {
|
||||||
|
toast.warning(LL.CUSTOMIZATIONS_FULL());
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// check size in bytes to match buffer in CPP, which is 2048
|
await sendCustomizationEntities({
|
||||||
const bytes = new TextEncoder().encode(JSON.stringify(masked_entities)).length;
|
id: selectedDevice,
|
||||||
if (bytes > MAX_BUFFER_SIZE) {
|
entity_ids: masked_entities
|
||||||
toast.warning(LL.CUSTOMIZATIONS_FULL());
|
}).catch((error: Error) => {
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
await sendCustomizationEntities({
|
|
||||||
id: selectedDevice,
|
|
||||||
entity_ids: masked_entities
|
|
||||||
})
|
|
||||||
.then(() => {
|
|
||||||
toast.success(LL.CUSTOMIZATIONS_SAVED());
|
|
||||||
})
|
|
||||||
.catch((error: Error) => {
|
|
||||||
if (error.message === 'Reboot required') {
|
if (error.message === 'Reboot required') {
|
||||||
setRestartNeeded(true);
|
setRestartNeeded(true);
|
||||||
} else {
|
} else {
|
||||||
toast.error(error.message);
|
toast.error(error.message);
|
||||||
}
|
}
|
||||||
})
|
|
||||||
.finally(() => {
|
|
||||||
setOriginalSettings(deviceEntities);
|
|
||||||
});
|
});
|
||||||
}, [devices, deviceEntities, selectedDevice, sendCustomizationEntities, LL]);
|
setOriginalSettings(deviceEntities);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const renameDevice = useCallback(async () => {
|
const renameDevice = async () => {
|
||||||
await sendDeviceName({
|
await sendDeviceName({ id: selectedDevice, name: selectedDeviceName })
|
||||||
id: selectedDevice,
|
|
||||||
name: selectedDeviceName,
|
|
||||||
brand: selectedDeviceBrand
|
|
||||||
})
|
|
||||||
.then(() => {
|
.then(() => {
|
||||||
toast.success(LL.UPDATED_OF(LL.NAME(1)));
|
toast.success(LL.UPDATED_OF(LL.NAME(1)));
|
||||||
})
|
})
|
||||||
.catch(() => {
|
.catch(() => {
|
||||||
toast.error(`${LL.UPDATE_OF(LL.NAME(1))} ${LL.FAILED(1)}`);
|
toast.error(LL.UPDATE_OF(LL.NAME(1)) + ' ' + LL.FAILED(1));
|
||||||
})
|
})
|
||||||
.finally(async () => {
|
.finally(async () => {
|
||||||
setRename(false);
|
setRename(false);
|
||||||
await fetchCoreData();
|
await fetchCoreData();
|
||||||
});
|
});
|
||||||
}, [
|
};
|
||||||
selectedDevice,
|
|
||||||
selectedDeviceName,
|
|
||||||
selectedDeviceBrand,
|
|
||||||
sendDeviceName,
|
|
||||||
LL,
|
|
||||||
fetchCoreData
|
|
||||||
]);
|
|
||||||
|
|
||||||
const renderDeviceList = () => (
|
const renderDeviceList = () => (
|
||||||
<>
|
<>
|
||||||
@@ -475,26 +428,15 @@ const Customizations = () => {
|
|||||||
</Box>
|
</Box>
|
||||||
<Box display="flex" flexWrap="wrap" alignItems="center" gap={2}>
|
<Box display="flex" flexWrap="wrap" alignItems="center" gap={2}>
|
||||||
{rename ? (
|
{rename ? (
|
||||||
<>
|
<TextField
|
||||||
<TextField
|
name="device"
|
||||||
name="device"
|
label={LL.EMS_DEVICE()}
|
||||||
label={LL.EMS_DEVICE()}
|
fullWidth
|
||||||
style={{ minWidth: '48%' }}
|
variant="outlined"
|
||||||
variant="outlined"
|
value={selectedDeviceName}
|
||||||
value={selectedDeviceName}
|
onChange={(e) => setSelectedDeviceName(e.target.value)}
|
||||||
onChange={(e) => setSelectedDeviceName(e.target.value)}
|
margin="normal"
|
||||||
margin="normal"
|
/>
|
||||||
/>
|
|
||||||
<TextField
|
|
||||||
name="brand"
|
|
||||||
label={LL.BRAND()}
|
|
||||||
style={{ minWidth: '48%' }}
|
|
||||||
variant="outlined"
|
|
||||||
value={selectedDeviceBrand}
|
|
||||||
onChange={(e) => setSelectedDeviceBrand(e.target.value)}
|
|
||||||
margin="normal"
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
) : (
|
) : (
|
||||||
<TextField
|
<TextField
|
||||||
name="device"
|
name="device"
|
||||||
@@ -540,38 +482,25 @@ const Customizations = () => {
|
|||||||
</Button>
|
</Button>
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
<>
|
<Button
|
||||||
<Button
|
startIcon={<EditIcon />}
|
||||||
startIcon={<EditIcon />}
|
variant="outlined"
|
||||||
variant="outlined"
|
onClick={() => setRename(true)}
|
||||||
onClick={() => setRename(true)}
|
>
|
||||||
>
|
{LL.RENAME()}
|
||||||
{LL.RENAME()}
|
</Button>
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
startIcon={<SettingsBackupRestoreIcon />}
|
|
||||||
variant="outlined"
|
|
||||||
color="error"
|
|
||||||
onClick={() => setConfirmReset(true)}
|
|
||||||
>
|
|
||||||
{LL.REMOVE_ALL()}
|
|
||||||
</Button>
|
|
||||||
</>
|
|
||||||
))}
|
))}
|
||||||
</Box>
|
</Box>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|
||||||
const filteredEntities = useMemo(
|
|
||||||
() => deviceEntities.filter((de) => filter_entity(de)),
|
|
||||||
[deviceEntities, filter_entity]
|
|
||||||
);
|
|
||||||
|
|
||||||
const renderDeviceData = () => {
|
const renderDeviceData = () => {
|
||||||
|
const shown_data = deviceEntities.filter((de) => filter_entity(de));
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Box color="warning.main">
|
<Box color="warning.main">
|
||||||
<Typography variant="body2" mt={1} mb={1}>
|
<Typography variant="body2" mt={1}>
|
||||||
<OptionIcon type="favorite" isSet={true} />={LL.CUSTOMIZATIONS_HELP_2()}
|
<OptionIcon type="favorite" isSet={true} />={LL.CUSTOMIZATIONS_HELP_2()}
|
||||||
|
|
||||||
<OptionIcon type="readonly" isSet={true} />={LL.CUSTOMIZATIONS_HELP_3()}
|
<OptionIcon type="readonly" isSet={true} />={LL.CUSTOMIZATIONS_HELP_3()}
|
||||||
@@ -597,7 +526,6 @@ const Customizations = () => {
|
|||||||
size="small"
|
size="small"
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
placeholder={LL.SEARCH()}
|
placeholder={LL.SEARCH()}
|
||||||
aria-label={LL.SEARCH()}
|
|
||||||
onChange={(event) => {
|
onChange={(event) => {
|
||||||
setSearch(event.target.value);
|
setSearch(event.target.value);
|
||||||
}}
|
}}
|
||||||
@@ -617,7 +545,7 @@ const Customizations = () => {
|
|||||||
size="small"
|
size="small"
|
||||||
color="secondary"
|
color="secondary"
|
||||||
value={getMaskString(selectedFilters)}
|
value={getMaskString(selectedFilters)}
|
||||||
onChange={(_, mask: string[]) => {
|
onChange={(event, mask: string[]) => {
|
||||||
setSelectedFilters(getMaskNumber(mask));
|
setSelectedFilters(getMaskNumber(mask));
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
@@ -666,13 +594,13 @@ const Customizations = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<Typography variant="subtitle2" color="grey">
|
<Typography variant="subtitle2" color="grey">
|
||||||
{LL.SHOWING()} {filteredEntities.length}/{deviceEntities.length}
|
{LL.SHOWING()} {shown_data.length}/{deviceEntities.length}
|
||||||
{LL.ENTITIES(deviceEntities.length)}
|
{LL.ENTITIES(deviceEntities.length)}
|
||||||
</Typography>
|
</Typography>
|
||||||
</Grid>
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Table
|
<Table
|
||||||
data={{ nodes: filteredEntities }}
|
data={{ nodes: shown_data }}
|
||||||
theme={entities_theme}
|
theme={entities_theme}
|
||||||
layout={{ custom: true }}
|
layout={{ custom: true }}
|
||||||
>
|
>
|
||||||
@@ -694,27 +622,14 @@ const Customizations = () => {
|
|||||||
<EntityMaskToggle onUpdate={updateDeviceEntity} de={de} />
|
<EntityMaskToggle onUpdate={updateDeviceEntity} de={de} />
|
||||||
</Cell>
|
</Cell>
|
||||||
<Cell>
|
<Cell>
|
||||||
<span
|
{formatName(de, false)} (
|
||||||
style={{
|
<Link
|
||||||
color:
|
target="_blank"
|
||||||
de.v === undefined && !isCommand(de) ? 'grey' : 'inherit'
|
href={APIURL + selectedDeviceTypeNameURL + '/' + de.id}
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
{formatName(de, false)} (
|
{de.id}
|
||||||
<Link
|
</Link>
|
||||||
style={{
|
)
|
||||||
color:
|
|
||||||
de.v === undefined && !isCommand(de)
|
|
||||||
? 'grey'
|
|
||||||
: 'primary'
|
|
||||||
}}
|
|
||||||
target="_blank"
|
|
||||||
href={APIURL + selectedDeviceTypeNameURL + '/' + de.id}
|
|
||||||
>
|
|
||||||
{de.id}
|
|
||||||
</Link>
|
|
||||||
)
|
|
||||||
</span>
|
|
||||||
</Cell>
|
</Cell>
|
||||||
<Cell>
|
<Cell>
|
||||||
{!(de.m & DeviceEntityMask.DV_READONLY) && formatValue(de.mi)}
|
{!(de.m & DeviceEntityMask.DV_READONLY) && formatValue(de.mi)}
|
||||||
@@ -739,7 +654,7 @@ const Customizations = () => {
|
|||||||
open={confirmReset}
|
open={confirmReset}
|
||||||
onClose={() => setConfirmReset(false)}
|
onClose={() => setConfirmReset(false)}
|
||||||
>
|
>
|
||||||
<DialogTitle>{LL.REMOVE_ALL()}</DialogTitle>
|
<DialogTitle>{LL.RESET(1)}</DialogTitle>
|
||||||
<DialogContent dividers>{LL.CUSTOMIZATIONS_RESET()}</DialogContent>
|
<DialogContent dividers>{LL.CUSTOMIZATIONS_RESET()}</DialogContent>
|
||||||
<DialogActions>
|
<DialogActions>
|
||||||
<Button
|
<Button
|
||||||
@@ -756,7 +671,7 @@ const Customizations = () => {
|
|||||||
onClick={resetCustomization}
|
onClick={resetCustomization}
|
||||||
color="error"
|
color="error"
|
||||||
>
|
>
|
||||||
{LL.REMOVE_ALL()}
|
{LL.RESET(0)}
|
||||||
</Button>
|
</Button>
|
||||||
</DialogActions>
|
</DialogActions>
|
||||||
</Dialog>
|
</Dialog>
|
||||||
@@ -767,9 +682,8 @@ const Customizations = () => {
|
|||||||
{devices && renderDeviceList()}
|
{devices && renderDeviceList()}
|
||||||
{selectedDevice !== -1 && !rename && renderDeviceData()}
|
{selectedDevice !== -1 && !rename && renderDeviceData()}
|
||||||
{restartNeeded ? (
|
{restartNeeded ? (
|
||||||
<MessageBox level="warning" message={LL.RESTART_TEXT(0)}>
|
<MessageBox my={2} level="warning" message={LL.RESTART_TEXT(0)}>
|
||||||
<Button
|
<Button
|
||||||
sx={{ ml: 2 }}
|
|
||||||
startIcon={<PowerSettingsNewIcon />}
|
startIcon={<PowerSettingsNewIcon />}
|
||||||
variant="contained"
|
variant="contained"
|
||||||
color="error"
|
color="error"
|
||||||
@@ -787,11 +701,7 @@ const Customizations = () => {
|
|||||||
startIcon={<CancelIcon />}
|
startIcon={<CancelIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="secondary"
|
color="secondary"
|
||||||
onClick={() => {
|
onClick={() => devices && sendDeviceEntities(selectedDevice)}
|
||||||
if (devices) {
|
|
||||||
void sendDeviceEntities(selectedDevice);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
{LL.CANCEL()}
|
{LL.CANCEL()}
|
||||||
</Button>
|
</Button>
|
||||||
@@ -806,18 +716,28 @@ const Customizations = () => {
|
|||||||
</ButtonRow>
|
</ButtonRow>
|
||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
|
{!rename && (
|
||||||
|
<ButtonRow mt={1}>
|
||||||
|
<Button
|
||||||
|
startIcon={<SettingsBackupRestoreIcon />}
|
||||||
|
variant="outlined"
|
||||||
|
color="error"
|
||||||
|
onClick={() => setConfirmReset(true)}
|
||||||
|
>
|
||||||
|
{LL.RESET(0)}
|
||||||
|
</Button>
|
||||||
|
</ButtonRow>
|
||||||
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
)}
|
)}
|
||||||
{renderResetDialog()}
|
{renderResetDialog()}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|
||||||
return restarting ? (
|
return (
|
||||||
<SystemMonitor />
|
|
||||||
) : (
|
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
{renderContent()}
|
{restarting ? <SystemMonitor /> : renderContent()}
|
||||||
{selectedDeviceEntity && (
|
{selectedDeviceEntity && (
|
||||||
<SettingsCustomizationsDialog
|
<SettingsCustomizationsDialog
|
||||||
open={dialogOpen}
|
open={dialogOpen}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useCallback, useEffect, useMemo, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import CloseIcon from '@mui/icons-material/Close';
|
import CloseIcon from '@mui/icons-material/Close';
|
||||||
@@ -10,7 +10,7 @@ import {
|
|||||||
DialogActions,
|
DialogActions,
|
||||||
DialogContent,
|
DialogContent,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
Grid,
|
Grid2 as Grid,
|
||||||
TextField,
|
TextField,
|
||||||
Typography
|
Typography
|
||||||
} from '@mui/material';
|
} from '@mui/material';
|
||||||
@@ -30,23 +30,6 @@ interface SettingsCustomizationsDialogProps {
|
|||||||
selectedItem: DeviceEntity;
|
selectedItem: DeviceEntity;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface LabelValueProps {
|
|
||||||
label: string;
|
|
||||||
value: React.ReactNode;
|
|
||||||
}
|
|
||||||
|
|
||||||
const LabelValue = memo(({ label, value }: LabelValueProps) => (
|
|
||||||
<Grid container direction="row">
|
|
||||||
<Typography variant="body2" color="warning.main">
|
|
||||||
{label}:
|
|
||||||
</Typography>
|
|
||||||
<Typography variant="body2">{value}</Typography>
|
|
||||||
</Grid>
|
|
||||||
));
|
|
||||||
LabelValue.displayName = 'LabelValue';
|
|
||||||
|
|
||||||
const ICON_SIZE = 16;
|
|
||||||
|
|
||||||
const CustomizationsDialog = ({
|
const CustomizationsDialog = ({
|
||||||
open,
|
open,
|
||||||
onClose,
|
onClose,
|
||||||
@@ -57,23 +40,12 @@ const CustomizationsDialog = ({
|
|||||||
const [editItem, setEditItem] = useState<DeviceEntity>(selectedItem);
|
const [editItem, setEditItem] = useState<DeviceEntity>(selectedItem);
|
||||||
const [error, setError] = useState<boolean>(false);
|
const [error, setError] = useState<boolean>(false);
|
||||||
|
|
||||||
const updateFormValue = useMemo(
|
const updateFormValue = updateValue(setEditItem);
|
||||||
() =>
|
|
||||||
updateValue(
|
|
||||||
setEditItem as unknown as React.Dispatch<
|
|
||||||
React.SetStateAction<Record<string, unknown>>
|
|
||||||
>
|
|
||||||
),
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
const isWriteableNumber = useMemo(
|
const isWriteableNumber =
|
||||||
() =>
|
typeof editItem.v === 'number' &&
|
||||||
typeof editItem.v === 'number' &&
|
editItem.w &&
|
||||||
editItem.w &&
|
!(editItem.m & DeviceEntityMask.DV_READONLY);
|
||||||
!(editItem.m & DeviceEntityMask.DV_READONLY),
|
|
||||||
[editItem.v, editItem.w, editItem.m]
|
|
||||||
);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
@@ -82,59 +54,63 @@ const CustomizationsDialog = ({
|
|||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const handleClose = useCallback(
|
const handleClose = (_event, reason: 'backdropClick' | 'escapeKeyDown') => {
|
||||||
(_event: React.SyntheticEvent, reason: 'backdropClick' | 'escapeKeyDown') => {
|
if (reason !== 'backdropClick') {
|
||||||
if (reason !== 'backdropClick') {
|
onClose();
|
||||||
onClose();
|
}
|
||||||
}
|
};
|
||||||
},
|
|
||||||
[onClose]
|
|
||||||
);
|
|
||||||
|
|
||||||
const save = useCallback(() => {
|
const save = () => {
|
||||||
if (
|
if (
|
||||||
isWriteableNumber &&
|
isWriteableNumber &&
|
||||||
editItem.mi &&
|
editItem.mi &&
|
||||||
editItem.ma &&
|
editItem.ma &&
|
||||||
editItem.mi > editItem.ma
|
editItem.mi > editItem?.ma
|
||||||
) {
|
) {
|
||||||
setError(true);
|
setError(true);
|
||||||
} else {
|
} else {
|
||||||
onSave(editItem);
|
onSave(editItem);
|
||||||
}
|
}
|
||||||
}, [isWriteableNumber, editItem, onSave]);
|
};
|
||||||
|
|
||||||
const updateDeviceEntity = useCallback((updatedItem: DeviceEntity) => {
|
const updateDeviceEntity = (updatedItem: DeviceEntity) => {
|
||||||
setEditItem((prev) => ({ ...prev, m: updatedItem.m }));
|
setEditItem({ ...editItem, m: updatedItem.m });
|
||||||
}, []);
|
};
|
||||||
|
|
||||||
const dialogTitle = useMemo(() => `${LL.EDIT()} ${LL.ENTITY()}`, [LL]);
|
|
||||||
|
|
||||||
const writeableIcon = useMemo(
|
|
||||||
() =>
|
|
||||||
editItem.w ? (
|
|
||||||
<DoneIcon color="success" sx={{ fontSize: ICON_SIZE }} />
|
|
||||||
) : (
|
|
||||||
<CloseIcon color="error" sx={{ fontSize: ICON_SIZE }} />
|
|
||||||
),
|
|
||||||
[editItem.w]
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
||||||
<DialogTitle>{dialogTitle}</DialogTitle>
|
<DialogTitle>{LL.EDIT() + ' ' + LL.ENTITY()}</DialogTitle>
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<LabelValue label={LL.ID_OF(LL.ENTITY())} value={editItem.id} />
|
<Grid container>
|
||||||
<LabelValue
|
<Typography variant="body2" color="warning.main">
|
||||||
label={`${LL.DEFAULT(1)} ${LL.ENTITY_NAME(1)}`}
|
{LL.ID_OF(LL.ENTITY())}:
|
||||||
value={editItem.n}
|
</Typography>
|
||||||
/>
|
<Typography variant="body2">{editItem.id}</Typography>
|
||||||
<LabelValue label={LL.WRITEABLE()} value={writeableIcon} />
|
</Grid>
|
||||||
|
|
||||||
|
<Grid container direction="row">
|
||||||
|
<Typography variant="body2" color="warning.main">
|
||||||
|
{LL.DEFAULT(1) + ' ' + LL.ENTITY_NAME(1)}:
|
||||||
|
</Typography>
|
||||||
|
<Typography variant="body2">{editItem.n}</Typography>
|
||||||
|
</Grid>
|
||||||
|
|
||||||
|
<Grid container direction="row">
|
||||||
|
<Typography variant="body2" color="warning.main">
|
||||||
|
{LL.WRITEABLE()}:
|
||||||
|
</Typography>
|
||||||
|
<Typography variant="body2">
|
||||||
|
{editItem.w ? (
|
||||||
|
<DoneIcon color="success" sx={{ fontSize: 16 }} />
|
||||||
|
) : (
|
||||||
|
<CloseIcon color="error" sx={{ fontSize: 16 }} />
|
||||||
|
)}
|
||||||
|
</Typography>
|
||||||
|
</Grid>
|
||||||
|
|
||||||
<Box mt={1} mb={2}>
|
<Box mt={1} mb={2}>
|
||||||
<EntityMaskToggle onUpdate={updateDeviceEntity} de={editItem} />
|
<EntityMaskToggle onUpdate={updateDeviceEntity} de={editItem} />
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
<Grid container spacing={2}>
|
<Grid container spacing={2}>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<TextField
|
||||||
@@ -170,14 +146,12 @@ const CustomizationsDialog = ({
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
{error && (
|
{error && (
|
||||||
<Typography variant="body2" color="error" mt={2}>
|
<Typography variant="body2" color="error" mt={2}>
|
||||||
Error: Check min and max values
|
Error: Check min and max values
|
||||||
</Typography>
|
</Typography>
|
||||||
)}
|
)}
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
|
|
||||||
<DialogActions>
|
<DialogActions>
|
||||||
<Button
|
<Button
|
||||||
startIcon={<CancelIcon />}
|
startIcon={<CancelIcon />}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useCallback, useContext, useEffect, useMemo, useState } from 'react';
|
import { useContext, useEffect, useState } from 'react';
|
||||||
import { IconContext } from 'react-icons/lib';
|
import { IconContext } from 'react-icons/lib';
|
||||||
import { Link } from 'react-router';
|
import { Link } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
@@ -14,7 +14,6 @@ import {
|
|||||||
IconButton,
|
IconButton,
|
||||||
ToggleButton,
|
ToggleButton,
|
||||||
ToggleButtonGroup,
|
ToggleButtonGroup,
|
||||||
Tooltip,
|
|
||||||
Typography
|
Typography
|
||||||
} from '@mui/material';
|
} from '@mui/material';
|
||||||
|
|
||||||
@@ -45,7 +44,7 @@ import {
|
|||||||
} from './types';
|
} from './types';
|
||||||
import { deviceValueItemValidation } from './validators';
|
import { deviceValueItemValidation } from './validators';
|
||||||
|
|
||||||
const Dashboard = memo(() => {
|
const Dashboard = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const { me } = useContext(AuthenticatedContext);
|
const { me } = useContext(AuthenticatedContext);
|
||||||
|
|
||||||
@@ -77,40 +76,35 @@ const Dashboard = memo(() => {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
const deviceValueDialogSave = useCallback(
|
const deviceValueDialogSave = async (devicevalue: DeviceValue) => {
|
||||||
async (devicevalue: DeviceValue) => {
|
if (!selectedDashboardItem) {
|
||||||
if (!selectedDashboardItem) {
|
return;
|
||||||
return;
|
}
|
||||||
}
|
const id = selectedDashboardItem.parentNode.id; // this is the parent ID
|
||||||
const id = selectedDashboardItem.parentNode.id; // this is the parent ID
|
await sendDeviceValue({ id, c: devicevalue.c ?? '', v: devicevalue.v })
|
||||||
await sendDeviceValue({ id, c: devicevalue.c ?? '', v: devicevalue.v })
|
.then(() => {
|
||||||
.then(() => {
|
toast.success(LL.WRITE_CMD_SENT());
|
||||||
toast.success(LL.WRITE_CMD_SENT());
|
})
|
||||||
})
|
.catch((error: Error) => {
|
||||||
.catch((error: Error) => {
|
toast.error(error.message);
|
||||||
toast.error(error.message);
|
})
|
||||||
})
|
.finally(() => {
|
||||||
.finally(() => {
|
setDeviceValueDialogOpen(false);
|
||||||
setDeviceValueDialogOpen(false);
|
setSelectedDashboardItem(undefined);
|
||||||
setSelectedDashboardItem(undefined);
|
});
|
||||||
});
|
};
|
||||||
},
|
|
||||||
[selectedDashboardItem, sendDeviceValue, LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const dashboard_theme = useMemo(
|
const dashboard_theme = useTheme({
|
||||||
() =>
|
Table: `
|
||||||
useTheme({
|
|
||||||
Table: `
|
|
||||||
--data-table-library_grid-template-columns: minmax(80px, auto) 120px 32px;
|
--data-table-library_grid-template-columns: minmax(80px, auto) 120px 32px;
|
||||||
`,
|
`,
|
||||||
BaseRow: `
|
BaseRow: `
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
.td {
|
.td {
|
||||||
height: 28px;
|
height: 28px;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
Row: `
|
Row: `
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
background-color: #1e1e1e;
|
background-color: #1e1e1e;
|
||||||
&:nth-of-type(odd) .td {
|
&:nth-of-type(odd) .td {
|
||||||
@@ -120,7 +114,7 @@ const Dashboard = memo(() => {
|
|||||||
background-color: #177ac9;
|
background-color: #177ac9;
|
||||||
},
|
},
|
||||||
`,
|
`,
|
||||||
BaseCell: `
|
BaseCell: `
|
||||||
&:nth-of-type(2) {
|
&:nth-of-type(2) {
|
||||||
text-align: right;
|
text-align: right;
|
||||||
}
|
}
|
||||||
@@ -128,14 +122,12 @@ const Dashboard = memo(() => {
|
|||||||
text-align: right;
|
text-align: right;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
}),
|
});
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
const tree = useTree(
|
const tree = useTree(
|
||||||
{ nodes: [...data.nodes] },
|
{ nodes: data.nodes },
|
||||||
{
|
{
|
||||||
onChange: () => {} // not used but needed
|
onChange: undefined // not used but needed
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
treeIcon: {
|
treeIcon: {
|
||||||
@@ -164,82 +156,65 @@ const Dashboard = memo(() => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const nodeIds = useMemo(
|
|
||||||
() => data.nodes.map((item: DashboardItem) => item.id),
|
|
||||||
[data.nodes]
|
|
||||||
);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
showAll
|
showAll
|
||||||
? tree.fns.onAddAll(nodeIds) // expand tree
|
? tree.fns.onAddAll(data.nodes.map((item: DashboardItem) => item.id)) // expand tree
|
||||||
: tree.fns.onRemoveAll(); // collapse tree
|
: tree.fns.onRemoveAll(); // collapse tree
|
||||||
}, [parentNodes]);
|
}, [parentNodes]);
|
||||||
|
|
||||||
const showType = useCallback(
|
const showType = (n?: string, t?: number) => {
|
||||||
(n?: string, t?: number) => {
|
// if we have a name show it
|
||||||
// if we have a name show it
|
if (n) {
|
||||||
if (n) {
|
return n;
|
||||||
return n;
|
}
|
||||||
|
if (t) {
|
||||||
|
// otherwise pick translation based on type
|
||||||
|
switch (t) {
|
||||||
|
case DeviceType.CUSTOM:
|
||||||
|
return LL.CUSTOM_ENTITIES(0);
|
||||||
|
case DeviceType.ANALOGSENSOR:
|
||||||
|
return LL.ANALOG_SENSORS();
|
||||||
|
case DeviceType.TEMPERATURESENSOR:
|
||||||
|
return LL.TEMP_SENSORS();
|
||||||
|
case DeviceType.SCHEDULER:
|
||||||
|
return LL.SCHEDULER();
|
||||||
|
default:
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
if (t) {
|
}
|
||||||
// otherwise pick translation based on type
|
return '';
|
||||||
switch (t) {
|
};
|
||||||
case DeviceType.CUSTOM:
|
|
||||||
return LL.CUSTOM_ENTITIES(0);
|
|
||||||
case DeviceType.ANALOGSENSOR:
|
|
||||||
return LL.ANALOG_SENSORS();
|
|
||||||
case DeviceType.TEMPERATURESENSOR:
|
|
||||||
return LL.TEMP_SENSORS();
|
|
||||||
case DeviceType.SCHEDULER:
|
|
||||||
return LL.SCHEDULER();
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return '';
|
|
||||||
},
|
|
||||||
[LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const showName = useCallback(
|
const showName = (di: DashboardItem) => {
|
||||||
(di: DashboardItem) => {
|
if (di.id < 100) {
|
||||||
if (di.id < 100) {
|
// if its a device (parent node) and has entities
|
||||||
// if its a device (parent node) and has entities
|
if (di.nodes?.length) {
|
||||||
if (di.nodes?.length) {
|
return (
|
||||||
return (
|
<span style={{ fontWeight: 'bold', fontSize: '14px' }}>
|
||||||
<span style={{ fontSize: '15px' }}>
|
<DeviceIcon type_id={di.t ?? 0} />
|
||||||
<DeviceIcon type_id={di.t ?? 0} />
|
{showType(di.n, di.t)}
|
||||||
{showType(di.n, di.t)}
|
<span style={{ color: 'lightblue' }}> ({di.nodes?.length})</span>
|
||||||
<span style={{ color: 'lightblue' }}> ({di.nodes?.length})</span>
|
</span>
|
||||||
</span>
|
);
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
if (di.dv) {
|
}
|
||||||
return <span>{di.dv.id.slice(2)}</span>;
|
if (di.dv) {
|
||||||
}
|
return <span>{di.dv.id.slice(2)}</span>;
|
||||||
return null;
|
}
|
||||||
},
|
};
|
||||||
[showType]
|
|
||||||
);
|
|
||||||
|
|
||||||
const hasMask = useCallback(
|
const hasMask = (id: string, mask: number) =>
|
||||||
(id: string, mask: number) => (parseInt(id.slice(0, 2), 16) & mask) === mask,
|
(parseInt(id.slice(0, 2), 16) & mask) === mask;
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
const editDashboardValue = useCallback(
|
const editDashboardValue = (di: DashboardItem) => {
|
||||||
(di: DashboardItem) => {
|
if (me.admin && di.dv?.c) {
|
||||||
if (me.admin && di.dv?.c) {
|
setSelectedDashboardItem(di);
|
||||||
setSelectedDashboardItem(di);
|
setDeviceValueDialogOpen(true);
|
||||||
setDeviceValueDialogOpen(true);
|
}
|
||||||
}
|
};
|
||||||
},
|
|
||||||
[me.admin]
|
|
||||||
);
|
|
||||||
|
|
||||||
const handleShowAll = (
|
const handleShowAll = (
|
||||||
_event: React.MouseEvent<HTMLElement>,
|
event: React.MouseEvent<HTMLElement>,
|
||||||
toggle: boolean | null
|
toggle: boolean | null
|
||||||
) => {
|
) => {
|
||||||
if (toggle !== null) {
|
if (toggle !== null) {
|
||||||
@@ -248,32 +223,19 @@ const Dashboard = memo(() => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const hasFavEntities = useMemo(
|
|
||||||
() => data.nodes.filter((item: DashboardItem) => item.id <= 90).length,
|
|
||||||
[data.nodes]
|
|
||||||
);
|
|
||||||
|
|
||||||
const renderContent = () => {
|
const renderContent = () => {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return (
|
return <FormLoader onRetry={fetchDashboard} errorMessage={error?.message} />;
|
||||||
<FormLoader onRetry={fetchDashboard} errorMessage={error?.message || ''} />
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const hasFavEntities = data.nodes.filter(
|
||||||
|
(item: DashboardItem) => item.id <= 90
|
||||||
|
).length;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{!data.connected && (
|
{!data.connected && (
|
||||||
<MessageBox level="error" message={LL.EMS_BUS_WARNING() + '.'}>
|
<MessageBox mb={2} level="error" message={LL.EMS_BUS_WARNING()} />
|
||||||
(
|
|
||||||
<Link
|
|
||||||
target="_blank"
|
|
||||||
to="https://docs.emsesp.org/Troubleshooting#ems-bus-is-not-connecting"
|
|
||||||
style={{ color: 'white' }}
|
|
||||||
>
|
|
||||||
{LL.ONLINE_HELP()}
|
|
||||||
</Link>
|
|
||||||
)
|
|
||||||
</MessageBox>
|
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{data.connected && data.nodes.length > 0 && !hasFavEntities && (
|
{data.connected && data.nodes.length > 0 && !hasFavEntities && (
|
||||||
@@ -293,121 +255,105 @@ const Dashboard = memo(() => {
|
|||||||
</MessageBox>
|
</MessageBox>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Box
|
{data.nodes.length > 0 && (
|
||||||
display="flex"
|
<>
|
||||||
justifyContent="flex-end"
|
<ToggleButtonGroup
|
||||||
flexWrap="nowrap"
|
color="primary"
|
||||||
whiteSpace="nowrap"
|
size="small"
|
||||||
>
|
value={showAll}
|
||||||
<ToggleButtonGroup
|
exclusive
|
||||||
size="small"
|
onChange={handleShowAll}
|
||||||
color="primary"
|
>
|
||||||
value={showAll}
|
<ButtonTooltip title={LL.ALLVALUES()} arrow>
|
||||||
exclusive
|
<ToggleButton value={true}>
|
||||||
onChange={handleShowAll}
|
<UnfoldMoreIcon sx={{ fontSize: 18 }} />
|
||||||
>
|
</ToggleButton>
|
||||||
<ButtonTooltip title={LL.ALLVALUES()}>
|
</ButtonTooltip>
|
||||||
<ToggleButton value={true}>
|
<ButtonTooltip title={LL.COMPACT()} arrow>
|
||||||
<UnfoldMoreIcon sx={{ fontSize: 18 }} />
|
<ToggleButton value={false}>
|
||||||
</ToggleButton>
|
<UnfoldLessIcon sx={{ fontSize: 18 }} />
|
||||||
|
</ToggleButton>
|
||||||
|
</ButtonTooltip>
|
||||||
|
</ToggleButtonGroup>
|
||||||
|
<ButtonTooltip title={LL.DASHBOARD_1()} arrow>
|
||||||
|
<HelpOutlineIcon color="primary" sx={{ ml: 1, fontSize: 20 }} />
|
||||||
</ButtonTooltip>
|
</ButtonTooltip>
|
||||||
<ButtonTooltip title={LL.COMPACT()}>
|
|
||||||
<ToggleButton value={false}>
|
|
||||||
<UnfoldLessIcon sx={{ fontSize: 18 }} />
|
|
||||||
</ToggleButton>
|
|
||||||
</ButtonTooltip>
|
|
||||||
</ToggleButtonGroup>
|
|
||||||
</Box>
|
|
||||||
|
|
||||||
{data.nodes.length > 0 ? (
|
<Box
|
||||||
<Box mt={1} justifyContent="center" flexDirection="column">
|
padding={1}
|
||||||
<IconContext.Provider
|
justifyContent="center"
|
||||||
value={{
|
flexDirection="column"
|
||||||
color: 'lightblue',
|
sx={{
|
||||||
size: '18',
|
borderRadius: 1,
|
||||||
style: { verticalAlign: 'middle' }
|
border: '1px solid grey'
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Table
|
<IconContext.Provider
|
||||||
data={{ nodes: data.nodes }}
|
value={{
|
||||||
theme={dashboard_theme}
|
color: 'lightblue',
|
||||||
layout={{ custom: true }}
|
size: '18',
|
||||||
tree={tree}
|
style: { verticalAlign: 'middle' }
|
||||||
>
|
|
||||||
{(tableList: DashboardItem[]) => (
|
|
||||||
<Body>
|
|
||||||
{tableList.map((di: DashboardItem) => (
|
|
||||||
<Row
|
|
||||||
key={di.id}
|
|
||||||
item={di}
|
|
||||||
onClick={() => editDashboardValue(di)}
|
|
||||||
>
|
|
||||||
{di.id > 99 ? (
|
|
||||||
<>
|
|
||||||
<Cell>{showName(di)}</Cell>
|
|
||||||
<Cell>
|
|
||||||
<ButtonTooltip
|
|
||||||
title={formatValue(LL, di.dv?.v, di.dv?.u)}
|
|
||||||
>
|
|
||||||
<span>{formatValue(LL, di.dv?.v, di.dv?.u)}</span>
|
|
||||||
</ButtonTooltip>
|
|
||||||
</Cell>
|
|
||||||
|
|
||||||
<Cell>
|
|
||||||
{me.admin &&
|
|
||||||
di.dv?.c &&
|
|
||||||
!hasMask(di.dv.id, DeviceEntityMask.DV_READONLY) && (
|
|
||||||
<IconButton
|
|
||||||
size="small"
|
|
||||||
aria-label={
|
|
||||||
LL.CHANGE_VALUE() + ' ' + LL.VALUE(0)
|
|
||||||
}
|
|
||||||
onClick={() => editDashboardValue(di)}
|
|
||||||
>
|
|
||||||
<EditIcon
|
|
||||||
color="primary"
|
|
||||||
sx={{ fontSize: 16 }}
|
|
||||||
/>
|
|
||||||
</IconButton>
|
|
||||||
)}
|
|
||||||
</Cell>
|
|
||||||
</>
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
<CellTree item={di}>{showName(di)}</CellTree>
|
|
||||||
<Cell />
|
|
||||||
<Cell />
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Row>
|
|
||||||
))}
|
|
||||||
</Body>
|
|
||||||
)}
|
|
||||||
</Table>
|
|
||||||
</IconContext.Provider>
|
|
||||||
</Box>
|
|
||||||
) : (
|
|
||||||
<Box
|
|
||||||
display="flex"
|
|
||||||
// justifyContent="flex-end"
|
|
||||||
// flexWrap="nowrap"
|
|
||||||
// whiteSpace="nowrap"
|
|
||||||
>
|
|
||||||
<Typography mt={1} color="warning.main" variant="body1">
|
|
||||||
no data
|
|
||||||
</Typography>
|
|
||||||
<Tooltip title={LL.DASHBOARD_1()}>
|
|
||||||
<HelpOutlineIcon
|
|
||||||
sx={{
|
|
||||||
ml: 1,
|
|
||||||
mt: 1,
|
|
||||||
fontSize: 20,
|
|
||||||
verticalAlign: 'middle'
|
|
||||||
}}
|
}}
|
||||||
color="primary"
|
>
|
||||||
/>
|
<Table
|
||||||
</Tooltip>
|
data={{ nodes: data.nodes }}
|
||||||
</Box>
|
theme={dashboard_theme}
|
||||||
|
layout={{ custom: true }}
|
||||||
|
tree={tree}
|
||||||
|
>
|
||||||
|
{(tableList: DashboardItem[]) => (
|
||||||
|
<Body>
|
||||||
|
{tableList.map((di: DashboardItem) => (
|
||||||
|
<Row
|
||||||
|
key={di.id}
|
||||||
|
item={di}
|
||||||
|
onClick={() => editDashboardValue(di)}
|
||||||
|
>
|
||||||
|
{di.id > 99 ? (
|
||||||
|
<>
|
||||||
|
<Cell>{showName(di)}</Cell>
|
||||||
|
<Cell>
|
||||||
|
<ButtonTooltip
|
||||||
|
title={formatValue(LL, di.dv?.v, di.dv?.u)}
|
||||||
|
>
|
||||||
|
<span>{formatValue(LL, di.dv?.v, di.dv?.u)}</span>
|
||||||
|
</ButtonTooltip>
|
||||||
|
</Cell>
|
||||||
|
|
||||||
|
<Cell>
|
||||||
|
{me.admin &&
|
||||||
|
di.dv?.c &&
|
||||||
|
!hasMask(
|
||||||
|
di.dv.id,
|
||||||
|
DeviceEntityMask.DV_READONLY
|
||||||
|
) && (
|
||||||
|
<IconButton
|
||||||
|
size="small"
|
||||||
|
onClick={() => editDashboardValue(di)}
|
||||||
|
>
|
||||||
|
<EditIcon
|
||||||
|
color="primary"
|
||||||
|
sx={{ fontSize: 16 }}
|
||||||
|
/>
|
||||||
|
</IconButton>
|
||||||
|
)}
|
||||||
|
</Cell>
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<CellTree item={di}>{showName(di)}</CellTree>
|
||||||
|
<Cell />
|
||||||
|
<Cell />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Row>
|
||||||
|
))}
|
||||||
|
</Body>
|
||||||
|
)}
|
||||||
|
</Table>
|
||||||
|
</IconContext.Provider>
|
||||||
|
</Box>
|
||||||
|
</>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
@@ -429,6 +375,6 @@ const Dashboard = memo(() => {
|
|||||||
)}
|
)}
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
});
|
};
|
||||||
|
|
||||||
export default Dashboard;
|
export default Dashboard;
|
||||||
|
|||||||
@@ -1,14 +1,13 @@
|
|||||||
import { memo } from 'react';
|
|
||||||
import { AiOutlineAlert, AiOutlineControl, AiOutlineGateway } from 'react-icons/ai';
|
import { AiOutlineAlert, AiOutlineControl, AiOutlineGateway } from 'react-icons/ai';
|
||||||
import { CgSmartHomeBoiler } from 'react-icons/cg';
|
import { CgSmartHomeBoiler } from 'react-icons/cg';
|
||||||
import { FaSolarPanel } from 'react-icons/fa';
|
import { FaSolarPanel } from 'react-icons/fa';
|
||||||
import { GiHeatHaze, GiTap } from 'react-icons/gi';
|
import { GiHeatHaze, GiTap } from 'react-icons/gi';
|
||||||
|
import { MdPlaylistAdd } from 'react-icons/md';
|
||||||
|
import { MdMoreTime } from 'react-icons/md';
|
||||||
import {
|
import {
|
||||||
MdMoreTime,
|
|
||||||
MdOutlineDevices,
|
MdOutlineDevices,
|
||||||
MdOutlinePool,
|
MdOutlinePool,
|
||||||
MdOutlineSensors,
|
MdOutlineSensors,
|
||||||
MdPlaylistAdd,
|
|
||||||
MdThermostatAuto
|
MdThermostatAuto
|
||||||
} from 'react-icons/md';
|
} from 'react-icons/md';
|
||||||
import { PiFan, PiGauge } from 'react-icons/pi';
|
import { PiFan, PiGauge } from 'react-icons/pi';
|
||||||
@@ -19,10 +18,9 @@ import type { SvgIconProps } from '@mui/material';
|
|||||||
|
|
||||||
import { DeviceType } from './types';
|
import { DeviceType } from './types';
|
||||||
|
|
||||||
const deviceIconLookup: Record<
|
const deviceIconLookup: {
|
||||||
DeviceType,
|
[key in DeviceType]: React.ComponentType<SvgIconProps> | undefined;
|
||||||
React.ComponentType<SvgIconProps> | null
|
} = {
|
||||||
> = {
|
|
||||||
[DeviceType.TEMPERATURESENSOR]: TiThermometer,
|
[DeviceType.TEMPERATURESENSOR]: TiThermometer,
|
||||||
[DeviceType.ANALOGSENSOR]: PiGauge,
|
[DeviceType.ANALOGSENSOR]: PiGauge,
|
||||||
[DeviceType.BOILER]: CgSmartHomeBoiler,
|
[DeviceType.BOILER]: CgSmartHomeBoiler,
|
||||||
@@ -41,19 +39,15 @@ const deviceIconLookup: Record<
|
|||||||
[DeviceType.POOL]: MdOutlinePool,
|
[DeviceType.POOL]: MdOutlinePool,
|
||||||
[DeviceType.CUSTOM]: MdPlaylistAdd,
|
[DeviceType.CUSTOM]: MdPlaylistAdd,
|
||||||
[DeviceType.UNKNOWN]: MdOutlineSensors,
|
[DeviceType.UNKNOWN]: MdOutlineSensors,
|
||||||
[DeviceType.SYSTEM]: null,
|
[DeviceType.SYSTEM]: undefined,
|
||||||
[DeviceType.SCHEDULER]: MdMoreTime,
|
[DeviceType.SCHEDULER]: MdMoreTime,
|
||||||
[DeviceType.GENERIC]: MdOutlineSensors,
|
[DeviceType.GENERIC]: MdOutlineSensors,
|
||||||
[DeviceType.VENTILATION]: PiFan
|
[DeviceType.VENTILATION]: PiFan
|
||||||
};
|
};
|
||||||
|
|
||||||
interface DeviceIconProps {
|
const DeviceIcon = ({ type_id }: { type_id: DeviceType }) => {
|
||||||
type_id: DeviceType;
|
|
||||||
}
|
|
||||||
|
|
||||||
const DeviceIcon = memo(({ type_id }: DeviceIconProps) => {
|
|
||||||
const Icon = deviceIconLookup[type_id];
|
const Icon = deviceIconLookup[type_id];
|
||||||
return Icon ? <Icon /> : null;
|
return Icon ? <Icon /> : null;
|
||||||
});
|
};
|
||||||
|
|
||||||
export default DeviceIcon;
|
export default DeviceIcon;
|
||||||
|
|||||||
@@ -1,14 +1,12 @@
|
|||||||
import {
|
import {
|
||||||
memo,
|
|
||||||
useCallback,
|
useCallback,
|
||||||
useContext,
|
useContext,
|
||||||
useEffect,
|
useEffect,
|
||||||
useLayoutEffect,
|
useLayoutEffect,
|
||||||
useMemo,
|
|
||||||
useState
|
useState
|
||||||
} from 'react';
|
} from 'react';
|
||||||
import { IconContext } from 'react-icons';
|
import { IconContext } from 'react-icons';
|
||||||
import { Link, useNavigate } from 'react-router';
|
import { useNavigate } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import CommentsDisabledOutlinedIcon from '@mui/icons-material/CommentsDisabledOutlined';
|
import CommentsDisabledOutlinedIcon from '@mui/icons-material/CommentsDisabledOutlined';
|
||||||
@@ -33,7 +31,7 @@ import {
|
|||||||
DialogActions,
|
DialogActions,
|
||||||
DialogContent,
|
DialogContent,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
Grid,
|
Grid2 as Grid,
|
||||||
IconButton,
|
IconButton,
|
||||||
InputAdornment,
|
InputAdornment,
|
||||||
List,
|
List,
|
||||||
@@ -77,7 +75,7 @@ import { DeviceEntityMask, DeviceType, DeviceValueUOM_s } from './types';
|
|||||||
import type { Device, DeviceValue } from './types';
|
import type { Device, DeviceValue } from './types';
|
||||||
import { deviceValueItemValidation } from './validators';
|
import { deviceValueItemValidation } from './validators';
|
||||||
|
|
||||||
const Devices = memo(() => {
|
const Devices = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const { me } = useContext(AuthenticatedContext);
|
const { me } = useContext(AuthenticatedContext);
|
||||||
|
|
||||||
@@ -93,7 +91,7 @@ const Devices = memo(() => {
|
|||||||
|
|
||||||
useLayoutTitle(LL.DEVICES());
|
useLayoutTitle(LL.DEVICES());
|
||||||
|
|
||||||
const { data: coreData, send: sendCoreData } = useRequest(readCoreData, {
|
const { data: coreData, send: sendCoreData } = useRequest(() => readCoreData(), {
|
||||||
initialData: {
|
initialData: {
|
||||||
connected: true,
|
connected: true,
|
||||||
devices: []
|
devices: []
|
||||||
@@ -118,36 +116,36 @@ const Devices = memo(() => {
|
|||||||
);
|
);
|
||||||
|
|
||||||
useLayoutEffect(() => {
|
useLayoutEffect(() => {
|
||||||
let raf = 0;
|
function updateSize() {
|
||||||
const updateSize = () => {
|
setSize([window.innerWidth, window.innerHeight]);
|
||||||
cancelAnimationFrame(raf);
|
}
|
||||||
raf = requestAnimationFrame(() => {
|
|
||||||
setSize([window.innerWidth, window.innerHeight]);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
window.addEventListener('resize', updateSize);
|
window.addEventListener('resize', updateSize);
|
||||||
updateSize();
|
updateSize();
|
||||||
return () => {
|
return () => window.removeEventListener('resize', updateSize);
|
||||||
window.removeEventListener('resize', updateSize);
|
|
||||||
cancelAnimationFrame(raf);
|
|
||||||
};
|
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const leftOffset = useCallback(() => {
|
const leftOffset = () => {
|
||||||
const devicesWindow = document.getElementById('devices-window');
|
const devicesWindow = document.getElementById('devices-window');
|
||||||
if (!devicesWindow) return 0;
|
if (!devicesWindow) {
|
||||||
const { left, right } = devicesWindow.getBoundingClientRect();
|
return 0;
|
||||||
if (!left || !right) return 0;
|
}
|
||||||
return left + (right - left < 400 ? 0 : 200);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const common_theme = useMemo(
|
const clientRect = devicesWindow.getBoundingClientRect();
|
||||||
() =>
|
const left = clientRect.left;
|
||||||
useTheme({
|
const right = clientRect.right;
|
||||||
BaseRow: `
|
|
||||||
|
if (!left || !right) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
return left + (right - left < 400 ? 0 : 200);
|
||||||
|
};
|
||||||
|
|
||||||
|
const common_theme = useTheme({
|
||||||
|
BaseRow: `
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
`,
|
`,
|
||||||
HeaderRow: `
|
HeaderRow: `
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
background-color: black;
|
background-color: black;
|
||||||
color: #90CAF9;
|
color: #90CAF9;
|
||||||
@@ -155,7 +153,7 @@ const Devices = memo(() => {
|
|||||||
border-bottom: 1px solid #565656;
|
border-bottom: 1px solid #565656;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
Row: `
|
Row: `
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
background-color: #1E1E1E;
|
background-color: #1E1E1E;
|
||||||
.td {
|
.td {
|
||||||
@@ -165,47 +163,30 @@ const Devices = memo(() => {
|
|||||||
background-color: #177ac9;
|
background-color: #177ac9;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
}),
|
});
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
const device_theme = useMemo(
|
const device_theme = useTheme([
|
||||||
() =>
|
common_theme,
|
||||||
useTheme([
|
{
|
||||||
common_theme,
|
Table: `
|
||||||
{
|
|
||||||
BaseRow: `
|
|
||||||
font-size: 15px;
|
|
||||||
.td {
|
|
||||||
height: 28px;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
Table: `
|
|
||||||
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) 130px;
|
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) 130px;
|
||||||
`,
|
`,
|
||||||
HeaderRow: `
|
HeaderRow: `
|
||||||
.th {
|
.th {
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
`,
|
`,
|
||||||
Row: `
|
Row: `
|
||||||
&:nth-of-type(odd) .td {
|
font-weight: bold;
|
||||||
background-color: #303030;
|
|
||||||
},
|
|
||||||
&:hover .td {
|
&:hover .td {
|
||||||
background-color: #177ac9;
|
background-color: #177ac9;
|
||||||
},
|
|
||||||
`
|
`
|
||||||
}
|
}
|
||||||
]),
|
]);
|
||||||
[common_theme]
|
|
||||||
);
|
|
||||||
|
|
||||||
const data_theme = useMemo(
|
const data_theme = useTheme([
|
||||||
() =>
|
common_theme,
|
||||||
useTheme([
|
{
|
||||||
common_theme,
|
Table: `
|
||||||
{
|
|
||||||
Table: `
|
|
||||||
--data-table-library_grid-template-columns: minmax(200px, auto) minmax(150px, auto) 40px;
|
--data-table-library_grid-template-columns: minmax(200px, auto) minmax(150px, auto) 40px;
|
||||||
height: auto;
|
height: auto;
|
||||||
max-height: 100%;
|
max-height: 100%;
|
||||||
@@ -214,12 +195,12 @@ const Devices = memo(() => {
|
|||||||
display:none;
|
display:none;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
BaseRow: `
|
BaseRow: `
|
||||||
.td {
|
.td {
|
||||||
height: 32px;
|
height: 32px;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
BaseCell: `
|
BaseCell: `
|
||||||
&:nth-of-type(1) {
|
&:nth-of-type(1) {
|
||||||
border-left: 1px solid #177ac9;
|
border-left: 1px solid #177ac9;
|
||||||
},
|
},
|
||||||
@@ -230,12 +211,12 @@ const Devices = memo(() => {
|
|||||||
border-right: 1px solid #177ac9;
|
border-right: 1px solid #177ac9;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
HeaderRow: `
|
HeaderRow: `
|
||||||
.th {
|
.th {
|
||||||
border-top: 1px solid #565656;
|
border-top: 1px solid #565656;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
Row: `
|
Row: `
|
||||||
&:nth-of-type(odd) .td {
|
&:nth-of-type(odd) .td {
|
||||||
background-color: #303030;
|
background-color: #303030;
|
||||||
},
|
},
|
||||||
@@ -243,10 +224,8 @@ const Devices = memo(() => {
|
|||||||
background-color: #177ac9;
|
background-color: #177ac9;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
}
|
}
|
||||||
]),
|
]);
|
||||||
[common_theme]
|
|
||||||
);
|
|
||||||
|
|
||||||
const getSortIcon = (state: State, sortKey: unknown) => {
|
const getSortIcon = (state: State, sortKey: unknown) => {
|
||||||
if (state.sortKey === sortKey && state.reverse) {
|
if (state.sortKey === sortKey && state.reverse) {
|
||||||
@@ -259,7 +238,7 @@ const Devices = memo(() => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const dv_sort = useSort(
|
const dv_sort = useSort(
|
||||||
{ nodes: [...deviceData.nodes] },
|
{ nodes: deviceData.nodes },
|
||||||
{},
|
{},
|
||||||
{
|
{
|
||||||
sortIcon: {
|
sortIcon: {
|
||||||
@@ -289,7 +268,7 @@ const Devices = memo(() => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const device_select = useRowSelect(
|
const device_select = useRowSelect(
|
||||||
{ nodes: [...coreData.devices] },
|
{ nodes: coreData.devices },
|
||||||
{
|
{
|
||||||
onChange: onSelectChange
|
onChange: onSelectChange
|
||||||
}
|
}
|
||||||
@@ -345,23 +324,18 @@ const Devices = memo(() => {
|
|||||||
return sc;
|
return sc;
|
||||||
};
|
};
|
||||||
|
|
||||||
const hasMask = useCallback(
|
const hasMask = (id: string, mask: number) =>
|
||||||
(id: string, mask: number) => (parseInt(id.slice(0, 2), 16) & mask) === mask,
|
(parseInt(id.slice(0, 2), 16) & mask) === mask;
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
const handleDownloadCsv = () => {
|
const handleDownloadCsv = () => {
|
||||||
const deviceIndex = coreData.devices.findIndex(
|
const deviceIndex = coreData.devices.findIndex(
|
||||||
(d: Device) => d.id === device_select.state.id
|
(d) => d.id === device_select.state.id
|
||||||
);
|
);
|
||||||
if (deviceIndex === -1) {
|
if (deviceIndex === -1) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const selectedDevice = coreData.devices[deviceIndex];
|
const filename =
|
||||||
if (!selectedDevice) {
|
coreData.devices[deviceIndex].tn + '_' + coreData.devices[deviceIndex].n;
|
||||||
return;
|
|
||||||
}
|
|
||||||
const filename = selectedDevice.tn + '_' + selectedDevice.n;
|
|
||||||
|
|
||||||
const columns = [
|
const columns = [
|
||||||
{
|
{
|
||||||
@@ -376,7 +350,7 @@ const Devices = memo(() => {
|
|||||||
{
|
{
|
||||||
accessor: (dv: DeviceValue) =>
|
accessor: (dv: DeviceValue) =>
|
||||||
dv.u !== undefined && DeviceValueUOM_s[dv.u]
|
dv.u !== undefined && DeviceValueUOM_s[dv.u]
|
||||||
? DeviceValueUOM_s[dv.u]?.replace(/[^a-zA-Z0-9]/g, '')
|
? DeviceValueUOM_s[dv.u].replace(/[^a-zA-Z0-9]/g, '')
|
||||||
: '',
|
: '',
|
||||||
name: 'UoM'
|
name: 'UoM'
|
||||||
},
|
},
|
||||||
@@ -399,9 +373,7 @@ const Devices = memo(() => {
|
|||||||
];
|
];
|
||||||
|
|
||||||
const data = onlyFav
|
const data = onlyFav
|
||||||
? deviceData.nodes.filter((dv: DeviceValue) =>
|
? deviceData.nodes.filter((dv) => hasMask(dv.id, DeviceEntityMask.DV_FAVORITE))
|
||||||
hasMask(dv.id, DeviceEntityMask.DV_FAVORITE)
|
|
||||||
)
|
|
||||||
: deviceData.nodes;
|
: deviceData.nodes;
|
||||||
|
|
||||||
const csvData = data.reduce(
|
const csvData = data.reduce(
|
||||||
@@ -461,14 +433,10 @@ const Devices = memo(() => {
|
|||||||
const renderDeviceDetails = () => {
|
const renderDeviceDetails = () => {
|
||||||
if (showDeviceInfo) {
|
if (showDeviceInfo) {
|
||||||
const deviceIndex = coreData.devices.findIndex(
|
const deviceIndex = coreData.devices.findIndex(
|
||||||
(d: Device) => d.id === device_select.state.id
|
(d) => d.id === device_select.state.id
|
||||||
);
|
);
|
||||||
if (deviceIndex === -1) {
|
if (deviceIndex === -1) {
|
||||||
return null;
|
return;
|
||||||
}
|
|
||||||
const deviceDetails = coreData.devices[deviceIndex];
|
|
||||||
if (!deviceDetails) {
|
|
||||||
return null;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -481,35 +449,47 @@ const Devices = memo(() => {
|
|||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<List dense={true}>
|
<List dense={true}>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemText primary={LL.TYPE(0)} secondary={deviceDetails.tn} />
|
<ListItemText
|
||||||
|
primary={LL.TYPE(0)}
|
||||||
|
secondary={coreData.devices[deviceIndex].tn}
|
||||||
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemText primary={LL.NAME(0)} secondary={deviceDetails.n} />
|
<ListItemText
|
||||||
|
primary={LL.NAME(0)}
|
||||||
|
secondary={coreData.devices[deviceIndex].n}
|
||||||
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
{deviceDetails.t !== DeviceType.CUSTOM && (
|
{coreData.devices[deviceIndex].t !== DeviceType.CUSTOM && (
|
||||||
<>
|
<>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemText primary={LL.BRAND()} secondary={deviceDetails.b} />
|
<ListItemText
|
||||||
|
primary={LL.BRAND()}
|
||||||
|
secondary={coreData.devices[deviceIndex].b}
|
||||||
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemText
|
<ListItemText
|
||||||
primary={LL.ID_OF(LL.DEVICE())}
|
primary={LL.ID_OF(LL.DEVICE())}
|
||||||
secondary={
|
secondary={
|
||||||
'0x' +
|
'0x' +
|
||||||
('00' + deviceDetails.d.toString(16).toUpperCase()).slice(-2)
|
(
|
||||||
|
'00' +
|
||||||
|
coreData.devices[deviceIndex].d.toString(16).toUpperCase()
|
||||||
|
).slice(-2)
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemText
|
<ListItemText
|
||||||
primary={LL.ID_OF(LL.PRODUCT())}
|
primary={LL.ID_OF(LL.PRODUCT())}
|
||||||
secondary={deviceDetails.p}
|
secondary={coreData.devices[deviceIndex].p}
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemText
|
<ListItemText
|
||||||
primary={LL.VERSION()}
|
primary={LL.VERSION()}
|
||||||
secondary={deviceDetails.v}
|
secondary={coreData.devices[deviceIndex].v}
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
</>
|
</>
|
||||||
@@ -528,70 +508,59 @@ const Devices = memo(() => {
|
|||||||
</Dialog>
|
</Dialog>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return null;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const renderCoreData = () => (
|
const renderCoreData = () => (
|
||||||
<>
|
<>
|
||||||
{!coreData.connected ? (
|
<IconContext.Provider
|
||||||
<MessageBox level="error" message={LL.EMS_BUS_WARNING() + '.'}>
|
value={{
|
||||||
(
|
color: 'lightblue',
|
||||||
<Link
|
size: '18',
|
||||||
target="_blank"
|
style: { verticalAlign: 'middle' }
|
||||||
to="https://docs.emsesp.org/Troubleshooting#ems-bus-is-not-connecting"
|
}}
|
||||||
style={{ color: 'white' }}
|
>
|
||||||
|
{!coreData.connected && (
|
||||||
|
<MessageBox my={2} level="error" message={LL.EMS_BUS_WARNING()} />
|
||||||
|
)}
|
||||||
|
|
||||||
|
{coreData.connected && (
|
||||||
|
<Table
|
||||||
|
data={{ nodes: coreData.devices }}
|
||||||
|
select={device_select}
|
||||||
|
theme={device_theme}
|
||||||
|
layout={{ custom: true }}
|
||||||
>
|
>
|
||||||
{LL.ONLINE_HELP()}
|
{(tableList: Device[]) => (
|
||||||
</Link>
|
<>
|
||||||
)
|
<Header>
|
||||||
</MessageBox>
|
<HeaderRow>
|
||||||
) : (
|
<HeaderCell resize>{LL.DESCRIPTION()}</HeaderCell>
|
||||||
<Box justifyContent="center" flexDirection="column">
|
<HeaderCell stiff>{LL.TYPE(0)}</HeaderCell>
|
||||||
<IconContext.Provider
|
</HeaderRow>
|
||||||
value={{
|
</Header>
|
||||||
color: 'lightblue',
|
<Body>
|
||||||
size: '18',
|
{tableList.length === 0 && (
|
||||||
style: { verticalAlign: 'middle' }
|
<CircularProgress sx={{ margin: 1 }} size={18} />
|
||||||
}}
|
)}
|
||||||
>
|
{tableList.map((device: Device) => (
|
||||||
<Table
|
<Row key={device.id} item={device}>
|
||||||
data={{ nodes: [...coreData.devices] }}
|
<Cell>
|
||||||
select={device_select}
|
<DeviceIcon type_id={device.t} />
|
||||||
theme={device_theme}
|
|
||||||
layout={{ custom: true }}
|
{device.n}
|
||||||
>
|
<span style={{ color: 'lightblue' }}>
|
||||||
{(tableList: Device[]) => (
|
({device.e})
|
||||||
<>
|
</span>
|
||||||
<Header>
|
</Cell>
|
||||||
<HeaderRow>
|
<Cell stiff>{device.tn}</Cell>
|
||||||
<HeaderCell resize>{LL.DESCRIPTION()}</HeaderCell>
|
</Row>
|
||||||
<HeaderCell stiff>{LL.TYPE(0)}</HeaderCell>
|
))}
|
||||||
</HeaderRow>
|
</Body>
|
||||||
</Header>
|
</>
|
||||||
<Body>
|
)}
|
||||||
{tableList.length === 0 && (
|
</Table>
|
||||||
<CircularProgress sx={{ margin: 1 }} size={18} />
|
)}
|
||||||
)}
|
</IconContext.Provider>
|
||||||
{tableList.map((device: Device) => (
|
|
||||||
<Row key={device.id} item={device}>
|
|
||||||
<Cell>
|
|
||||||
<DeviceIcon type_id={device.t} />
|
|
||||||
|
|
||||||
{device.n}
|
|
||||||
<span style={{ color: 'lightblue' }}>
|
|
||||||
({device.e})
|
|
||||||
</span>
|
|
||||||
</Cell>
|
|
||||||
<Cell stiff>{device.tn}</Cell>
|
|
||||||
</Row>
|
|
||||||
))}
|
|
||||||
</Body>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Table>
|
|
||||||
</IconContext.Provider>
|
|
||||||
</Box>
|
|
||||||
)}
|
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -607,77 +576,64 @@ const Devices = memo(() => {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const showDeviceValue = useCallback((dv: DeviceValue) => {
|
const showDeviceValue = (dv: DeviceValue) => {
|
||||||
setSelectedDeviceValue(dv);
|
setSelectedDeviceValue(dv);
|
||||||
setDeviceValueDialogOpen(true);
|
setDeviceValueDialogOpen(true);
|
||||||
}, []);
|
};
|
||||||
|
|
||||||
const renderNameCell = useCallback(
|
const renderNameCell = (dv: DeviceValue) => (
|
||||||
(dv: DeviceValue) => (
|
<>
|
||||||
<>
|
{dv.id.slice(2)}
|
||||||
{dv.id.slice(2)}
|
{hasMask(dv.id, DeviceEntityMask.DV_FAVORITE) && (
|
||||||
{hasMask(dv.id, DeviceEntityMask.DV_FAVORITE) && (
|
<StarIcon color="primary" sx={{ fontSize: 12 }} />
|
||||||
<StarIcon color="primary" sx={{ fontSize: 12 }} />
|
)}
|
||||||
)}
|
{hasMask(dv.id, DeviceEntityMask.DV_READONLY) && (
|
||||||
{hasMask(dv.id, DeviceEntityMask.DV_READONLY) && (
|
<EditOffOutlinedIcon color="primary" sx={{ fontSize: 12 }} />
|
||||||
<EditOffOutlinedIcon color="primary" sx={{ fontSize: 12 }} />
|
)}
|
||||||
)}
|
{hasMask(dv.id, DeviceEntityMask.DV_API_MQTT_EXCLUDE) && (
|
||||||
{hasMask(dv.id, DeviceEntityMask.DV_API_MQTT_EXCLUDE) && (
|
<CommentsDisabledOutlinedIcon color="primary" sx={{ fontSize: 12 }} />
|
||||||
<CommentsDisabledOutlinedIcon color="primary" sx={{ fontSize: 12 }} />
|
)}
|
||||||
)}
|
</>
|
||||||
</>
|
|
||||||
),
|
|
||||||
[hasMask]
|
|
||||||
);
|
);
|
||||||
|
|
||||||
const shown_data = useMemo(() => {
|
const shown_data = onlyFav
|
||||||
if (onlyFav) {
|
? deviceData.nodes.filter(
|
||||||
return deviceData.nodes.filter(
|
(dv) =>
|
||||||
(dv: DeviceValue) =>
|
|
||||||
hasMask(dv.id, DeviceEntityMask.DV_FAVORITE) &&
|
hasMask(dv.id, DeviceEntityMask.DV_FAVORITE) &&
|
||||||
dv.id.slice(2).toLowerCase().includes(search.toLowerCase())
|
dv.id.slice(2).includes(search)
|
||||||
);
|
)
|
||||||
}
|
: deviceData.nodes.filter((dv) => dv.id.slice(2).includes(search));
|
||||||
return deviceData.nodes.filter((dv: DeviceValue) =>
|
|
||||||
dv.id.slice(2).toLowerCase().includes(search.toLowerCase())
|
|
||||||
);
|
|
||||||
}, [deviceData.nodes, onlyFav, search]);
|
|
||||||
|
|
||||||
const deviceIndex = coreData.devices.findIndex(
|
const deviceIndex = coreData.devices.findIndex(
|
||||||
(d: Device) => d.id === device_select.state.id
|
(d) => d.id === device_select.state.id
|
||||||
);
|
);
|
||||||
if (deviceIndex === -1) {
|
if (deviceIndex === -1) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const deviceInfo = coreData.devices[deviceIndex];
|
|
||||||
if (!deviceInfo) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const [, height] = size;
|
|
||||||
return (
|
return (
|
||||||
<Box
|
<Box
|
||||||
sx={{
|
sx={{
|
||||||
backgroundColor: 'black',
|
backgroundColor: 'black',
|
||||||
position: 'absolute',
|
position: 'absolute',
|
||||||
left: leftOffset,
|
left: () => leftOffset(),
|
||||||
right: 0,
|
right: 0,
|
||||||
bottom: 0,
|
bottom: 0,
|
||||||
top: 64,
|
top: 64,
|
||||||
zIndex: 'modal',
|
zIndex: 'modal',
|
||||||
maxHeight: () => (height || 0) - 126,
|
maxHeight: () => size[1] - 126,
|
||||||
border: '1px solid #177ac9'
|
border: '1px solid #177ac9'
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Box sx={{ p: 1 }}>
|
<Box sx={{ p: 1 }}>
|
||||||
<Grid container justifyContent="space-between">
|
<Grid container justifyContent="space-between">
|
||||||
<Typography noWrap variant="subtitle1" color="warning.main">
|
<Typography noWrap variant="subtitle1" color="warning.main">
|
||||||
{deviceInfo.n} (
|
{coreData.devices[deviceIndex].n} (
|
||||||
{deviceInfo.tn})
|
{coreData.devices[deviceIndex].tn})
|
||||||
</Typography>
|
</Typography>
|
||||||
<Grid justifyContent="flex-end">
|
<Grid justifyContent="flex-end">
|
||||||
<ButtonTooltip title={LL.CLOSE()}>
|
<ButtonTooltip title={LL.CLOSE()}>
|
||||||
<IconButton onClick={resetDeviceSelect} aria-label={LL.CLOSE()}>
|
<IconButton onClick={resetDeviceSelect}>
|
||||||
<HighlightOffIcon color="primary" sx={{ fontSize: 18 }} />
|
<HighlightOffIcon color="primary" sx={{ fontSize: 18 }} />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</ButtonTooltip>
|
</ButtonTooltip>
|
||||||
@@ -689,7 +645,6 @@ const Devices = memo(() => {
|
|||||||
variant="outlined"
|
variant="outlined"
|
||||||
sx={{ width: '22ch' }}
|
sx={{ width: '22ch' }}
|
||||||
placeholder={LL.SEARCH()}
|
placeholder={LL.SEARCH()}
|
||||||
aria-label={LL.SEARCH()}
|
|
||||||
onChange={(event) => {
|
onChange={(event) => {
|
||||||
setSearch(event.target.value);
|
setSearch(event.target.value);
|
||||||
}}
|
}}
|
||||||
@@ -704,22 +659,19 @@ const Devices = memo(() => {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<ButtonTooltip title={LL.DEVICE_DETAILS()}>
|
<ButtonTooltip title={LL.DEVICE_DETAILS()}>
|
||||||
<IconButton
|
<IconButton onClick={() => setShowDeviceInfo(true)}>
|
||||||
onClick={() => setShowDeviceInfo(true)}
|
|
||||||
aria-label={LL.DEVICE_DETAILS()}
|
|
||||||
>
|
|
||||||
<InfoOutlinedIcon color="primary" sx={{ fontSize: 18 }} />
|
<InfoOutlinedIcon color="primary" sx={{ fontSize: 18 }} />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</ButtonTooltip>
|
</ButtonTooltip>
|
||||||
{me.admin && (
|
{me.admin && (
|
||||||
<ButtonTooltip title={LL.CUSTOMIZATIONS()}>
|
<ButtonTooltip title={LL.CUSTOMIZATIONS()}>
|
||||||
<IconButton onClick={customize} aria-label={LL.CUSTOMIZATIONS()}>
|
<IconButton onClick={customize}>
|
||||||
<ConstructionIcon color="primary" sx={{ fontSize: 18 }} />
|
<ConstructionIcon color="primary" sx={{ fontSize: 18 }} />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</ButtonTooltip>
|
</ButtonTooltip>
|
||||||
)}
|
)}
|
||||||
<ButtonTooltip title={LL.EXPORT()}>
|
<ButtonTooltip title={LL.EXPORT()}>
|
||||||
<IconButton onClick={handleDownloadCsv} aria-label={LL.EXPORT()}>
|
<IconButton onClick={handleDownloadCsv}>
|
||||||
<DownloadIcon color="primary" sx={{ fontSize: 18 }} />
|
<DownloadIcon color="primary" sx={{ fontSize: 18 }} />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</ButtonTooltip>
|
</ButtonTooltip>
|
||||||
@@ -747,14 +699,14 @@ const Devices = memo(() => {
|
|||||||
' ' +
|
' ' +
|
||||||
shown_data.length +
|
shown_data.length +
|
||||||
'/' +
|
'/' +
|
||||||
deviceInfo.e +
|
coreData.devices[deviceIndex].e +
|
||||||
' ' +
|
' ' +
|
||||||
LL.ENTITIES(shown_data.length)}
|
LL.ENTITIES(shown_data.length)}
|
||||||
</span>
|
</span>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
<Table
|
<Table
|
||||||
data={{ nodes: Array.from(shown_data) }}
|
data={{ nodes: shown_data }}
|
||||||
theme={data_theme}
|
theme={data_theme}
|
||||||
sort={dv_sort}
|
sort={dv_sort}
|
||||||
layout={{ custom: true, fixedHeader: true }}
|
layout={{ custom: true, fixedHeader: true }}
|
||||||
@@ -838,6 +790,6 @@ const Devices = memo(() => {
|
|||||||
)}
|
)}
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
});
|
};
|
||||||
|
|
||||||
export default Devices;
|
export default Devices;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useEffect, useMemo, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
@@ -11,7 +11,7 @@ import {
|
|||||||
DialogContent,
|
DialogContent,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
FormHelperText,
|
FormHelperText,
|
||||||
Grid,
|
Grid2 as Grid,
|
||||||
InputAdornment,
|
InputAdornment,
|
||||||
MenuItem,
|
MenuItem,
|
||||||
TextField,
|
TextField,
|
||||||
@@ -52,7 +52,7 @@ const DevicesDialog = ({
|
|||||||
const [editItem, setEditItem] = useState<DeviceValue>(selectedItem);
|
const [editItem, setEditItem] = useState<DeviceValue>(selectedItem);
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
|
|
||||||
const updateFormValue = useMemo(() => updateValue(setEditItem), [setEditItem]);
|
const updateFormValue = updateValue(setEditItem);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
@@ -61,7 +61,11 @@ const DevicesDialog = ({
|
|||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const save = useCallback(async () => {
|
const close = () => {
|
||||||
|
onClose();
|
||||||
|
};
|
||||||
|
|
||||||
|
const save = async () => {
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(validator, editItem);
|
await validate(validator, editItem);
|
||||||
@@ -69,66 +73,46 @@ const DevicesDialog = ({
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
}, [validator, editItem, onSave]);
|
};
|
||||||
|
|
||||||
const setUom = useCallback(
|
const setUom = (uom?: DeviceValueUOM) => {
|
||||||
(uom?: DeviceValueUOM) => {
|
if (uom === undefined) {
|
||||||
if (uom === undefined) {
|
return;
|
||||||
return;
|
|
||||||
}
|
|
||||||
switch (uom) {
|
|
||||||
case DeviceValueUOM.HOURS:
|
|
||||||
return LL.HOURS();
|
|
||||||
case DeviceValueUOM.MINUTES:
|
|
||||||
return LL.MINUTES();
|
|
||||||
case DeviceValueUOM.SECONDS:
|
|
||||||
return LL.SECONDS();
|
|
||||||
default:
|
|
||||||
return DeviceValueUOM_s[uom];
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const showHelperText = useCallback((dv: DeviceValue) => {
|
|
||||||
if (dv.h) return dv.h;
|
|
||||||
if (dv.l) return dv.l.join(' | ');
|
|
||||||
if (dv.m !== undefined && dv.x !== undefined) {
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
{dv.m} → {dv.x}
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
return undefined;
|
switch (uom) {
|
||||||
}, []);
|
case DeviceValueUOM.HOURS:
|
||||||
|
return LL.HOURS();
|
||||||
|
case DeviceValueUOM.MINUTES:
|
||||||
|
return LL.MINUTES();
|
||||||
|
case DeviceValueUOM.SECONDS:
|
||||||
|
return LL.SECONDS();
|
||||||
|
default:
|
||||||
|
return DeviceValueUOM_s[uom];
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const isCommand = useMemo(
|
const showHelperText = (dv: DeviceValue) =>
|
||||||
() => selectedItem.v === '' && selectedItem.c,
|
dv.h ? (
|
||||||
[selectedItem.v, selectedItem.c]
|
dv.h
|
||||||
);
|
) : dv.l ? (
|
||||||
|
dv.l.join(' | ')
|
||||||
const dialogTitle = useMemo(() => {
|
) : dv.m !== undefined && dv.x !== undefined ? (
|
||||||
if (isCommand) return LL.RUN_COMMAND();
|
<>
|
||||||
return writeable ? LL.CHANGE_VALUE() : LL.VALUE(0);
|
{dv.m} → {dv.x}
|
||||||
}, [isCommand, writeable, LL]);
|
</>
|
||||||
|
) : undefined;
|
||||||
const buttonLabel = useMemo(() => {
|
|
||||||
return isCommand ? LL.EXECUTE() : LL.UPDATE();
|
|
||||||
}, [isCommand, LL]);
|
|
||||||
|
|
||||||
const helperText = useMemo(
|
|
||||||
() => showHelperText(editItem),
|
|
||||||
[editItem, showHelperText]
|
|
||||||
);
|
|
||||||
|
|
||||||
const valueLabel = LL.VALUE(0);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} open={open} onClose={onClose}>
|
<Dialog sx={dialogStyle} open={open} onClose={close}>
|
||||||
<DialogTitle>{dialogTitle}</DialogTitle>
|
<DialogTitle>
|
||||||
|
{selectedItem.v === '' && selectedItem.c
|
||||||
|
? LL.RUN_COMMAND()
|
||||||
|
: writeable
|
||||||
|
? LL.CHANGE_VALUE()
|
||||||
|
: LL.VALUE(0)}
|
||||||
|
</DialogTitle>
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<Box color="warning.main" mb={2}>
|
<Box color="warning.main" p={0} pl={0} pr={0} mt={0} mb={2}>
|
||||||
<Typography variant="body2">{editItem.id.slice(2)}</Typography>
|
<Typography variant="body2">{editItem.id.slice(2)}</Typography>
|
||||||
</Box>
|
</Box>
|
||||||
<Grid container>
|
<Grid container>
|
||||||
@@ -136,8 +120,8 @@ const DevicesDialog = ({
|
|||||||
{editItem.l ? (
|
{editItem.l ? (
|
||||||
<TextField
|
<TextField
|
||||||
name="v"
|
name="v"
|
||||||
|
label={LL.VALUE(0)}
|
||||||
value={editItem.v}
|
value={editItem.v}
|
||||||
aria-label={valueLabel}
|
|
||||||
disabled={!writeable}
|
disabled={!writeable}
|
||||||
sx={{ width: '30ch' }}
|
sx={{ width: '30ch' }}
|
||||||
select
|
select
|
||||||
@@ -151,9 +135,9 @@ const DevicesDialog = ({
|
|||||||
</TextField>
|
</TextField>
|
||||||
) : editItem.s || editItem.u !== DeviceValueUOM.NONE ? (
|
) : editItem.s || editItem.u !== DeviceValueUOM.NONE ? (
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="v"
|
name="v"
|
||||||
label={valueLabel}
|
label={LL.VALUE(0)}
|
||||||
value={numberValue(Math.round((editItem.v as number) * 10) / 10)}
|
value={numberValue(Math.round((editItem.v as number) * 10) / 10)}
|
||||||
autoFocus
|
autoFocus
|
||||||
disabled={!writeable}
|
disabled={!writeable}
|
||||||
@@ -175,9 +159,9 @@ const DevicesDialog = ({
|
|||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="v"
|
name="v"
|
||||||
label={valueLabel}
|
label={LL.VALUE(0)}
|
||||||
value={editItem.v}
|
value={editItem.v}
|
||||||
disabled={!writeable}
|
disabled={!writeable}
|
||||||
sx={{ width: '30ch' }}
|
sx={{ width: '30ch' }}
|
||||||
@@ -186,9 +170,9 @@ const DevicesDialog = ({
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</Grid>
|
</Grid>
|
||||||
{writeable && helperText && (
|
{writeable && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<FormHelperText>{helperText}</FormHelperText>
|
<FormHelperText>{showHelperText(editItem)}</FormHelperText>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
</Grid>
|
</Grid>
|
||||||
@@ -207,7 +191,7 @@ const DevicesDialog = ({
|
|||||||
<Button
|
<Button
|
||||||
startIcon={<CancelIcon />}
|
startIcon={<CancelIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={onClose}
|
onClick={close}
|
||||||
color="secondary"
|
color="secondary"
|
||||||
>
|
>
|
||||||
{LL.CANCEL()}
|
{LL.CANCEL()}
|
||||||
@@ -218,7 +202,7 @@ const DevicesDialog = ({
|
|||||||
onClick={save}
|
onClick={save}
|
||||||
color="primary"
|
color="primary"
|
||||||
>
|
>
|
||||||
{buttonLabel}
|
{selectedItem.v === '' && selectedItem.c ? LL.EXECUTE() : LL.UPDATE()}
|
||||||
</Button>
|
</Button>
|
||||||
{progress && (
|
{progress && (
|
||||||
<CircularProgress
|
<CircularProgress
|
||||||
@@ -233,7 +217,7 @@ const DevicesDialog = ({
|
|||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
) : (
|
) : (
|
||||||
<Button variant="outlined" onClick={onClose} color="secondary">
|
<Button variant="outlined" onClick={close} color="secondary">
|
||||||
{LL.CLOSE()}
|
{LL.CLOSE()}
|
||||||
</Button>
|
</Button>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
import { useCallback, useMemo } from 'react';
|
|
||||||
|
|
||||||
import { ToggleButton, ToggleButtonGroup } from '@mui/material';
|
import { ToggleButton, ToggleButtonGroup } from '@mui/material';
|
||||||
|
|
||||||
import OptionIcon from './OptionIcon';
|
import OptionIcon from './OptionIcon';
|
||||||
@@ -11,132 +9,92 @@ interface EntityMaskToggleProps {
|
|||||||
de: DeviceEntity;
|
de: DeviceEntity;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Available mask values
|
|
||||||
const MASK_VALUES = [
|
|
||||||
DeviceEntityMask.DV_WEB_EXCLUDE, // 1
|
|
||||||
DeviceEntityMask.DV_API_MQTT_EXCLUDE, // 2
|
|
||||||
DeviceEntityMask.DV_READONLY, // 4
|
|
||||||
DeviceEntityMask.DV_FAVORITE, // 8
|
|
||||||
DeviceEntityMask.DV_DELETED // 128
|
|
||||||
];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Converts an array of mask strings to a bitmask number
|
|
||||||
*/
|
|
||||||
const getMaskNumber = (newMask: string[]): number => {
|
|
||||||
return newMask.reduce((mask, entry) => mask | Number(entry), 0);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Converts a bitmask number to an array of mask strings
|
|
||||||
*/
|
|
||||||
const getMaskString = (mask: number): string[] => {
|
|
||||||
return MASK_VALUES.filter((value) => (mask & value) === value).map((value) =>
|
|
||||||
String(value)
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Checks if a specific mask bit is set
|
|
||||||
*/
|
|
||||||
const hasMask = (mask: number, flag: number): boolean => (mask & flag) === flag;
|
|
||||||
|
|
||||||
const EntityMaskToggle = ({ onUpdate, de }: EntityMaskToggleProps) => {
|
const EntityMaskToggle = ({ onUpdate, de }: EntityMaskToggleProps) => {
|
||||||
const handleChange = useCallback(
|
const getMaskNumber = (newMask: string[]) => {
|
||||||
(_event: unknown, mask: string[]) => {
|
let new_mask = 0;
|
||||||
// Convert selected masks to a number
|
for (const entry of newMask) {
|
||||||
const newMask = getMaskNumber(mask);
|
new_mask |= Number(entry);
|
||||||
const updatedDe = { ...de };
|
}
|
||||||
|
return new_mask;
|
||||||
|
};
|
||||||
|
|
||||||
// Apply business logic for mask interactions
|
const getMaskString = (m: number) => {
|
||||||
// If entity has no name and is set to readonly, also exclude from web
|
const new_masks: string[] = [];
|
||||||
if (updatedDe.n === '' && hasMask(newMask, DeviceEntityMask.DV_READONLY)) {
|
if ((m & 1) === 1) {
|
||||||
updatedDe.m = newMask | DeviceEntityMask.DV_WEB_EXCLUDE;
|
new_masks.push('1');
|
||||||
} else {
|
}
|
||||||
updatedDe.m = newMask;
|
if ((m & 2) === 2) {
|
||||||
}
|
new_masks.push('2');
|
||||||
|
}
|
||||||
// If excluded from web, cannot be favorite
|
if ((m & 4) === 4) {
|
||||||
if (hasMask(updatedDe.m, DeviceEntityMask.DV_WEB_EXCLUDE)) {
|
new_masks.push('4');
|
||||||
updatedDe.m = updatedDe.m & ~DeviceEntityMask.DV_FAVORITE;
|
}
|
||||||
}
|
if ((m & 8) === 8) {
|
||||||
|
new_masks.push('8');
|
||||||
onUpdate(updatedDe);
|
}
|
||||||
},
|
if ((m & 128) === 128) {
|
||||||
[de, onUpdate]
|
new_masks.push('128');
|
||||||
);
|
}
|
||||||
|
return new_masks;
|
||||||
// Memoize mask string value
|
};
|
||||||
const maskStringValue = useMemo(() => getMaskString(de.m), [de.m]);
|
|
||||||
|
|
||||||
// Memoize disabled states
|
|
||||||
const isFavoriteDisabled = useMemo(
|
|
||||||
() =>
|
|
||||||
hasMask(de.m, DeviceEntityMask.DV_WEB_EXCLUDE | DeviceEntityMask.DV_DELETED) ||
|
|
||||||
de.n === undefined,
|
|
||||||
[de.m, de.n]
|
|
||||||
);
|
|
||||||
|
|
||||||
const isReadonlyDisabled = useMemo(
|
|
||||||
() =>
|
|
||||||
!de.w ||
|
|
||||||
hasMask(de.m, DeviceEntityMask.DV_WEB_EXCLUDE | DeviceEntityMask.DV_FAVORITE),
|
|
||||||
[de.w, de.m]
|
|
||||||
);
|
|
||||||
|
|
||||||
const isApiMqttExcludeDisabled = useMemo(
|
|
||||||
() => de.n === '' || hasMask(de.m, DeviceEntityMask.DV_DELETED),
|
|
||||||
[de.n, de.m]
|
|
||||||
);
|
|
||||||
|
|
||||||
const isWebExcludeDisabled = useMemo(
|
|
||||||
() => de.n === undefined || hasMask(de.m, DeviceEntityMask.DV_DELETED),
|
|
||||||
[de.n, de.m]
|
|
||||||
);
|
|
||||||
|
|
||||||
// Memoize mask flag checks
|
|
||||||
const isFavoriteSet = useMemo(
|
|
||||||
() => hasMask(de.m, DeviceEntityMask.DV_FAVORITE),
|
|
||||||
[de.m]
|
|
||||||
);
|
|
||||||
const isReadonlySet = useMemo(
|
|
||||||
() => hasMask(de.m, DeviceEntityMask.DV_READONLY),
|
|
||||||
[de.m]
|
|
||||||
);
|
|
||||||
const isApiMqttExcludeSet = useMemo(
|
|
||||||
() => hasMask(de.m, DeviceEntityMask.DV_API_MQTT_EXCLUDE),
|
|
||||||
[de.m]
|
|
||||||
);
|
|
||||||
const isWebExcludeSet = useMemo(
|
|
||||||
() => hasMask(de.m, DeviceEntityMask.DV_WEB_EXCLUDE),
|
|
||||||
[de.m]
|
|
||||||
);
|
|
||||||
const isDeletedSet = useMemo(
|
|
||||||
() => hasMask(de.m, DeviceEntityMask.DV_DELETED),
|
|
||||||
[de.m]
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ToggleButtonGroup
|
<ToggleButtonGroup
|
||||||
size="small"
|
size="small"
|
||||||
color="secondary"
|
color="secondary"
|
||||||
value={maskStringValue}
|
value={getMaskString(de.m)}
|
||||||
onChange={handleChange}
|
onChange={(event, mask: string[]) => {
|
||||||
|
de.m = getMaskNumber(mask);
|
||||||
|
if (de.n === '' && de.m & DeviceEntityMask.DV_READONLY) {
|
||||||
|
de.m = de.m | DeviceEntityMask.DV_WEB_EXCLUDE;
|
||||||
|
}
|
||||||
|
if (de.m & DeviceEntityMask.DV_WEB_EXCLUDE) {
|
||||||
|
de.m = de.m & ~DeviceEntityMask.DV_FAVORITE;
|
||||||
|
}
|
||||||
|
onUpdate(de);
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
<ToggleButton value="8" disabled={isFavoriteDisabled}>
|
<ToggleButton value="8" disabled={(de.m & 0x81) !== 0 || de.n === undefined}>
|
||||||
<OptionIcon type="favorite" isSet={isFavoriteSet} />
|
<OptionIcon
|
||||||
|
type="favorite"
|
||||||
|
isSet={
|
||||||
|
(de.m & DeviceEntityMask.DV_FAVORITE) === DeviceEntityMask.DV_FAVORITE
|
||||||
|
}
|
||||||
|
/>
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value="4" disabled={isReadonlyDisabled}>
|
<ToggleButton value="4" disabled={!de.w || (de.m & 0x83) >= 3}>
|
||||||
<OptionIcon type="readonly" isSet={isReadonlySet} />
|
<OptionIcon
|
||||||
|
type="readonly"
|
||||||
|
isSet={
|
||||||
|
(de.m & DeviceEntityMask.DV_READONLY) === DeviceEntityMask.DV_READONLY
|
||||||
|
}
|
||||||
|
/>
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value="2" disabled={isApiMqttExcludeDisabled}>
|
<ToggleButton value="2" disabled={de.n === '' || (de.m & 0x80) !== 0}>
|
||||||
<OptionIcon type="api_mqtt_exclude" isSet={isApiMqttExcludeSet} />
|
<OptionIcon
|
||||||
|
type="api_mqtt_exclude"
|
||||||
|
isSet={
|
||||||
|
(de.m & DeviceEntityMask.DV_API_MQTT_EXCLUDE) ===
|
||||||
|
DeviceEntityMask.DV_API_MQTT_EXCLUDE
|
||||||
|
}
|
||||||
|
/>
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value="1" disabled={isWebExcludeDisabled}>
|
<ToggleButton value="1" disabled={de.n === undefined || (de.m & 0x80) !== 0}>
|
||||||
<OptionIcon type="web_exclude" isSet={isWebExcludeSet} />
|
<OptionIcon
|
||||||
|
type="web_exclude"
|
||||||
|
isSet={
|
||||||
|
(de.m & DeviceEntityMask.DV_WEB_EXCLUDE) ===
|
||||||
|
DeviceEntityMask.DV_WEB_EXCLUDE
|
||||||
|
}
|
||||||
|
/>
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value="128">
|
<ToggleButton value="128">
|
||||||
<OptionIcon type="deleted" isSet={isDeletedSet} />
|
<OptionIcon
|
||||||
|
type="deleted"
|
||||||
|
isSet={
|
||||||
|
(de.m & DeviceEntityMask.DV_DELETED) === DeviceEntityMask.DV_DELETED
|
||||||
|
}
|
||||||
|
/>
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</ToggleButtonGroup>
|
</ToggleButtonGroup>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
import { memo, useCallback, useContext, useMemo, useState } from 'react';
|
import { useContext, useState } from 'react';
|
||||||
import type { ReactElement } from 'react';
|
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import CommentIcon from '@mui/icons-material/CommentTwoTone';
|
import CommentIcon from '@mui/icons-material/CommentTwoTone';
|
||||||
@@ -20,7 +19,6 @@ import {
|
|||||||
Stack,
|
Stack,
|
||||||
Typography
|
Typography
|
||||||
} from '@mui/material';
|
} from '@mui/material';
|
||||||
import type { SxProps, Theme } from '@mui/material/styles';
|
|
||||||
|
|
||||||
import { useRequest } from 'alova/client';
|
import { useRequest } from 'alova/client';
|
||||||
import { SectionContent, useLayoutTitle } from 'components';
|
import { SectionContent, useLayoutTitle } from 'components';
|
||||||
@@ -31,61 +29,25 @@ import { saveFile } from 'utils';
|
|||||||
import { API, callAction } from '../../api/app';
|
import { API, callAction } from '../../api/app';
|
||||||
import type { APIcall } from './types';
|
import type { APIcall } from './types';
|
||||||
|
|
||||||
interface HelpLink {
|
const Help = () => {
|
||||||
href: string;
|
|
||||||
icon: ReactElement;
|
|
||||||
label: () => string;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface CustomSupport {
|
|
||||||
img_url: string | null;
|
|
||||||
html: string | null;
|
|
||||||
}
|
|
||||||
|
|
||||||
const DEFAULT_IMAGE_URL = 'https://emsesp.org/_media/images/installer.jpeg';
|
|
||||||
|
|
||||||
const SUPPORT_BOX_STYLES: SxProps<Theme> = {
|
|
||||||
borderRadius: 3,
|
|
||||||
border: '1px solid lightblue',
|
|
||||||
justifyContent: 'space-evenly',
|
|
||||||
alignItems: 'center'
|
|
||||||
};
|
|
||||||
|
|
||||||
const IMAGE_STYLES: SxProps<Theme> = {
|
|
||||||
maxHeight: { xs: 100, md: 250 }
|
|
||||||
};
|
|
||||||
|
|
||||||
const AVATAR_STYLES: SxProps<Theme> = {
|
|
||||||
bgcolor: '#72caf9'
|
|
||||||
};
|
|
||||||
|
|
||||||
const HelpComponent = () => {
|
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
useLayoutTitle(LL.HELP());
|
useLayoutTitle(LL.HELP());
|
||||||
|
|
||||||
const { me } = useContext(AuthenticatedContext);
|
const { me } = useContext(AuthenticatedContext);
|
||||||
|
|
||||||
const [customSupport, setCustomSupport] = useState<CustomSupport>({
|
const [customSupportIMG, setCustomSupportIMG] = useState<string | null>(null);
|
||||||
img_url: null,
|
const [customSupportHTML, setCustomSupportHTML] = useState<string | null>(null);
|
||||||
html: null
|
const [notFound, setNotFound] = useState<boolean>(false);
|
||||||
});
|
|
||||||
const [imgError, setImgError] = useState<boolean>(false);
|
|
||||||
|
|
||||||
// Memoize the request method to prevent re-creation on every render
|
useRequest(() => callAction({ action: 'getCustomSupport' })).onSuccess((event) => {
|
||||||
const getCustomSupportMethod = useMemo(
|
if (event && event.data && Object.keys(event.data).length !== 0) {
|
||||||
() => callAction({ action: 'getCustomSupport' }),
|
const data = event.data.Support;
|
||||||
[]
|
if (data.img_url) {
|
||||||
);
|
setCustomSupportIMG(data.img_url);
|
||||||
|
}
|
||||||
useRequest(getCustomSupportMethod).onSuccess((event) => {
|
if (data.html) {
|
||||||
if (event?.data && Object.keys(event.data).length !== 0) {
|
setCustomSupportHTML(data.html.join('<br/>'));
|
||||||
const { Support } = event.data as {
|
}
|
||||||
Support: { img_url?: string; html?: string[] };
|
|
||||||
};
|
|
||||||
setCustomSupport({
|
|
||||||
img_url: Support.img_url || null,
|
|
||||||
html: Support.html?.join('<br/>') || null
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -97,91 +59,93 @@ const HelpComponent = () => {
|
|||||||
toast.info(LL.DOWNLOAD_SUCCESSFUL());
|
toast.info(LL.DOWNLOAD_SUCCESSFUL());
|
||||||
})
|
})
|
||||||
.onError((error) => {
|
.onError((error) => {
|
||||||
toast.error(String(error.error?.message || 'An error occurred'));
|
toast.error(error.message);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Optimize API call memoization
|
|
||||||
const apiCall = useMemo(() => ({ device: 'system', cmd: 'info', id: 0 }), []);
|
|
||||||
|
|
||||||
const handleDownloadSystemInfo = useCallback(() => {
|
|
||||||
void sendAPI(apiCall);
|
|
||||||
}, [sendAPI, apiCall]);
|
|
||||||
|
|
||||||
const handleImageError = useCallback(() => {
|
|
||||||
setImgError(true);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
// Memoize help links to prevent recreation on every render
|
|
||||||
const helpLinks: HelpLink[] = useMemo(
|
|
||||||
() => [
|
|
||||||
{
|
|
||||||
href: 'https://emsesp.org',
|
|
||||||
icon: <MenuBookIcon />,
|
|
||||||
label: () => LL.HELP_INFORMATION_1()
|
|
||||||
},
|
|
||||||
{
|
|
||||||
href: 'https://discord.gg/GP9DPSgeJq',
|
|
||||||
icon: <CommentIcon />,
|
|
||||||
label: () => LL.HELP_INFORMATION_2()
|
|
||||||
},
|
|
||||||
{
|
|
||||||
href: 'https://github.com/emsesp/EMS-ESP32/issues/new/choose',
|
|
||||||
icon: <GitHubIcon />,
|
|
||||||
label: () => LL.HELP_INFORMATION_3()
|
|
||||||
}
|
|
||||||
],
|
|
||||||
[LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const isAdmin = useMemo(() => me?.admin ?? false, [me?.admin]);
|
|
||||||
|
|
||||||
// Memoize image source computation
|
|
||||||
const imageSrc = useMemo(
|
|
||||||
() =>
|
|
||||||
imgError || !customSupport.img_url ? DEFAULT_IMAGE_URL : customSupport.img_url,
|
|
||||||
[imgError, customSupport.img_url]
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
{customSupport.html && (
|
{customSupportHTML && (
|
||||||
<Stack
|
<Stack
|
||||||
padding={1}
|
padding={1}
|
||||||
mb={2}
|
mb={2}
|
||||||
direction="row"
|
direction="row"
|
||||||
divider={<Divider orientation="vertical" flexItem />}
|
divider={<Divider orientation="vertical" flexItem />}
|
||||||
sx={SUPPORT_BOX_STYLES}
|
sx={{
|
||||||
|
borderRadius: 3,
|
||||||
|
border: '2px solid grey',
|
||||||
|
justifyContent: 'space-evenly',
|
||||||
|
alignItems: 'center'
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
<Typography variant="subtitle1">
|
<Typography variant="subtitle1">
|
||||||
<div dangerouslySetInnerHTML={{ __html: customSupport.html }} />
|
<div dangerouslySetInnerHTML={{ __html: customSupportHTML }} />
|
||||||
</Typography>
|
</Typography>
|
||||||
<Box
|
<Box
|
||||||
component="img"
|
component="img"
|
||||||
referrerPolicy="no-referrer"
|
referrerPolicy="no-referrer"
|
||||||
sx={IMAGE_STYLES}
|
sx={{
|
||||||
onError={handleImageError}
|
maxHeight: { xs: 100, md: 250 }
|
||||||
src={imageSrc}
|
}}
|
||||||
|
onError={() => setNotFound(true)}
|
||||||
|
src={
|
||||||
|
notFound
|
||||||
|
? ''
|
||||||
|
: customSupportIMG ||
|
||||||
|
'https://docs.emsesp.org/_media/images/installer.jpeg'
|
||||||
|
}
|
||||||
/>
|
/>
|
||||||
</Stack>
|
</Stack>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{isAdmin && (
|
{me.admin && (
|
||||||
<List>
|
<List sx={{ borderRadius: 3, border: '2px solid grey' }}>
|
||||||
{helpLinks.map(({ href, icon, label }) => (
|
<ListItem>
|
||||||
<ListItem key={href}>
|
<ListItemButton
|
||||||
<ListItemButton
|
component="a"
|
||||||
component="a"
|
target="_blank"
|
||||||
target="_blank"
|
rel="noreferrer"
|
||||||
rel="noreferrer"
|
href="https://docs.emsesp.org"
|
||||||
href={href}
|
>
|
||||||
>
|
<ListItemAvatar>
|
||||||
<ListItemAvatar>
|
<Avatar sx={{ bgcolor: '#72caf9' }}>
|
||||||
<Avatar sx={AVATAR_STYLES}>{icon}</Avatar>
|
<MenuBookIcon />
|
||||||
</ListItemAvatar>
|
</Avatar>
|
||||||
<ListItemText primary={label()} />
|
</ListItemAvatar>
|
||||||
</ListItemButton>
|
<ListItemText primary={LL.HELP_INFORMATION_1()} />
|
||||||
</ListItem>
|
</ListItemButton>
|
||||||
))}
|
</ListItem>
|
||||||
|
|
||||||
|
<ListItem>
|
||||||
|
<ListItemButton
|
||||||
|
component="a"
|
||||||
|
target="_blank"
|
||||||
|
rel="noreferrer"
|
||||||
|
href="https://discord.gg/3J3GgnzpyT"
|
||||||
|
>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar sx={{ bgcolor: '#72caf9' }}>
|
||||||
|
<CommentIcon />
|
||||||
|
</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText primary={LL.HELP_INFORMATION_2()} />
|
||||||
|
</ListItemButton>
|
||||||
|
</ListItem>
|
||||||
|
|
||||||
|
<ListItem>
|
||||||
|
<ListItemButton
|
||||||
|
component="a"
|
||||||
|
target="_blank"
|
||||||
|
rel="noreferrer"
|
||||||
|
href="https://github.com/emsesp/EMS-ESP32/issues/new/choose"
|
||||||
|
>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar sx={{ bgcolor: '#72caf9' }}>
|
||||||
|
<GitHubIcon />
|
||||||
|
</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText primary={LL.HELP_INFORMATION_3()} />
|
||||||
|
</ListItemButton>
|
||||||
|
</ListItem>
|
||||||
</List>
|
</List>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
@@ -193,7 +157,7 @@ const HelpComponent = () => {
|
|||||||
startIcon={<DownloadIcon />}
|
startIcon={<DownloadIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="primary"
|
color="primary"
|
||||||
onClick={handleDownloadSystemInfo}
|
onClick={() => sendAPI({ device: 'system', cmd: 'info', id: 0 })}
|
||||||
>
|
>
|
||||||
{LL.SUPPORT_INFORMATION(0)}
|
{LL.SUPPORT_INFORMATION(0)}
|
||||||
</Button>
|
</Button>
|
||||||
@@ -209,14 +173,11 @@ const HelpComponent = () => {
|
|||||||
href="https://emsesp.org"
|
href="https://emsesp.org"
|
||||||
color="primary"
|
color="primary"
|
||||||
>
|
>
|
||||||
emsesp.org
|
{'emsesp.org'}
|
||||||
</Link>
|
</Link>
|
||||||
</Typography>
|
</Typography>
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
// Memoize the component to prevent unnecessary re-renders
|
|
||||||
const Help = memo(HelpComponent);
|
|
||||||
|
|
||||||
export default Help;
|
export default Help;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useCallback, useMemo, useState } from 'react';
|
import { useCallback, useState } from 'react';
|
||||||
import { useBlocker } from 'react-router';
|
import { useBlocker } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
@@ -31,19 +31,6 @@ import { readModules, writeModules } from '../../api/app';
|
|||||||
import ModulesDialog from './ModulesDialog';
|
import ModulesDialog from './ModulesDialog';
|
||||||
import type { ModuleItem } from './types';
|
import type { ModuleItem } from './types';
|
||||||
|
|
||||||
const PENDING_COLOR = 'red';
|
|
||||||
const ACTIVATED_COLOR = '#00FF7F';
|
|
||||||
|
|
||||||
const hasModulesChanged = (mi: ModuleItem): boolean =>
|
|
||||||
mi.enabled !== mi.o_enabled || mi.license !== mi.o_license;
|
|
||||||
|
|
||||||
const ColorStatus = memo(({ status }: { status: number }) => {
|
|
||||||
if (status === 1) {
|
|
||||||
return <div style={{ color: PENDING_COLOR }}>Pending Activation</div>;
|
|
||||||
}
|
|
||||||
return <div style={{ color: ACTIVATED_COLOR }}>Activated</div>;
|
|
||||||
});
|
|
||||||
|
|
||||||
const Modules = () => {
|
const Modules = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [numChanges, setNumChanges] = useState<number>(0);
|
const [numChanges, setNumChanges] = useState<number>(0);
|
||||||
@@ -69,111 +56,105 @@ const Modules = () => {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
const modules_theme = useTheme(
|
const modules_theme = useTheme({
|
||||||
useMemo(
|
Table: `
|
||||||
() => ({
|
--data-table-library_grid-template-columns: 48px 180px 120px 100px repeat(1, minmax(160px, 1fr)) 180px;
|
||||||
Table: `
|
`,
|
||||||
--data-table-library_grid-template-columns: 48px 180px 120px 100px repeat(1, minmax(160px, 1fr)) 180px;
|
BaseRow: `
|
||||||
`,
|
font-size: 14px;
|
||||||
BaseRow: `
|
.td {
|
||||||
font-size: 14px;
|
height: 32px;
|
||||||
.td {
|
}
|
||||||
height: 32px;
|
`,
|
||||||
}
|
BaseCell: `
|
||||||
`,
|
&:nth-of-type(1) {
|
||||||
BaseCell: `
|
text-align: center;
|
||||||
&:nth-of-type(1) {
|
}
|
||||||
text-align: center;
|
`,
|
||||||
}
|
HeaderRow: `
|
||||||
`,
|
text-transform: uppercase;
|
||||||
HeaderRow: `
|
background-color: black;
|
||||||
text-transform: uppercase;
|
color: #90CAF9;
|
||||||
background-color: black;
|
.th {
|
||||||
color: #90CAF9;
|
border-bottom: 1px solid #565656;
|
||||||
.th {
|
height: 36px;
|
||||||
border-bottom: 1px solid #565656;
|
}
|
||||||
height: 36px;
|
`,
|
||||||
}
|
Row: `
|
||||||
`,
|
background-color: #1e1e1e;
|
||||||
Row: `
|
position: relative;
|
||||||
background-color: #1e1e1e;
|
cursor: pointer;
|
||||||
position: relative;
|
.td {
|
||||||
cursor: pointer;
|
border-top: 1px solid #565656;
|
||||||
.td {
|
border-bottom: 1px solid #565656;
|
||||||
border-top: 1px solid #565656;
|
}
|
||||||
border-bottom: 1px solid #565656;
|
&:hover .td {
|
||||||
}
|
border-top: 1px solid #177ac9;
|
||||||
&:hover .td {
|
border-bottom: 1px solid #177ac9;
|
||||||
border-top: 1px solid #177ac9;
|
}
|
||||||
border-bottom: 1px solid #177ac9;
|
&:nth-of-type(odd) .td {
|
||||||
}
|
background-color: #303030;
|
||||||
&:nth-of-type(odd) .td {
|
}
|
||||||
background-color: #303030;
|
`
|
||||||
}
|
});
|
||||||
`
|
|
||||||
}),
|
|
||||||
[]
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
const onDialogClose = useCallback(() => {
|
const onDialogClose = () => {
|
||||||
setDialogOpen(false);
|
setDialogOpen(false);
|
||||||
}, []);
|
};
|
||||||
|
|
||||||
const updateModuleItem = useCallback((updatedItem: ModuleItem) => {
|
const onDialogSave = (updatedItem: ModuleItem) => {
|
||||||
void updateState(readModules(), (data: ModuleItem[]) => {
|
setDialogOpen(false);
|
||||||
const new_data = data.map((mi) =>
|
updateModuleItem(updatedItem);
|
||||||
mi.id === updatedItem.id ? { ...mi, ...updatedItem } : mi
|
};
|
||||||
);
|
|
||||||
setNumChanges(new_data.filter(hasModulesChanged).length);
|
|
||||||
return new_data;
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const onDialogSave = useCallback(
|
|
||||||
(updatedItem: ModuleItem) => {
|
|
||||||
setDialogOpen(false);
|
|
||||||
updateModuleItem(updatedItem);
|
|
||||||
},
|
|
||||||
[updateModuleItem]
|
|
||||||
);
|
|
||||||
|
|
||||||
const editModuleItem = useCallback((mi: ModuleItem) => {
|
const editModuleItem = useCallback((mi: ModuleItem) => {
|
||||||
setSelectedModuleItem(mi);
|
setSelectedModuleItem(mi);
|
||||||
setDialogOpen(true);
|
setDialogOpen(true);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const onCancel = useCallback(async () => {
|
const onCancel = async () => {
|
||||||
await fetchModules().then(() => {
|
await fetchModules().then(() => {
|
||||||
setNumChanges(0);
|
setNumChanges(0);
|
||||||
});
|
});
|
||||||
}, [fetchModules]);
|
};
|
||||||
|
|
||||||
const saveModules = useCallback(async () => {
|
function hasModulesChanged(mi: ModuleItem) {
|
||||||
try {
|
return mi.enabled !== mi.o_enabled || mi.license !== mi.o_license;
|
||||||
await Promise.all(
|
}
|
||||||
modules.map((condensed_mi: ModuleItem) =>
|
|
||||||
updateModules({
|
const updateModuleItem = (updatedItem: ModuleItem) => {
|
||||||
key: condensed_mi.key,
|
void updateState(readModules(), (data: ModuleItem[]) => {
|
||||||
enabled: condensed_mi.enabled,
|
const new_data = data.map((mi) =>
|
||||||
license: condensed_mi.license
|
mi.id === updatedItem.id ? { ...mi, ...updatedItem } : mi
|
||||||
})
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
toast.success(LL.MODULES_UPDATED());
|
setNumChanges(new_data.filter((mi) => hasModulesChanged(mi)).length);
|
||||||
} catch (error) {
|
return new_data;
|
||||||
toast.error(error instanceof Error ? error.message : String(error));
|
});
|
||||||
} finally {
|
};
|
||||||
await fetchModules();
|
|
||||||
setNumChanges(0);
|
|
||||||
}
|
|
||||||
}, [modules, updateModules, LL, fetchModules]);
|
|
||||||
|
|
||||||
const content = useMemo(() => {
|
const saveModules = async () => {
|
||||||
|
await updateModules({
|
||||||
|
modules: modules.map((condensed_mi) => ({
|
||||||
|
key: condensed_mi.key,
|
||||||
|
enabled: condensed_mi.enabled,
|
||||||
|
license: condensed_mi.license
|
||||||
|
}))
|
||||||
|
})
|
||||||
|
.then(() => {
|
||||||
|
toast.success(LL.MODULES_UPDATED());
|
||||||
|
})
|
||||||
|
.catch((error: Error) => {
|
||||||
|
toast.error(error.message);
|
||||||
|
})
|
||||||
|
.finally(async () => {
|
||||||
|
await fetchModules();
|
||||||
|
setNumChanges(0);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const renderContent = () => {
|
||||||
if (!modules) {
|
if (!modules) {
|
||||||
return (
|
return <FormLoader onRetry={fetchModules} errorMessage={error?.message} />;
|
||||||
<FormLoader onRetry={fetchModules} errorMessage={error?.message || ''} />
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (modules.length === 0) {
|
if (modules.length === 0) {
|
||||||
@@ -184,6 +165,13 @@ const Modules = () => {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const colorStatus = (status: number) => {
|
||||||
|
if (status === 1) {
|
||||||
|
return <div style={{ color: 'red' }}>Pending Activation</div>;
|
||||||
|
}
|
||||||
|
return <div style={{ color: '#00FF7F' }}>Activated</div>;
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Box mb={2} color="warning.main">
|
<Box mb={2} color="warning.main">
|
||||||
@@ -226,9 +214,7 @@ const Modules = () => {
|
|||||||
<Cell>{mi.author}</Cell>
|
<Cell>{mi.author}</Cell>
|
||||||
<Cell>{mi.version}</Cell>
|
<Cell>{mi.version}</Cell>
|
||||||
<Cell>{mi.message}</Cell>
|
<Cell>{mi.message}</Cell>
|
||||||
<Cell>
|
<Cell>{colorStatus(mi.status)}</Cell>
|
||||||
<ColorStatus status={mi.status} />
|
|
||||||
</Cell>
|
|
||||||
</Row>
|
</Row>
|
||||||
))}
|
))}
|
||||||
</Body>
|
</Body>
|
||||||
@@ -262,22 +248,12 @@ const Modules = () => {
|
|||||||
</Box>
|
</Box>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}, [
|
};
|
||||||
modules,
|
|
||||||
fetchModules,
|
|
||||||
error,
|
|
||||||
modules_theme,
|
|
||||||
editModuleItem,
|
|
||||||
LL,
|
|
||||||
numChanges,
|
|
||||||
onCancel,
|
|
||||||
saveModules
|
|
||||||
]);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
{content}
|
{renderContent()}
|
||||||
{selectedModuleItem && (
|
{selectedModuleItem && (
|
||||||
<ModulesDialog
|
<ModulesDialog
|
||||||
open={dialogOpen}
|
open={dialogOpen}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useEffect, useMemo, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import DoneIcon from '@mui/icons-material/Done';
|
import DoneIcon from '@mui/icons-material/Done';
|
||||||
@@ -10,7 +10,7 @@ import {
|
|||||||
DialogActions,
|
DialogActions,
|
||||||
DialogContent,
|
DialogContent,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
Grid,
|
Grid2 as Grid,
|
||||||
TextField
|
TextField
|
||||||
} from '@mui/material';
|
} from '@mui/material';
|
||||||
|
|
||||||
@@ -37,35 +37,25 @@ const ModulesDialog = ({
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [editItem, setEditItem] = useState<ModuleItem>(selectedItem);
|
const [editItem, setEditItem] = useState<ModuleItem>(selectedItem);
|
||||||
|
|
||||||
const updateFormValue = useMemo(
|
const updateFormValue = updateValue(setEditItem);
|
||||||
() =>
|
|
||||||
updateValue(
|
|
||||||
setEditItem as unknown as React.Dispatch<
|
|
||||||
React.SetStateAction<Record<string, unknown>>
|
|
||||||
>
|
|
||||||
),
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
// Sync form state when dialog opens or selected item changes
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
setEditItem(selectedItem);
|
setEditItem(selectedItem);
|
||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const handleSave = useCallback(() => {
|
const close = () => {
|
||||||
onSave(editItem);
|
onClose();
|
||||||
}, [editItem, onSave]);
|
};
|
||||||
|
|
||||||
const dialogTitle = useMemo(
|
const save = () => {
|
||||||
() => `${LL.EDIT()} ${editItem.key}`,
|
onSave(editItem);
|
||||||
[LL, editItem.key]
|
};
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} fullWidth maxWidth="xs" open={open} onClose={onClose}>
|
<Dialog sx={dialogStyle} fullWidth maxWidth="xs" open={open} onClose={onClose}>
|
||||||
<DialogTitle>{dialogTitle}</DialogTitle>
|
<DialogTitle>{LL.EDIT() + ' ' + editItem.key}</DialogTitle>
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<Grid container>
|
<Grid container>
|
||||||
<BlockFormControlLabel
|
<BlockFormControlLabel
|
||||||
@@ -95,7 +85,7 @@ const ModulesDialog = ({
|
|||||||
<Button
|
<Button
|
||||||
startIcon={<CancelIcon />}
|
startIcon={<CancelIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={onClose}
|
onClick={close}
|
||||||
color="secondary"
|
color="secondary"
|
||||||
>
|
>
|
||||||
{LL.CANCEL()}
|
{LL.CANCEL()}
|
||||||
@@ -103,7 +93,7 @@ const ModulesDialog = ({
|
|||||||
<Button
|
<Button
|
||||||
startIcon={<DoneIcon />}
|
startIcon={<DoneIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={handleSave}
|
onClick={save}
|
||||||
color="primary"
|
color="primary"
|
||||||
>
|
>
|
||||||
{LL.UPDATE()}
|
{LL.UPDATE()}
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
import { memo } from 'react';
|
|
||||||
|
|
||||||
import CommentsDisabledOutlinedIcon from '@mui/icons-material/CommentsDisabledOutlined';
|
import CommentsDisabledOutlinedIcon from '@mui/icons-material/CommentsDisabledOutlined';
|
||||||
import DeleteForeverIcon from '@mui/icons-material/DeleteForever';
|
import DeleteForeverIcon from '@mui/icons-material/DeleteForever';
|
||||||
import DeleteOutlineIcon from '@mui/icons-material/DeleteOutline';
|
import DeleteOutlineIcon from '@mui/icons-material/DeleteOutline';
|
||||||
@@ -12,39 +10,33 @@ import VisibilityOffOutlinedIcon from '@mui/icons-material/VisibilityOffOutlined
|
|||||||
import VisibilityOutlinedIcon from '@mui/icons-material/VisibilityOutlined';
|
import VisibilityOutlinedIcon from '@mui/icons-material/VisibilityOutlined';
|
||||||
import type { SvgIconProps } from '@mui/material';
|
import type { SvgIconProps } from '@mui/material';
|
||||||
|
|
||||||
export type OptionType =
|
type OptionType =
|
||||||
| 'deleted'
|
| 'deleted'
|
||||||
| 'readonly'
|
| 'readonly'
|
||||||
| 'web_exclude'
|
| 'web_exclude'
|
||||||
| 'api_mqtt_exclude'
|
| 'api_mqtt_exclude'
|
||||||
| 'favorite';
|
| 'favorite';
|
||||||
|
|
||||||
type IconPair = [
|
const OPTION_ICONS: {
|
||||||
React.ComponentType<SvgIconProps>,
|
[type in OptionType]: [
|
||||||
React.ComponentType<SvgIconProps>
|
React.ComponentType<SvgIconProps>,
|
||||||
];
|
React.ComponentType<SvgIconProps>
|
||||||
|
];
|
||||||
const OPTION_ICONS: Record<OptionType, IconPair> = {
|
} = {
|
||||||
deleted: [DeleteForeverIcon, DeleteOutlineIcon],
|
deleted: [DeleteForeverIcon, DeleteOutlineIcon],
|
||||||
readonly: [EditOffOutlinedIcon, EditOutlinedIcon],
|
readonly: [EditOffOutlinedIcon, EditOutlinedIcon],
|
||||||
web_exclude: [VisibilityOffOutlinedIcon, VisibilityOutlinedIcon],
|
web_exclude: [VisibilityOffOutlinedIcon, VisibilityOutlinedIcon],
|
||||||
api_mqtt_exclude: [CommentsDisabledOutlinedIcon, InsertCommentOutlinedIcon],
|
api_mqtt_exclude: [CommentsDisabledOutlinedIcon, InsertCommentOutlinedIcon],
|
||||||
favorite: [StarIcon, StarOutlineIcon]
|
favorite: [StarIcon, StarOutlineIcon]
|
||||||
} as const;
|
|
||||||
|
|
||||||
const ICON_SIZE = 16;
|
|
||||||
const ICON_SX = { fontSize: ICON_SIZE, verticalAlign: 'middle' } as const;
|
|
||||||
|
|
||||||
export interface OptionIconProps {
|
|
||||||
readonly type: OptionType;
|
|
||||||
readonly isSet: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
const OptionIcon = ({ type, isSet }: OptionIconProps) => {
|
|
||||||
const [SetIcon, UnsetIcon] = OPTION_ICONS[type];
|
|
||||||
const Icon = isSet ? SetIcon : UnsetIcon;
|
|
||||||
|
|
||||||
return <Icon {...(isSet && { color: 'primary' })} sx={ICON_SX} />;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default memo(OptionIcon);
|
const OptionIcon = ({ type, isSet }: { type: OptionType; isSet: boolean }) => {
|
||||||
|
const Icon = OPTION_ICONS[type][isSet ? 0 : 1];
|
||||||
|
return isSet ? (
|
||||||
|
<Icon color="primary" sx={{ fontSize: 16, verticalAlign: 'middle' }} />
|
||||||
|
) : (
|
||||||
|
<Icon sx={{ fontSize: 16, verticalAlign: 'middle' }} />
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default OptionIcon;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useEffect, useMemo, useState } from 'react';
|
import { useCallback, useEffect, useState } from 'react';
|
||||||
import { useBlocker } from 'react-router';
|
import { useBlocker } from 'react-router';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
@@ -27,7 +27,6 @@ import {
|
|||||||
useLayoutTitle
|
useLayoutTitle
|
||||||
} from 'components';
|
} from 'components';
|
||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
import { useInterval } from 'utils';
|
|
||||||
|
|
||||||
import { readSchedule, writeSchedule } from '../../api/app';
|
import { readSchedule, writeSchedule } from '../../api/app';
|
||||||
import SettingsSchedulerDialog from './SchedulerDialog';
|
import SettingsSchedulerDialog from './SchedulerDialog';
|
||||||
@@ -35,77 +34,6 @@ import { ScheduleFlag } from './types';
|
|||||||
import type { Schedule, ScheduleItem } from './types';
|
import type { Schedule, ScheduleItem } from './types';
|
||||||
import { schedulerItemValidation } from './validators';
|
import { schedulerItemValidation } from './validators';
|
||||||
|
|
||||||
// Constants
|
|
||||||
const INTERVAL_DELAY = 30000; // 30 seconds
|
|
||||||
const MIN_ID = -100;
|
|
||||||
const MAX_ID = 100;
|
|
||||||
const ICON_SIZE = 16;
|
|
||||||
const SCHEDULE_FLAG_THRESHOLD = 127;
|
|
||||||
const FLAG_ALL_DAYS = 127;
|
|
||||||
const REFERENCE_YEAR = 2017;
|
|
||||||
const REFERENCE_MONTH = '01';
|
|
||||||
const LOG_2 = Math.log(2);
|
|
||||||
|
|
||||||
// Days of week starting from Monday (1-7)
|
|
||||||
const WEEK_DAYS = [1, 2, 3, 4, 5, 6, 7] as const;
|
|
||||||
|
|
||||||
const DEFAULT_SCHEDULE_ITEM: Omit<ScheduleItem, 'id' | 'o_id'> = {
|
|
||||||
active: false,
|
|
||||||
deleted: false,
|
|
||||||
flags: FLAG_ALL_DAYS,
|
|
||||||
time: '',
|
|
||||||
cmd: '',
|
|
||||||
value: '',
|
|
||||||
name: ''
|
|
||||||
};
|
|
||||||
|
|
||||||
const scheduleTheme = {
|
|
||||||
Table: `
|
|
||||||
--data-table-library_grid-template-columns: 36px 210px 100px 192px repeat(1, minmax(100px, 1fr)) 160px;
|
|
||||||
`,
|
|
||||||
BaseRow: `
|
|
||||||
font-size: 14px;
|
|
||||||
.td {
|
|
||||||
height: 32px;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
BaseCell: `
|
|
||||||
&:nth-of-type(2) {
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
&:nth-of-type(1) {
|
|
||||||
text-align: center;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
HeaderRow: `
|
|
||||||
text-transform: uppercase;
|
|
||||||
background-color: black;
|
|
||||||
color: #90CAF9;
|
|
||||||
.th {
|
|
||||||
border-bottom: 1px solid #565656;
|
|
||||||
height: 36px;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
Row: `
|
|
||||||
background-color: #1e1e1e;
|
|
||||||
position: relative;
|
|
||||||
cursor: pointer;
|
|
||||||
.td {
|
|
||||||
border-bottom: 1px solid #565656;
|
|
||||||
}
|
|
||||||
&:hover .td {
|
|
||||||
background-color: #177ac9;
|
|
||||||
}
|
|
||||||
`
|
|
||||||
};
|
|
||||||
|
|
||||||
const scheduleTypeLabels: Record<number, string> = {
|
|
||||||
[ScheduleFlag.SCHEDULE_IMMEDIATE]: 'Immediate',
|
|
||||||
[ScheduleFlag.SCHEDULE_TIMER]: 'Timer',
|
|
||||||
[ScheduleFlag.SCHEDULE_CONDITION]: 'Condition',
|
|
||||||
[ScheduleFlag.SCHEDULE_ONCHANGE]: 'On Change'
|
|
||||||
};
|
|
||||||
|
|
||||||
const Scheduler = () => {
|
const Scheduler = () => {
|
||||||
const { LL, locale } = useI18nContext();
|
const { LL, locale } = useI18nContext();
|
||||||
const [numChanges, setNumChanges] = useState<number>(0);
|
const [numChanges, setNumChanges] = useState<number>(0);
|
||||||
@@ -132,7 +60,7 @@ const Scheduler = () => {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
const hasScheduleChanged = useCallback((si: ScheduleItem) => {
|
function hasScheduleChanged(si: ScheduleItem) {
|
||||||
return (
|
return (
|
||||||
si.id !== si.o_id ||
|
si.id !== si.o_id ||
|
||||||
(si.name || '') !== (si.o_name || '') ||
|
(si.name || '') !== (si.o_name || '') ||
|
||||||
@@ -143,56 +71,85 @@ const Scheduler = () => {
|
|||||||
si.cmd !== si.o_cmd ||
|
si.cmd !== si.o_cmd ||
|
||||||
si.value !== si.o_value
|
si.value !== si.o_value
|
||||||
);
|
);
|
||||||
}, []);
|
}
|
||||||
|
|
||||||
const intervalCallback = useCallback(() => {
|
|
||||||
if (numChanges === 0) {
|
|
||||||
void fetchSchedule();
|
|
||||||
}
|
|
||||||
}, [numChanges, fetchSchedule]);
|
|
||||||
|
|
||||||
useInterval(intervalCallback, INTERVAL_DELAY);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const formatter = new Intl.DateTimeFormat(locale, {
|
const formatter = new Intl.DateTimeFormat(locale, {
|
||||||
weekday: 'short',
|
weekday: 'short',
|
||||||
timeZone: 'UTC'
|
timeZone: 'UTC'
|
||||||
});
|
});
|
||||||
const days = WEEK_DAYS.map((day) => {
|
const days = [1, 2, 3, 4, 5, 6, 7].map((day) => {
|
||||||
const dayStr = String(day).padStart(2, '0');
|
const dd = day < 10 ? `0${day}` : day;
|
||||||
return new Date(
|
return new Date(`2017-01-${dd}T00:00:00+00:00`);
|
||||||
`${REFERENCE_YEAR}-${REFERENCE_MONTH}-${dayStr}T00:00:00+00:00`
|
|
||||||
);
|
|
||||||
});
|
});
|
||||||
setDow(days.map((date) => formatter.format(date)));
|
setDow(days.map((date) => formatter.format(date)));
|
||||||
}, [locale]);
|
}, [locale]);
|
||||||
|
|
||||||
const schedule_theme = useTheme(scheduleTheme);
|
const schedule_theme = useTheme({
|
||||||
|
Table: `
|
||||||
|
--data-table-library_grid-template-columns: 36px 210px 100px 192px repeat(1, minmax(100px, 1fr)) 160px;
|
||||||
|
`,
|
||||||
|
BaseRow: `
|
||||||
|
font-size: 14px;
|
||||||
|
.td {
|
||||||
|
height: 32px;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
BaseCell: `
|
||||||
|
&:nth-of-type(2) {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
&:nth-of-type(1) {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
HeaderRow: `
|
||||||
|
text-transform: uppercase;
|
||||||
|
background-color: black;
|
||||||
|
color: #90CAF9;
|
||||||
|
.th {
|
||||||
|
border-bottom: 1px solid #565656;
|
||||||
|
height: 36px;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
Row: `
|
||||||
|
background-color: #1e1e1e;
|
||||||
|
position: relative;
|
||||||
|
cursor: pointer;
|
||||||
|
.td {
|
||||||
|
border-bottom: 1px solid #565656;
|
||||||
|
}
|
||||||
|
&:hover .td {
|
||||||
|
background-color: #177ac9;
|
||||||
|
}
|
||||||
|
`
|
||||||
|
});
|
||||||
|
|
||||||
const saveSchedule = useCallback(async () => {
|
const saveSchedule = async () => {
|
||||||
try {
|
await updateSchedule({
|
||||||
await updateSchedule({
|
schedule: schedule
|
||||||
schedule: schedule
|
.filter((si) => !si.deleted)
|
||||||
.filter((si: ScheduleItem) => !si.deleted)
|
.map((condensed_si) => ({
|
||||||
.map((condensed_si: ScheduleItem) => ({
|
id: condensed_si.id,
|
||||||
id: condensed_si.id,
|
active: condensed_si.active,
|
||||||
active: condensed_si.active,
|
flags: condensed_si.flags,
|
||||||
flags: condensed_si.flags,
|
time: condensed_si.time,
|
||||||
time: condensed_si.time,
|
cmd: condensed_si.cmd,
|
||||||
cmd: condensed_si.cmd,
|
value: condensed_si.value,
|
||||||
value: condensed_si.value,
|
name: condensed_si.name
|
||||||
name: condensed_si.name
|
}))
|
||||||
}))
|
})
|
||||||
|
.then(() => {
|
||||||
|
toast.success(LL.SCHEDULE_UPDATED());
|
||||||
|
})
|
||||||
|
.catch((error: Error) => {
|
||||||
|
toast.error(error.message);
|
||||||
|
})
|
||||||
|
.finally(async () => {
|
||||||
|
await fetchSchedule();
|
||||||
|
setNumChanges(0);
|
||||||
});
|
});
|
||||||
toast.success(LL.SCHEDULE_UPDATED());
|
};
|
||||||
} catch (error: unknown) {
|
|
||||||
const message = error instanceof Error ? error.message : String(error);
|
|
||||||
toast.error(message);
|
|
||||||
} finally {
|
|
||||||
await fetchSchedule();
|
|
||||||
setNumChanges(0);
|
|
||||||
}
|
|
||||||
}, [LL, schedule, updateSchedule, fetchSchedule]);
|
|
||||||
|
|
||||||
const editScheduleItem = useCallback((si: ScheduleItem) => {
|
const editScheduleItem = useCallback((si: ScheduleItem) => {
|
||||||
setCreating(false);
|
setCreating(false);
|
||||||
@@ -203,93 +160,93 @@ const Scheduler = () => {
|
|||||||
}
|
}
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const onDialogClose = useCallback(() => {
|
const onDialogClose = () => {
|
||||||
setDialogOpen(false);
|
setDialogOpen(false);
|
||||||
}, []);
|
};
|
||||||
|
|
||||||
const onDialogCancel = useCallback(async () => {
|
const onDialogCancel = async () => {
|
||||||
await fetchSchedule().then(() => {
|
await fetchSchedule().then(() => {
|
||||||
setNumChanges(0);
|
setNumChanges(0);
|
||||||
});
|
});
|
||||||
}, [fetchSchedule]);
|
};
|
||||||
|
|
||||||
const onDialogSave = useCallback(
|
const onDialogSave = (updatedItem: ScheduleItem) => {
|
||||||
(updatedItem: ScheduleItem) => {
|
setDialogOpen(false);
|
||||||
setDialogOpen(false);
|
void updateState(readSchedule(), (data: ScheduleItem[]) => {
|
||||||
void updateState(readSchedule(), (data: ScheduleItem[]) => {
|
const new_data = creating
|
||||||
const new_data = creating
|
? [
|
||||||
? [...data, updatedItem]
|
...data.filter((si) => creating || si.o_id !== updatedItem.o_id),
|
||||||
: data.map((si) =>
|
updatedItem
|
||||||
si.id === updatedItem.id ? { ...si, ...updatedItem } : si
|
]
|
||||||
);
|
: data.map((si) =>
|
||||||
|
si.id === updatedItem.id ? { ...si, ...updatedItem } : si
|
||||||
|
);
|
||||||
|
|
||||||
setNumChanges(new_data.filter((si) => hasScheduleChanged(si)).length);
|
setNumChanges(new_data.filter((si) => hasScheduleChanged(si)).length);
|
||||||
|
|
||||||
return new_data;
|
return new_data;
|
||||||
});
|
});
|
||||||
},
|
};
|
||||||
[creating, hasScheduleChanged]
|
|
||||||
);
|
|
||||||
|
|
||||||
const addScheduleItem = useCallback(() => {
|
const addScheduleItem = () => {
|
||||||
setCreating(true);
|
setCreating(true);
|
||||||
const newItem: ScheduleItem = {
|
setSelectedScheduleItem({
|
||||||
id: Math.floor(Math.random() * (MAX_ID - MIN_ID) + MIN_ID),
|
id: Math.floor(Math.random() * (Math.floor(200) - 100) + 100),
|
||||||
...DEFAULT_SCHEDULE_ITEM
|
active: false,
|
||||||
};
|
deleted: false,
|
||||||
setSelectedScheduleItem(newItem);
|
flags: ScheduleFlag.SCHEDULE_DAY,
|
||||||
|
time: '',
|
||||||
|
cmd: '',
|
||||||
|
value: '',
|
||||||
|
name: ''
|
||||||
|
});
|
||||||
setDialogOpen(true);
|
setDialogOpen(true);
|
||||||
}, []);
|
};
|
||||||
|
|
||||||
const filteredAndSortedSchedule = useMemo(
|
const renderSchedule = () => {
|
||||||
() =>
|
if (!schedule) {
|
||||||
schedule
|
return <FormLoader onRetry={fetchSchedule} errorMessage={error?.message} />;
|
||||||
.filter((si: ScheduleItem) => !si.deleted)
|
}
|
||||||
.sort((a: ScheduleItem, b: ScheduleItem) => a.flags - b.flags),
|
|
||||||
[schedule]
|
|
||||||
);
|
|
||||||
|
|
||||||
const dayBox = useCallback(
|
const dayBox = (si: ScheduleItem, flag: number) => (
|
||||||
(si: ScheduleItem, flag: number) => {
|
<>
|
||||||
const dayIndex = Math.log(flag) / LOG_2;
|
<Box>
|
||||||
const isActive = (si.flags & flag) === flag;
|
<Typography
|
||||||
|
sx={{ fontSize: 11 }}
|
||||||
|
color={(si.flags & flag) === flag ? 'primary' : 'grey'}
|
||||||
|
>
|
||||||
|
{dow[Math.log(flag) / Math.log(2)]}
|
||||||
|
</Typography>
|
||||||
|
</Box>
|
||||||
|
<Divider orientation="vertical" flexItem />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
const scheduleType = (si: ScheduleItem) => (
|
||||||
<>
|
|
||||||
<Box>
|
|
||||||
<Typography sx={{ fontSize: 11 }} color={isActive ? 'primary' : 'grey'}>
|
|
||||||
{dow[dayIndex]}
|
|
||||||
</Typography>
|
|
||||||
</Box>
|
|
||||||
<Divider orientation="vertical" flexItem />
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
[dow]
|
|
||||||
);
|
|
||||||
|
|
||||||
const scheduleType = useCallback((si: ScheduleItem) => {
|
|
||||||
const label = scheduleTypeLabels[si.flags];
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Box>
|
<Box>
|
||||||
<Typography sx={{ fontSize: 11 }} color="primary">
|
<Typography sx={{ fontSize: 11 }} color="primary">
|
||||||
{label || ''}
|
{si.flags === ScheduleFlag.SCHEDULE_IMMEDIATE ? (
|
||||||
|
<>Immediate</>
|
||||||
|
) : si.flags === ScheduleFlag.SCHEDULE_TIMER ? (
|
||||||
|
<>Timer</>
|
||||||
|
) : si.flags === ScheduleFlag.SCHEDULE_CONDITION ? (
|
||||||
|
<>Condition</>
|
||||||
|
) : si.flags === ScheduleFlag.SCHEDULE_ONCHANGE ? (
|
||||||
|
<>On Change</>
|
||||||
|
) : (
|
||||||
|
<></>
|
||||||
|
)}
|
||||||
</Typography>
|
</Typography>
|
||||||
</Box>
|
</Box>
|
||||||
);
|
);
|
||||||
}, []);
|
|
||||||
|
|
||||||
const renderSchedule = useCallback(() => {
|
|
||||||
if (!schedule) {
|
|
||||||
return (
|
|
||||||
<FormLoader onRetry={fetchSchedule} errorMessage={error?.message || ''} />
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Table
|
<Table
|
||||||
data={{ nodes: filteredAndSortedSchedule }}
|
data={{
|
||||||
|
nodes: schedule
|
||||||
|
.filter((si) => !si.deleted)
|
||||||
|
.sort((a, b) => a.flags - b.flags)
|
||||||
|
}}
|
||||||
theme={schedule_theme}
|
theme={schedule_theme}
|
||||||
layout={{ custom: true }}
|
layout={{ custom: true }}
|
||||||
>
|
>
|
||||||
@@ -309,15 +266,22 @@ const Scheduler = () => {
|
|||||||
{tableList.map((si: ScheduleItem) => (
|
{tableList.map((si: ScheduleItem) => (
|
||||||
<Row key={si.id} item={si} onClick={() => editScheduleItem(si)}>
|
<Row key={si.id} item={si} onClick={() => editScheduleItem(si)}>
|
||||||
<Cell stiff>
|
<Cell stiff>
|
||||||
<CircleIcon
|
{si.active ? (
|
||||||
color={si.active ? 'success' : 'error'}
|
<CircleIcon
|
||||||
sx={{ fontSize: ICON_SIZE, verticalAlign: 'middle' }}
|
color="success"
|
||||||
/>
|
sx={{ fontSize: 16, verticalAlign: 'middle' }}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<CircleIcon
|
||||||
|
color="error"
|
||||||
|
sx={{ fontSize: 16, verticalAlign: 'middle' }}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
</Cell>
|
</Cell>
|
||||||
<Cell stiff>
|
<Cell stiff>
|
||||||
<Stack spacing={0.5} direction="row">
|
<Stack spacing={0.5} direction="row">
|
||||||
<Divider orientation="vertical" flexItem />
|
<Divider orientation="vertical" flexItem />
|
||||||
{si.flags > SCHEDULE_FLAG_THRESHOLD ? (
|
{si.flags > 127 ? (
|
||||||
scheduleType(si)
|
scheduleType(si)
|
||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
@@ -343,17 +307,7 @@ const Scheduler = () => {
|
|||||||
)}
|
)}
|
||||||
</Table>
|
</Table>
|
||||||
);
|
);
|
||||||
}, [
|
};
|
||||||
schedule,
|
|
||||||
error,
|
|
||||||
fetchSchedule,
|
|
||||||
filteredAndSortedSchedule,
|
|
||||||
schedule_theme,
|
|
||||||
editScheduleItem,
|
|
||||||
LL,
|
|
||||||
dayBox,
|
|
||||||
scheduleType
|
|
||||||
]);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
@@ -375,7 +329,7 @@ const Scheduler = () => {
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<Box display="flex" flexWrap="wrap">
|
<Box mt={1} display="flex" flexWrap="wrap">
|
||||||
<Box flexGrow={1}>
|
<Box flexGrow={1}>
|
||||||
{numChanges !== 0 && (
|
{numChanges !== 0 && (
|
||||||
<ButtonRow>
|
<ButtonRow>
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useEffect, useMemo, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
import AddIcon from '@mui/icons-material/Add';
|
import AddIcon from '@mui/icons-material/Add';
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -13,7 +13,7 @@ import {
|
|||||||
DialogActions,
|
DialogActions,
|
||||||
DialogContent,
|
DialogContent,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
Grid,
|
Grid2 as Grid,
|
||||||
TextField,
|
TextField,
|
||||||
ToggleButton,
|
ToggleButton,
|
||||||
ToggleButtonGroup,
|
ToggleButtonGroup,
|
||||||
@@ -31,35 +31,6 @@ import { validate } from 'validators';
|
|||||||
import { ScheduleFlag } from './types';
|
import { ScheduleFlag } from './types';
|
||||||
import type { ScheduleItem } from './types';
|
import type { ScheduleItem } from './types';
|
||||||
|
|
||||||
// Constants
|
|
||||||
const FLAG_MASK_127 = 127;
|
|
||||||
const SCHEDULE_TYPE_THRESHOLD = 127;
|
|
||||||
const FLAG_ALL_DAYS = 127;
|
|
||||||
const DEFAULT_TIME = '00:00';
|
|
||||||
const TYPOGRAPHY_FONT_SIZE = 10;
|
|
||||||
|
|
||||||
// Day of week flag configuration (static, defined outside component)
|
|
||||||
const DAY_FLAGS = [
|
|
||||||
{ value: '2', flag: ScheduleFlag.SCHEDULE_MON },
|
|
||||||
{ value: '4', flag: ScheduleFlag.SCHEDULE_TUE },
|
|
||||||
{ value: '8', flag: ScheduleFlag.SCHEDULE_WED },
|
|
||||||
{ value: '16', flag: ScheduleFlag.SCHEDULE_THU },
|
|
||||||
{ value: '32', flag: ScheduleFlag.SCHEDULE_FRI },
|
|
||||||
{ value: '64', flag: ScheduleFlag.SCHEDULE_SAT },
|
|
||||||
{ value: '1', flag: ScheduleFlag.SCHEDULE_SUN }
|
|
||||||
] as const;
|
|
||||||
|
|
||||||
// Day of week flag values array (static)
|
|
||||||
const FLAG_VALUES = [
|
|
||||||
ScheduleFlag.SCHEDULE_SUN,
|
|
||||||
ScheduleFlag.SCHEDULE_MON,
|
|
||||||
ScheduleFlag.SCHEDULE_TUE,
|
|
||||||
ScheduleFlag.SCHEDULE_WED,
|
|
||||||
ScheduleFlag.SCHEDULE_THU,
|
|
||||||
ScheduleFlag.SCHEDULE_FRI,
|
|
||||||
ScheduleFlag.SCHEDULE_SAT
|
|
||||||
] as const;
|
|
||||||
|
|
||||||
interface SchedulerDialogProps {
|
interface SchedulerDialogProps {
|
||||||
open: boolean;
|
open: boolean;
|
||||||
creating: boolean;
|
creating: boolean;
|
||||||
@@ -82,164 +53,107 @@ const SchedulerDialog = ({
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [editItem, setEditItem] = useState<ScheduleItem>(selectedItem);
|
const [editItem, setEditItem] = useState<ScheduleItem>(selectedItem);
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
|
|
||||||
const [scheduleType, setScheduleType] = useState<ScheduleFlag>();
|
const [scheduleType, setScheduleType] = useState<ScheduleFlag>();
|
||||||
|
|
||||||
const updateFormValue = useMemo(
|
const updateFormValue = updateValue(setEditItem);
|
||||||
() =>
|
|
||||||
updateValue(
|
|
||||||
setEditItem as unknown as React.Dispatch<
|
|
||||||
React.SetStateAction<Record<string, unknown>>
|
|
||||||
>
|
|
||||||
),
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
setEditItem(selectedItem);
|
setEditItem(selectedItem);
|
||||||
// Set the flags based on type when page is loaded:
|
// set the flags based on type when page is loaded...
|
||||||
// 0-127 is day schedule
|
// 0-127 is day schedule
|
||||||
// 128 is timer
|
// 128 is timer
|
||||||
// 129 is on change
|
// 129 is on change
|
||||||
// 130 is on condition
|
// 130 is on condition
|
||||||
// 132 is immediate
|
// 132 is immediate
|
||||||
setScheduleType(
|
setScheduleType(
|
||||||
selectedItem.flags <= SCHEDULE_TYPE_THRESHOLD
|
selectedItem.flags < 128 ? ScheduleFlag.SCHEDULE_DAY : selectedItem.flags
|
||||||
? ScheduleFlag.SCHEDULE_DAY
|
|
||||||
: selectedItem.flags
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
// Helper function to handle save operations
|
const save = async () => {
|
||||||
const handleSave = useCallback(
|
try {
|
||||||
async (itemToSave: ScheduleItem) => {
|
setFieldErrors(undefined);
|
||||||
try {
|
await validate(validator, editItem);
|
||||||
setFieldErrors(undefined);
|
onSave(editItem);
|
||||||
await validate(validator, itemToSave);
|
} catch (error) {
|
||||||
onSave(itemToSave);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
} catch (error) {
|
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[validator, onSave]
|
|
||||||
);
|
|
||||||
|
|
||||||
const save = useCallback(async () => {
|
|
||||||
await handleSave(editItem);
|
|
||||||
}, [editItem, handleSave]);
|
|
||||||
|
|
||||||
const saveandactivate = useCallback(async () => {
|
|
||||||
await handleSave({ ...editItem, active: true });
|
|
||||||
}, [editItem, handleSave]);
|
|
||||||
|
|
||||||
const remove = useCallback(() => {
|
|
||||||
onSave({ ...editItem, deleted: true });
|
|
||||||
}, [editItem, onSave]);
|
|
||||||
|
|
||||||
// Optimize DOW flag conversion
|
|
||||||
const getFlagDOWnumber = useCallback((flags: string[]) => {
|
|
||||||
return flags.reduce((acc, flag) => acc | Number(flag), 0) & FLAG_MASK_127;
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const getFlagDOWstring = useCallback((f: number) => {
|
|
||||||
return FLAG_VALUES.filter((flag) => (f & flag) === flag).map((flag) =>
|
|
||||||
String(flag)
|
|
||||||
);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
// Day of week display component
|
|
||||||
const DayOfWeekButton = useCallback(
|
|
||||||
(flag: number) => {
|
|
||||||
const dayIndex = Math.log2(flag);
|
|
||||||
const isSelected = (editItem.flags & flag) === flag;
|
|
||||||
return (
|
|
||||||
<Typography
|
|
||||||
sx={{ fontSize: TYPOGRAPHY_FONT_SIZE }}
|
|
||||||
color={isSelected ? 'primary' : 'grey'}
|
|
||||||
>
|
|
||||||
{dow[dayIndex]}
|
|
||||||
</Typography>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
[editItem.flags, dow]
|
|
||||||
);
|
|
||||||
|
|
||||||
const handleClose = useCallback(
|
|
||||||
(_event: React.SyntheticEvent, reason: 'backdropClick' | 'escapeKeyDown') => {
|
|
||||||
if (reason !== 'backdropClick') {
|
|
||||||
onClose();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[onClose]
|
|
||||||
);
|
|
||||||
|
|
||||||
const handleScheduleTypeChange = useCallback(
|
|
||||||
(_event: React.SyntheticEvent<HTMLElement>, flag: ScheduleFlag | null) => {
|
|
||||||
if (flag !== null) {
|
|
||||||
setFieldErrors(undefined); // clear any validation errors
|
|
||||||
setScheduleType(flag);
|
|
||||||
// wipe the time field when changing the schedule type
|
|
||||||
// set the flags based on type
|
|
||||||
const newFlags = flag === ScheduleFlag.SCHEDULE_DAY ? FLAG_ALL_DAYS : flag;
|
|
||||||
setEditItem((prev) => ({ ...prev, time: '', flags: newFlags }));
|
|
||||||
}
|
|
||||||
},
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
const handleDOWChange = useCallback(
|
|
||||||
(_event: React.SyntheticEvent<HTMLElement>, flags: string[]) => {
|
|
||||||
const newFlags =
|
|
||||||
getFlagDOWnumber(flags) === 0 ? FLAG_ALL_DAYS : getFlagDOWnumber(flags);
|
|
||||||
setEditItem((prev) => ({ ...prev, flags: newFlags }));
|
|
||||||
},
|
|
||||||
[getFlagDOWnumber]
|
|
||||||
);
|
|
||||||
|
|
||||||
// Memoize derived values
|
|
||||||
const isDaySchedule = useMemo(
|
|
||||||
() => scheduleType === ScheduleFlag.SCHEDULE_DAY,
|
|
||||||
[scheduleType]
|
|
||||||
);
|
|
||||||
const isTimerSchedule = useMemo(
|
|
||||||
() => scheduleType === ScheduleFlag.SCHEDULE_TIMER,
|
|
||||||
[scheduleType]
|
|
||||||
);
|
|
||||||
const isImmediateSchedule = useMemo(
|
|
||||||
() => scheduleType === ScheduleFlag.SCHEDULE_IMMEDIATE,
|
|
||||||
[scheduleType]
|
|
||||||
);
|
|
||||||
const needsTimeField = useMemo(
|
|
||||||
() => isDaySchedule || isTimerSchedule,
|
|
||||||
[isDaySchedule, isTimerSchedule]
|
|
||||||
);
|
|
||||||
|
|
||||||
const dowFlags = useMemo(
|
|
||||||
() => getFlagDOWstring(editItem.flags),
|
|
||||||
[editItem.flags, getFlagDOWstring]
|
|
||||||
);
|
|
||||||
|
|
||||||
const timeFieldValue = useMemo(() => {
|
|
||||||
if (needsTimeField) {
|
|
||||||
return editItem.time === '' ? DEFAULT_TIME : editItem.time;
|
|
||||||
}
|
}
|
||||||
return editItem.time === DEFAULT_TIME ? '' : editItem.time;
|
};
|
||||||
}, [editItem.time, needsTimeField]);
|
|
||||||
|
|
||||||
const timeFieldLabel = useMemo(() => {
|
const saveandactivate = async () => {
|
||||||
if (scheduleType === ScheduleFlag.SCHEDULE_TIMER) return LL.TIMER(1);
|
editItem.active = true;
|
||||||
if (scheduleType === ScheduleFlag.SCHEDULE_CONDITION) return LL.CONDITION();
|
try {
|
||||||
if (scheduleType === ScheduleFlag.SCHEDULE_ONCHANGE) return LL.ONCHANGE();
|
setFieldErrors(undefined);
|
||||||
if (scheduleType === ScheduleFlag.SCHEDULE_IMMEDIATE) return LL.IMMEDIATE();
|
await validate(validator, editItem);
|
||||||
return LL.TIME(1);
|
onSave(editItem);
|
||||||
}, [scheduleType, LL]);
|
} catch (error) {
|
||||||
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const remove = () => {
|
||||||
|
editItem.deleted = true;
|
||||||
|
onSave(editItem);
|
||||||
|
};
|
||||||
|
|
||||||
|
const getFlagDOWnumber = (newFlag: string[]) => {
|
||||||
|
let new_flag = 0;
|
||||||
|
for (const entry of newFlag) {
|
||||||
|
new_flag |= Number(entry);
|
||||||
|
}
|
||||||
|
return new_flag & 127;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getFlagDOWstring = (f: number) => {
|
||||||
|
const new_flags: string[] = [];
|
||||||
|
if ((f & 129) === 1) {
|
||||||
|
new_flags.push('1');
|
||||||
|
}
|
||||||
|
if ((f & 130) === 2) {
|
||||||
|
new_flags.push('2');
|
||||||
|
}
|
||||||
|
if ((f & 4) === 4) {
|
||||||
|
new_flags.push('4');
|
||||||
|
}
|
||||||
|
if ((f & 8) === 8) {
|
||||||
|
new_flags.push('8');
|
||||||
|
}
|
||||||
|
if ((f & 16) === 16) {
|
||||||
|
new_flags.push('16');
|
||||||
|
}
|
||||||
|
if ((f & 32) === 32) {
|
||||||
|
new_flags.push('32');
|
||||||
|
}
|
||||||
|
if ((f & 64) === 64) {
|
||||||
|
new_flags.push('64');
|
||||||
|
}
|
||||||
|
|
||||||
|
return new_flags;
|
||||||
|
};
|
||||||
|
|
||||||
|
const showDOW = (si: ScheduleItem, flag: number) => (
|
||||||
|
<Typography
|
||||||
|
sx={{ fontSize: 10 }}
|
||||||
|
color={(si.flags & flag) === flag ? 'primary' : 'grey'}
|
||||||
|
>
|
||||||
|
{dow[Math.log(flag) / Math.log(2)]}
|
||||||
|
</Typography>
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleClose = (_event, reason: 'backdropClick' | 'escapeKeyDown') => {
|
||||||
|
if (reason !== 'backdropClick') {
|
||||||
|
onClose();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
||||||
<DialogTitle>
|
<DialogTitle>
|
||||||
{creating ? `${LL.ADD(1)} ${LL.NEW(0)}` : LL.EDIT()}
|
{creating ? LL.ADD(1) + ' ' + LL.NEW(0) : LL.EDIT()}
|
||||||
{LL.SCHEDULE(1)}
|
{LL.SCHEDULE(1)}
|
||||||
</DialogTitle>
|
</DialogTitle>
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
@@ -249,27 +163,47 @@ const SchedulerDialog = ({
|
|||||||
value={scheduleType}
|
value={scheduleType}
|
||||||
exclusive
|
exclusive
|
||||||
disabled={!creating}
|
disabled={!creating}
|
||||||
onChange={handleScheduleTypeChange}
|
onChange={(_event, flag: ScheduleFlag) => {
|
||||||
|
if (flag !== null) {
|
||||||
|
setFieldErrors(undefined); // clear any validation errors
|
||||||
|
setScheduleType(flag);
|
||||||
|
// wipe the time field when changing the schedule type
|
||||||
|
setEditItem({ ...editItem, time: '' });
|
||||||
|
// set the flags based on type
|
||||||
|
// 0-127 is day schedule
|
||||||
|
// 128 is timer
|
||||||
|
// 129 is on change
|
||||||
|
// 130 is on condition
|
||||||
|
// 132 is immediate
|
||||||
|
setEditItem(
|
||||||
|
flag === ScheduleFlag.SCHEDULE_DAY
|
||||||
|
? { ...editItem, flags: 0 }
|
||||||
|
: { ...editItem, flags: flag }
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
<ToggleButton value={ScheduleFlag.SCHEDULE_DAY}>
|
<ToggleButton value={ScheduleFlag.SCHEDULE_DAY}>
|
||||||
<Typography
|
<Typography
|
||||||
sx={{ fontSize: TYPOGRAPHY_FONT_SIZE }}
|
sx={{ fontSize: 10 }}
|
||||||
color={isDaySchedule ? 'primary' : 'grey'}
|
color={scheduleType === ScheduleFlag.SCHEDULE_DAY ? 'primary' : 'grey'}
|
||||||
>
|
>
|
||||||
{LL.SCHEDULE(0)}
|
{LL.SCHEDULE(0)}
|
||||||
</Typography>
|
</Typography>
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value={ScheduleFlag.SCHEDULE_TIMER}>
|
<ToggleButton value={ScheduleFlag.SCHEDULE_TIMER}>
|
||||||
<Typography
|
<Typography
|
||||||
sx={{ fontSize: TYPOGRAPHY_FONT_SIZE }}
|
sx={{ fontSize: 10 }}
|
||||||
color={isTimerSchedule ? 'primary' : 'grey'}
|
color={
|
||||||
|
scheduleType === ScheduleFlag.SCHEDULE_TIMER ? 'primary' : 'grey'
|
||||||
|
}
|
||||||
>
|
>
|
||||||
{LL.TIMER(0)}
|
{LL.TIMER(0)}
|
||||||
</Typography>
|
</Typography>
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value={ScheduleFlag.SCHEDULE_ONCHANGE}>
|
<ToggleButton value={ScheduleFlag.SCHEDULE_ONCHANGE}>
|
||||||
<Typography
|
<Typography
|
||||||
sx={{ fontSize: TYPOGRAPHY_FONT_SIZE }}
|
sx={{ fontSize: 10 }}
|
||||||
color={
|
color={
|
||||||
scheduleType === ScheduleFlag.SCHEDULE_ONCHANGE ? 'primary' : 'grey'
|
scheduleType === ScheduleFlag.SCHEDULE_ONCHANGE ? 'primary' : 'grey'
|
||||||
}
|
}
|
||||||
@@ -279,7 +213,7 @@ const SchedulerDialog = ({
|
|||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value={ScheduleFlag.SCHEDULE_CONDITION}>
|
<ToggleButton value={ScheduleFlag.SCHEDULE_CONDITION}>
|
||||||
<Typography
|
<Typography
|
||||||
sx={{ fontSize: TYPOGRAPHY_FONT_SIZE }}
|
sx={{ fontSize: 10 }}
|
||||||
color={
|
color={
|
||||||
scheduleType === ScheduleFlag.SCHEDULE_CONDITION ? 'primary' : 'grey'
|
scheduleType === ScheduleFlag.SCHEDULE_CONDITION ? 'primary' : 'grey'
|
||||||
}
|
}
|
||||||
@@ -289,30 +223,50 @@ const SchedulerDialog = ({
|
|||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
<ToggleButton value={ScheduleFlag.SCHEDULE_IMMEDIATE}>
|
<ToggleButton value={ScheduleFlag.SCHEDULE_IMMEDIATE}>
|
||||||
<Typography
|
<Typography
|
||||||
sx={{ fontSize: TYPOGRAPHY_FONT_SIZE }}
|
sx={{ fontSize: 10 }}
|
||||||
color={isImmediateSchedule ? 'primary' : 'grey'}
|
color={
|
||||||
|
scheduleType === ScheduleFlag.SCHEDULE_IMMEDIATE ? 'primary' : 'grey'
|
||||||
|
}
|
||||||
>
|
>
|
||||||
{LL.IMMEDIATE()}
|
{LL.IMMEDIATE()}
|
||||||
</Typography>
|
</Typography>
|
||||||
</ToggleButton>
|
</ToggleButton>
|
||||||
</ToggleButtonGroup>
|
</ToggleButtonGroup>
|
||||||
|
|
||||||
{isDaySchedule && (
|
{scheduleType === ScheduleFlag.SCHEDULE_DAY && (
|
||||||
<ToggleButtonGroup
|
<ToggleButtonGroup
|
||||||
size="small"
|
size="small"
|
||||||
color="secondary"
|
color="secondary"
|
||||||
value={dowFlags}
|
value={getFlagDOWstring(editItem.flags)}
|
||||||
onChange={handleDOWChange}
|
onChange={(_event, flag: string[]) => {
|
||||||
|
setEditItem({ ...editItem, flags: getFlagDOWnumber(flag) });
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
{DAY_FLAGS.map(({ value, flag }) => (
|
<ToggleButton value="2">
|
||||||
<ToggleButton key={value} value={value}>
|
{showDOW(editItem, ScheduleFlag.SCHEDULE_MON)}
|
||||||
{DayOfWeekButton(flag)}
|
</ToggleButton>
|
||||||
</ToggleButton>
|
<ToggleButton value="4">
|
||||||
))}
|
{showDOW(editItem, ScheduleFlag.SCHEDULE_TUE)}
|
||||||
|
</ToggleButton>
|
||||||
|
<ToggleButton value="8">
|
||||||
|
{showDOW(editItem, ScheduleFlag.SCHEDULE_WED)}
|
||||||
|
</ToggleButton>
|
||||||
|
<ToggleButton value="16">
|
||||||
|
{showDOW(editItem, ScheduleFlag.SCHEDULE_THU)}
|
||||||
|
</ToggleButton>
|
||||||
|
<ToggleButton value="32">
|
||||||
|
{showDOW(editItem, ScheduleFlag.SCHEDULE_FRI)}
|
||||||
|
</ToggleButton>
|
||||||
|
<ToggleButton value="64">
|
||||||
|
{showDOW(editItem, ScheduleFlag.SCHEDULE_SAT)}
|
||||||
|
</ToggleButton>
|
||||||
|
<ToggleButton value="1">
|
||||||
|
{showDOW(editItem, ScheduleFlag.SCHEDULE_SUN)}
|
||||||
|
</ToggleButton>
|
||||||
</ToggleButtonGroup>
|
</ToggleButtonGroup>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{!isImmediateSchedule && (
|
{scheduleType !== ScheduleFlag.SCHEDULE_IMMEDIATE && (
|
||||||
<>
|
<>
|
||||||
<Grid container>
|
<Grid container>
|
||||||
<BlockFormControlLabel
|
<BlockFormControlLabel
|
||||||
@@ -327,17 +281,22 @@ const SchedulerDialog = ({
|
|||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid container>
|
<Grid container>
|
||||||
{needsTimeField ? (
|
{scheduleType === ScheduleFlag.SCHEDULE_DAY ||
|
||||||
|
scheduleType === ScheduleFlag.SCHEDULE_TIMER ? (
|
||||||
<>
|
<>
|
||||||
<TextField
|
<TextField
|
||||||
name="time"
|
name="time"
|
||||||
type="time"
|
type="time"
|
||||||
label={timeFieldLabel}
|
label={
|
||||||
value={timeFieldValue}
|
scheduleType === ScheduleFlag.SCHEDULE_TIMER
|
||||||
|
? LL.TIMER(1)
|
||||||
|
: LL.TIME(1)
|
||||||
|
}
|
||||||
|
value={editItem.time === '' ? '00:00' : editItem.time}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
/>
|
/>
|
||||||
{isTimerSchedule && (
|
{scheduleType === ScheduleFlag.SCHEDULE_TIMER && (
|
||||||
<Box color="warning.main" ml={2} mt={4}>
|
<Box color="warning.main" ml={2} mt={4}>
|
||||||
<Typography variant="body2">
|
<Typography variant="body2">
|
||||||
{LL.SCHEDULER_HELP_2()}
|
{LL.SCHEDULER_HELP_2()}
|
||||||
@@ -348,10 +307,16 @@ const SchedulerDialog = ({
|
|||||||
) : (
|
) : (
|
||||||
<TextField
|
<TextField
|
||||||
name="time"
|
name="time"
|
||||||
label={timeFieldLabel}
|
label={
|
||||||
|
scheduleType === ScheduleFlag.SCHEDULE_CONDITION
|
||||||
|
? LL.CONDITION()
|
||||||
|
: scheduleType === ScheduleFlag.SCHEDULE_ONCHANGE
|
||||||
|
? LL.ONCHANGE()
|
||||||
|
: LL.IMMEDIATE()
|
||||||
|
}
|
||||||
multiline
|
multiline
|
||||||
fullWidth
|
fullWidth
|
||||||
value={timeFieldValue}
|
value={editItem.time === '00:00' ? '' : editItem.time}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
/>
|
/>
|
||||||
@@ -360,7 +325,7 @@ const SchedulerDialog = ({
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="cmd"
|
name="cmd"
|
||||||
label={LL.COMMAND(0)}
|
label={LL.COMMAND(0)}
|
||||||
multiline
|
multiline
|
||||||
@@ -379,7 +344,7 @@ const SchedulerDialog = ({
|
|||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
/>
|
/>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="name"
|
name="name"
|
||||||
label={LL.NAME(0) + ' (' + LL.OPTIONAL() + ')'}
|
label={LL.NAME(0) + ' (' + LL.OPTIONAL() + ')'}
|
||||||
value={editItem.name}
|
value={editItem.name}
|
||||||
@@ -418,7 +383,7 @@ const SchedulerDialog = ({
|
|||||||
>
|
>
|
||||||
{creating ? LL.ADD(0) : LL.UPDATE()}
|
{creating ? LL.ADD(0) : LL.UPDATE()}
|
||||||
</Button>
|
</Button>
|
||||||
{isImmediateSchedule && editItem.cmd !== '' && (
|
{scheduleType === ScheduleFlag.SCHEDULE_IMMEDIATE && editItem.cmd !== '' && (
|
||||||
<Button
|
<Button
|
||||||
startIcon={<PlayArrowIcon />}
|
startIcon={<PlayArrowIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useContext, useMemo, useRef, useState } from 'react';
|
import { useContext, useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import AddCircleOutlineOutlinedIcon from '@mui/icons-material/AddCircleOutlineOutlined';
|
import AddCircleOutlineOutlinedIcon from '@mui/icons-material/AddCircleOutlineOutlined';
|
||||||
@@ -49,74 +49,6 @@ import {
|
|||||||
temperatureSensorItemValidation
|
temperatureSensorItemValidation
|
||||||
} from './validators';
|
} from './validators';
|
||||||
|
|
||||||
// Constants
|
|
||||||
const MS_PER_SECOND = 1000;
|
|
||||||
const MS_PER_MINUTE = 60 * MS_PER_SECOND;
|
|
||||||
const MS_PER_HOUR = 60 * MS_PER_MINUTE;
|
|
||||||
const MS_PER_DAY = 24 * MS_PER_HOUR;
|
|
||||||
const MIN_TEMP_ID = -100;
|
|
||||||
const MAX_TEMP_ID = 100;
|
|
||||||
const GPIO_25 = 25;
|
|
||||||
const GPIO_26 = 26;
|
|
||||||
|
|
||||||
const HEADER_BUTTON_STYLE: React.CSSProperties = {
|
|
||||||
fontSize: '14px',
|
|
||||||
justifyContent: 'flex-start'
|
|
||||||
};
|
|
||||||
|
|
||||||
const HEADER_BUTTON_STYLE_END: React.CSSProperties = {
|
|
||||||
fontSize: '14px',
|
|
||||||
justifyContent: 'flex-end'
|
|
||||||
};
|
|
||||||
|
|
||||||
const common_theme = {
|
|
||||||
BaseRow: `
|
|
||||||
font-size: 14px;
|
|
||||||
.td {
|
|
||||||
height: 32px;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
HeaderRow: `
|
|
||||||
text-transform: uppercase;
|
|
||||||
background-color: black;
|
|
||||||
color: #90CAF9;
|
|
||||||
.th {
|
|
||||||
border-bottom: 1px solid #565656;
|
|
||||||
height: 36px;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
Row: `
|
|
||||||
background-color: #1e1e1e;
|
|
||||||
position: relative;
|
|
||||||
cursor: pointer;
|
|
||||||
.td {
|
|
||||||
padding: 8px;
|
|
||||||
border-bottom: 1px solid #565656;
|
|
||||||
}
|
|
||||||
&:hover .td {
|
|
||||||
background-color: #177ac9;
|
|
||||||
color: white;
|
|
||||||
}
|
|
||||||
`,
|
|
||||||
Cell: `
|
|
||||||
&:last-of-type {
|
|
||||||
text-align: right;
|
|
||||||
},
|
|
||||||
`
|
|
||||||
};
|
|
||||||
|
|
||||||
const temperature_theme_config = {
|
|
||||||
Table: `
|
|
||||||
--data-table-library_grid-template-columns: minmax(0, 1fr) 35%;
|
|
||||||
`
|
|
||||||
};
|
|
||||||
|
|
||||||
const analog_theme_config = {
|
|
||||||
Table: `
|
|
||||||
--data-table-library_grid-template-columns: 80px repeat(1, minmax(0, 1fr)) 120px 110px;
|
|
||||||
`
|
|
||||||
};
|
|
||||||
|
|
||||||
const Sensors = () => {
|
const Sensors = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const { me } = useContext(AuthenticatedContext);
|
const { me } = useContext(AuthenticatedContext);
|
||||||
@@ -127,22 +59,18 @@ const Sensors = () => {
|
|||||||
const [temperatureDialogOpen, setTemperatureDialogOpen] = useState<boolean>(false);
|
const [temperatureDialogOpen, setTemperatureDialogOpen] = useState<boolean>(false);
|
||||||
const [analogDialogOpen, setAnalogDialogOpen] = useState<boolean>(false);
|
const [analogDialogOpen, setAnalogDialogOpen] = useState<boolean>(false);
|
||||||
const [creating, setCreating] = useState<boolean>(false);
|
const [creating, setCreating] = useState<boolean>(false);
|
||||||
const firstAvailableGPIO = useRef<number>(undefined);
|
|
||||||
|
|
||||||
const { data: sensorData, send: fetchSensorData } = useRequest(readSensorData, {
|
const { data: sensorData, send: fetchSensorData } = useRequest(
|
||||||
initialData: {
|
() => readSensorData(),
|
||||||
ts: [],
|
{
|
||||||
as: [],
|
initialData: {
|
||||||
analog_enabled: false,
|
ts: [],
|
||||||
available_gpios: [] as number[],
|
as: [],
|
||||||
platform: 'ESP32'
|
analog_enabled: false,
|
||||||
|
platform: 'ESP32'
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}).onSuccess((event) => {
|
);
|
||||||
// store the first available GPIO in a ref
|
|
||||||
if (event.data.available_gpios.length > 0) {
|
|
||||||
firstAvailableGPIO.current = event.data.available_gpios[0];
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
const { send: sendTemperatureSensor } = useRequest(
|
const { send: sendTemperatureSensor } = useRequest(
|
||||||
(data: WriteTemperatureSensor) => writeTemperatureSensor(data),
|
(data: WriteTemperatureSensor) => writeTemperatureSensor(data),
|
||||||
@@ -158,18 +86,118 @@ const Sensors = () => {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
const intervalCallback = useCallback(() => {
|
useInterval(() => {
|
||||||
if (!temperatureDialogOpen && !analogDialogOpen) {
|
if (!temperatureDialogOpen && !analogDialogOpen) {
|
||||||
void fetchSensorData();
|
void fetchSensorData();
|
||||||
}
|
}
|
||||||
}, [temperatureDialogOpen, analogDialogOpen, fetchSensorData]);
|
});
|
||||||
|
|
||||||
useInterval(intervalCallback);
|
const common_theme = useTheme({
|
||||||
|
BaseRow: `
|
||||||
|
font-size: 14px;
|
||||||
|
.td {
|
||||||
|
height: 32px;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
HeaderRow: `
|
||||||
|
text-transform: uppercase;
|
||||||
|
background-color: black;
|
||||||
|
color: #90CAF9;
|
||||||
|
.th {
|
||||||
|
border-bottom: 1px solid #565656;
|
||||||
|
}
|
||||||
|
.th {
|
||||||
|
height: 36px;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
Row: `
|
||||||
|
background-color: #1e1e1e;
|
||||||
|
position: relative;
|
||||||
|
cursor: pointer;
|
||||||
|
.td {
|
||||||
|
padding: 8px;
|
||||||
|
border-bottom: 1px solid #565656;
|
||||||
|
}
|
||||||
|
&:hover .td {
|
||||||
|
background-color: #177ac9;
|
||||||
|
}
|
||||||
|
`,
|
||||||
|
Cell: `
|
||||||
|
&:last-of-type {
|
||||||
|
text-align: right;
|
||||||
|
},
|
||||||
|
`
|
||||||
|
});
|
||||||
|
|
||||||
const temperature_theme = useTheme([common_theme, temperature_theme_config]);
|
const temperature_theme = useTheme([
|
||||||
const analog_theme = useTheme([common_theme, analog_theme_config]);
|
common_theme,
|
||||||
|
{
|
||||||
|
Table: `
|
||||||
|
--data-table-library_grid-template-columns: minmax(0, 1fr) 35%;
|
||||||
|
`
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
|
||||||
const getSortIcon = useCallback((state: State, sortKey: unknown) => {
|
const analog_theme = useTheme([
|
||||||
|
common_theme,
|
||||||
|
{
|
||||||
|
Table: `
|
||||||
|
--data-table-library_grid-template-columns: 80px repeat(1, minmax(0, 1fr)) 120px 110px;
|
||||||
|
`
|
||||||
|
}
|
||||||
|
]);
|
||||||
|
|
||||||
|
const RenderTemperatureSensors = () => (
|
||||||
|
<Table
|
||||||
|
data={{ nodes: sensorData.ts }}
|
||||||
|
theme={temperature_theme}
|
||||||
|
sort={temperature_sort}
|
||||||
|
layout={{ custom: true }}
|
||||||
|
>
|
||||||
|
{(tableList: TemperatureSensor[]) => (
|
||||||
|
<>
|
||||||
|
<Header>
|
||||||
|
<HeaderRow>
|
||||||
|
<HeaderCell resize>
|
||||||
|
<Button
|
||||||
|
fullWidth
|
||||||
|
style={{ fontSize: '14px', justifyContent: 'flex-start' }}
|
||||||
|
endIcon={getSortIcon(temperature_sort.state, 'NAME')}
|
||||||
|
onClick={() =>
|
||||||
|
temperature_sort.fns.onToggleSort({ sortKey: 'NAME' })
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{LL.NAME(0)}
|
||||||
|
</Button>
|
||||||
|
</HeaderCell>
|
||||||
|
<HeaderCell stiff>
|
||||||
|
<Button
|
||||||
|
fullWidth
|
||||||
|
style={{ fontSize: '14px', justifyContent: 'flex-end' }}
|
||||||
|
endIcon={getSortIcon(temperature_sort.state, 'VALUE')}
|
||||||
|
onClick={() =>
|
||||||
|
temperature_sort.fns.onToggleSort({ sortKey: 'VALUE' })
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{LL.VALUE(0)}
|
||||||
|
</Button>
|
||||||
|
</HeaderCell>
|
||||||
|
</HeaderRow>
|
||||||
|
</Header>
|
||||||
|
<Body>
|
||||||
|
{tableList.map((ts: TemperatureSensor) => (
|
||||||
|
<Row key={ts.id} item={ts} onClick={() => updateTemperatureSensor(ts)}>
|
||||||
|
<Cell>{ts.n}</Cell>
|
||||||
|
<Cell>{formatValue(ts.t, ts.u)}</Cell>
|
||||||
|
</Row>
|
||||||
|
))}
|
||||||
|
</Body>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Table>
|
||||||
|
);
|
||||||
|
|
||||||
|
const getSortIcon = (state: State, sortKey: unknown) => {
|
||||||
if (state.sortKey === sortKey && state.reverse) {
|
if (state.sortKey === sortKey && state.reverse) {
|
||||||
return <KeyboardArrowDownOutlinedIcon />;
|
return <KeyboardArrowDownOutlinedIcon />;
|
||||||
}
|
}
|
||||||
@@ -177,7 +205,7 @@ const Sensors = () => {
|
|||||||
return <KeyboardArrowUpOutlinedIcon />;
|
return <KeyboardArrowUpOutlinedIcon />;
|
||||||
}
|
}
|
||||||
return <UnfoldMoreOutlinedIcon />;
|
return <UnfoldMoreOutlinedIcon />;
|
||||||
}, []);
|
};
|
||||||
|
|
||||||
const analog_sort = useSort(
|
const analog_sort = useSort(
|
||||||
{ nodes: sensorData.as },
|
{ nodes: sensorData.as },
|
||||||
@@ -190,20 +218,11 @@ const Sensors = () => {
|
|||||||
},
|
},
|
||||||
sortToggleType: SortToggleType.AlternateWithReset,
|
sortToggleType: SortToggleType.AlternateWithReset,
|
||||||
sortFns: {
|
sortFns: {
|
||||||
GPIO: (array) =>
|
GPIO: (array) => array.sort((a, b) => a.g - b.g),
|
||||||
[...array].sort(
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-return
|
||||||
(a, b) => ((a as AnalogSensor)?.g ?? 0) - ((b as AnalogSensor)?.g ?? 0)
|
NAME: (array) => array.sort((a, b) => a.n.localeCompare(b.n)),
|
||||||
),
|
TYPE: (array) => array.sort((a, b) => a.t - b.t),
|
||||||
NAME: (array) =>
|
VALUE: (array) => array.sort((a, b) => a.v - b.v)
|
||||||
[...array].sort((a, b) =>
|
|
||||||
((a as AnalogSensor)?.n ?? '').localeCompare(
|
|
||||||
(b as AnalogSensor)?.n ?? ''
|
|
||||||
)
|
|
||||||
),
|
|
||||||
TYPE: (array) =>
|
|
||||||
[...array].sort((a, b) => (a as AnalogSensor).t - (b as AnalogSensor).t),
|
|
||||||
VALUE: (array) =>
|
|
||||||
[...array].sort((a, b) => (a as AnalogSensor).v - (b as AnalogSensor).v)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@@ -219,349 +238,226 @@ const Sensors = () => {
|
|||||||
},
|
},
|
||||||
sortToggleType: SortToggleType.AlternateWithReset,
|
sortToggleType: SortToggleType.AlternateWithReset,
|
||||||
sortFns: {
|
sortFns: {
|
||||||
NAME: (array) =>
|
// eslint-disable-next-line @typescript-eslint/no-unsafe-return
|
||||||
[...array].sort((a, b) =>
|
NAME: (array) => array.sort((a, b) => a.n.localeCompare(b.n)),
|
||||||
(a as TemperatureSensor).n.localeCompare((b as TemperatureSensor).n)
|
VALUE: (array) => array.sort((a, b) => a.t - b.t)
|
||||||
),
|
|
||||||
VALUE: (array) =>
|
|
||||||
[...array].sort(
|
|
||||||
(a, b) =>
|
|
||||||
((a as TemperatureSensor).t ?? 0) - ((b as TemperatureSensor).t ?? 0)
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
useLayoutTitle(LL.SENSORS());
|
useLayoutTitle(LL.SENSORS());
|
||||||
|
|
||||||
const formatDurationMin = useCallback(
|
const formatDurationMin = (duration_min: number) => {
|
||||||
(duration_min: number) => {
|
const days = Math.trunc((duration_min * 60000) / 86400000);
|
||||||
const totalMs = duration_min * MS_PER_MINUTE;
|
const hours = Math.trunc((duration_min * 60000) / 3600000) % 24;
|
||||||
const days = Math.trunc(totalMs / MS_PER_DAY);
|
const minutes = Math.trunc((duration_min * 60000) / 60000) % 60;
|
||||||
const hours = Math.trunc(totalMs / MS_PER_HOUR) % 24;
|
|
||||||
const minutes = Math.trunc(totalMs / MS_PER_MINUTE) % 60;
|
|
||||||
|
|
||||||
const parts: string[] = [];
|
let formatted = '';
|
||||||
if (days > 0) {
|
if (days) {
|
||||||
parts.push(LL.NUM_DAYS({ num: days }));
|
formatted += LL.NUM_DAYS({ num: days }) + ' ';
|
||||||
}
|
}
|
||||||
if (hours > 0) {
|
if (hours) {
|
||||||
parts.push(LL.NUM_HOURS({ num: hours }));
|
formatted += LL.NUM_HOURS({ num: hours }) + ' ';
|
||||||
}
|
}
|
||||||
if (minutes > 0) {
|
if (minutes) {
|
||||||
parts.push(LL.NUM_MINUTES({ num: minutes }));
|
formatted += LL.NUM_MINUTES({ num: minutes });
|
||||||
}
|
}
|
||||||
return parts.join(' ');
|
return formatted;
|
||||||
},
|
};
|
||||||
[LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const formatValue = useCallback(
|
function formatValue(value: unknown, uom: DeviceValueUOM) {
|
||||||
(value: unknown, uom: DeviceValueUOM) => {
|
if (value === undefined) {
|
||||||
if (value === undefined) {
|
return '';
|
||||||
return '';
|
}
|
||||||
}
|
if (typeof value !== 'number') {
|
||||||
if (typeof value !== 'number') {
|
return value as string;
|
||||||
return value as string;
|
}
|
||||||
}
|
switch (uom) {
|
||||||
switch (uom) {
|
case DeviceValueUOM.HOURS:
|
||||||
case DeviceValueUOM.HOURS:
|
return value ? formatDurationMin(value * 60) : LL.NUM_HOURS({ num: 0 });
|
||||||
return value ? formatDurationMin(value * 60) : LL.NUM_HOURS({ num: 0 });
|
case DeviceValueUOM.MINUTES:
|
||||||
case DeviceValueUOM.MINUTES:
|
return value ? formatDurationMin(value) : LL.NUM_MINUTES({ num: 0 });
|
||||||
return value ? formatDurationMin(value) : LL.NUM_MINUTES({ num: 0 });
|
case DeviceValueUOM.SECONDS:
|
||||||
case DeviceValueUOM.SECONDS:
|
return LL.NUM_SECONDS({ num: value });
|
||||||
return LL.NUM_SECONDS({ num: value });
|
case DeviceValueUOM.NONE:
|
||||||
case DeviceValueUOM.NONE:
|
return new Intl.NumberFormat().format(value);
|
||||||
return new Intl.NumberFormat().format(value);
|
case DeviceValueUOM.DEGREES:
|
||||||
case DeviceValueUOM.DEGREES:
|
case DeviceValueUOM.DEGREES_R:
|
||||||
case DeviceValueUOM.DEGREES_R:
|
case DeviceValueUOM.FAHRENHEIT:
|
||||||
case DeviceValueUOM.FAHRENHEIT:
|
return (
|
||||||
return (
|
new Intl.NumberFormat(undefined, {
|
||||||
new Intl.NumberFormat(undefined, {
|
minimumFractionDigits: 1
|
||||||
minimumFractionDigits: 1
|
}).format(value) +
|
||||||
}).format(value) +
|
' ' +
|
||||||
' ' +
|
DeviceValueUOM_s[uom]
|
||||||
DeviceValueUOM_s[uom]
|
);
|
||||||
);
|
default:
|
||||||
default:
|
return new Intl.NumberFormat().format(value) + ' ' + DeviceValueUOM_s[uom];
|
||||||
return new Intl.NumberFormat().format(value) + ' ' + DeviceValueUOM_s[uom];
|
}
|
||||||
}
|
}
|
||||||
},
|
|
||||||
[formatDurationMin, LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const updateTemperatureSensor = useCallback(
|
const updateTemperatureSensor = (ts: TemperatureSensor) => {
|
||||||
(ts: TemperatureSensor) => {
|
if (me.admin) {
|
||||||
if (me.admin) {
|
ts.o_n = ts.n;
|
||||||
ts.o_n = ts.n;
|
setSelectedTemperatureSensor(ts);
|
||||||
setSelectedTemperatureSensor(ts);
|
setTemperatureDialogOpen(true);
|
||||||
setTemperatureDialogOpen(true);
|
}
|
||||||
}
|
};
|
||||||
},
|
|
||||||
[me.admin]
|
|
||||||
);
|
|
||||||
|
|
||||||
const onTemperatureDialogClose = useCallback(() => {
|
const onTemperatureDialogClose = () => {
|
||||||
setTemperatureDialogOpen(false);
|
setTemperatureDialogOpen(false);
|
||||||
void fetchSensorData();
|
void fetchSensorData();
|
||||||
}, [fetchSensorData]);
|
};
|
||||||
|
|
||||||
const onTemperatureDialogSave = useCallback(
|
const onTemperatureDialogSave = async (ts: TemperatureSensor) => {
|
||||||
async (ts: TemperatureSensor) => {
|
await sendTemperatureSensor({ id: ts.id, name: ts.n, offset: ts.o })
|
||||||
await sendTemperatureSensor({
|
.then(() => {
|
||||||
id: ts.id,
|
toast.success(LL.UPDATED_OF(LL.SENSOR(1)));
|
||||||
name: ts.n,
|
|
||||||
offset: ts.o,
|
|
||||||
is_system: ts.s
|
|
||||||
})
|
})
|
||||||
.then(() => {
|
.catch(() => {
|
||||||
toast.success(LL.UPDATED_OF(LL.SENSOR(1)));
|
toast.error(LL.UPDATE_OF(LL.SENSOR(2)) + ' ' + LL.FAILED(1));
|
||||||
})
|
})
|
||||||
.catch(() => {
|
.finally(() => {
|
||||||
toast.error(LL.UPDATE_OF(LL.SENSOR(2)) + ' ' + LL.FAILED(1));
|
setTemperatureDialogOpen(false);
|
||||||
})
|
setSelectedTemperatureSensor(undefined);
|
||||||
.finally(() => {
|
void fetchSensorData();
|
||||||
setTemperatureDialogOpen(false);
|
});
|
||||||
setSelectedTemperatureSensor(undefined);
|
};
|
||||||
void fetchSensorData();
|
|
||||||
});
|
|
||||||
},
|
|
||||||
[sendTemperatureSensor, LL, fetchSensorData]
|
|
||||||
);
|
|
||||||
|
|
||||||
const updateAnalogSensor = useCallback(
|
const updateAnalogSensor = (as: AnalogSensor) => {
|
||||||
(as: AnalogSensor) => {
|
if (me.admin) {
|
||||||
if (me.admin) {
|
setCreating(false);
|
||||||
setCreating(false);
|
as.o_n = as.n;
|
||||||
as.o_n = as.n;
|
setSelectedAnalogSensor(as);
|
||||||
setSelectedAnalogSensor(as);
|
setAnalogDialogOpen(true);
|
||||||
setAnalogDialogOpen(true);
|
}
|
||||||
}
|
};
|
||||||
},
|
|
||||||
[me.admin]
|
|
||||||
);
|
|
||||||
|
|
||||||
const onAnalogDialogClose = useCallback(() => {
|
const onAnalogDialogClose = () => {
|
||||||
setAnalogDialogOpen(false);
|
setAnalogDialogOpen(false);
|
||||||
void fetchSensorData();
|
void fetchSensorData();
|
||||||
}, [fetchSensorData]);
|
};
|
||||||
|
|
||||||
const addAnalogSensor = useCallback(() => {
|
const addAnalogSensor = () => {
|
||||||
if (firstAvailableGPIO.current === undefined) {
|
|
||||||
toast.error('No available GPIO found');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setCreating(true);
|
setCreating(true);
|
||||||
setSelectedAnalogSensor({
|
setSelectedAnalogSensor({
|
||||||
id: Math.floor(Math.random() * (MAX_TEMP_ID - MIN_TEMP_ID) + MIN_TEMP_ID),
|
id: Math.floor(Math.random() * (Math.floor(200) - 100) + 100),
|
||||||
n: '',
|
n: '',
|
||||||
g: firstAvailableGPIO.current,
|
g: 21, // default GPIO 21 which is safe for all platforms
|
||||||
u: DeviceValueUOM.NONE,
|
u: 0,
|
||||||
v: 0,
|
v: 0,
|
||||||
o: 0,
|
o: 0,
|
||||||
|
t: 0,
|
||||||
f: 1,
|
f: 1,
|
||||||
t: AnalogType.DIGITAL_IN, // default to digital in 1
|
|
||||||
d: false,
|
d: false,
|
||||||
s: false,
|
|
||||||
o_n: ''
|
o_n: ''
|
||||||
});
|
});
|
||||||
setAnalogDialogOpen(true);
|
setAnalogDialogOpen(true);
|
||||||
}, []);
|
};
|
||||||
|
|
||||||
const onAnalogDialogSave = useCallback(
|
const onAnalogDialogSave = async (as: AnalogSensor) => {
|
||||||
async (as: AnalogSensor) => {
|
await sendAnalogSensor({
|
||||||
await sendAnalogSensor({
|
id: as.id,
|
||||||
id: as.id,
|
gpio: as.g,
|
||||||
gpio: as.g,
|
name: as.n,
|
||||||
name: as.n,
|
offset: as.o,
|
||||||
offset: as.o,
|
factor: as.f,
|
||||||
factor: as.f,
|
uom: as.u,
|
||||||
uom: as.u,
|
type: as.t,
|
||||||
type: as.t,
|
deleted: as.d
|
||||||
deleted: as.d,
|
})
|
||||||
is_system: as.s
|
.then(() => {
|
||||||
|
toast.success(LL.UPDATED_OF(LL.ANALOG_SENSOR(2)));
|
||||||
})
|
})
|
||||||
.then(() => {
|
.catch(() => {
|
||||||
toast.success(LL.UPDATED_OF(LL.ANALOG_SENSOR(2)));
|
toast.error(LL.UPDATE_OF(LL.ANALOG_SENSOR(5)) + ' ' + LL.FAILED(1));
|
||||||
})
|
})
|
||||||
.catch(() => {
|
.finally(() => {
|
||||||
toast.error(LL.UPDATE_OF(LL.ANALOG_SENSOR(5)) + ' ' + LL.FAILED(1));
|
setAnalogDialogOpen(false);
|
||||||
})
|
setSelectedAnalogSensor(undefined);
|
||||||
.finally(() => {
|
void fetchSensorData();
|
||||||
setAnalogDialogOpen(false);
|
});
|
||||||
setSelectedAnalogSensor(undefined);
|
};
|
||||||
void fetchSensorData();
|
|
||||||
});
|
|
||||||
},
|
|
||||||
[sendAnalogSensor, LL, fetchSensorData]
|
|
||||||
);
|
|
||||||
|
|
||||||
const RenderAnalogSensors = useMemo(
|
const RenderAnalogSensors = () => (
|
||||||
() => (
|
<Table
|
||||||
<Table
|
data={{ nodes: sensorData.as }}
|
||||||
data={{ nodes: sensorData.as }}
|
theme={analog_theme}
|
||||||
theme={analog_theme}
|
sort={analog_sort}
|
||||||
sort={analog_sort}
|
layout={{ custom: true }}
|
||||||
layout={{ custom: true }}
|
>
|
||||||
>
|
{(tableList: AnalogSensor[]) => (
|
||||||
{(tableList: AnalogSensor[]) => (
|
<>
|
||||||
<>
|
<Header>
|
||||||
<Header>
|
<HeaderRow>
|
||||||
<HeaderRow>
|
<HeaderCell stiff>
|
||||||
<HeaderCell stiff>
|
<Button
|
||||||
<Button
|
fullWidth
|
||||||
fullWidth
|
style={{ fontSize: '14px', justifyContent: 'flex-start' }}
|
||||||
style={HEADER_BUTTON_STYLE}
|
endIcon={getSortIcon(analog_sort.state, 'GPIO')}
|
||||||
endIcon={getSortIcon(analog_sort.state, 'GPIO')}
|
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'GPIO' })}
|
||||||
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'GPIO' })}
|
|
||||||
>
|
|
||||||
GPIO
|
|
||||||
</Button>
|
|
||||||
</HeaderCell>
|
|
||||||
<HeaderCell resize>
|
|
||||||
<Button
|
|
||||||
fullWidth
|
|
||||||
style={HEADER_BUTTON_STYLE}
|
|
||||||
endIcon={getSortIcon(analog_sort.state, 'NAME')}
|
|
||||||
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'NAME' })}
|
|
||||||
>
|
|
||||||
{LL.NAME(0)}
|
|
||||||
</Button>
|
|
||||||
</HeaderCell>
|
|
||||||
<HeaderCell stiff>
|
|
||||||
<Button
|
|
||||||
fullWidth
|
|
||||||
style={HEADER_BUTTON_STYLE}
|
|
||||||
endIcon={getSortIcon(analog_sort.state, 'TYPE')}
|
|
||||||
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'TYPE' })}
|
|
||||||
>
|
|
||||||
{LL.TYPE(0)}
|
|
||||||
</Button>
|
|
||||||
</HeaderCell>
|
|
||||||
<HeaderCell stiff>
|
|
||||||
<Button
|
|
||||||
fullWidth
|
|
||||||
style={HEADER_BUTTON_STYLE_END}
|
|
||||||
endIcon={getSortIcon(analog_sort.state, 'VALUE')}
|
|
||||||
onClick={() =>
|
|
||||||
analog_sort.fns.onToggleSort({ sortKey: 'VALUE' })
|
|
||||||
}
|
|
||||||
>
|
|
||||||
{LL.VALUE(0)}
|
|
||||||
</Button>
|
|
||||||
</HeaderCell>
|
|
||||||
</HeaderRow>
|
|
||||||
</Header>
|
|
||||||
<Body>
|
|
||||||
{tableList.map((as: AnalogSensor) => (
|
|
||||||
<Row
|
|
||||||
style={{ color: as.s ? 'grey' : 'inherit' }}
|
|
||||||
key={as.id}
|
|
||||||
item={as}
|
|
||||||
onClick={() => updateAnalogSensor(as)}
|
|
||||||
>
|
>
|
||||||
<Cell stiff>{as.g}</Cell>
|
GPIO
|
||||||
<Cell>{as.n}</Cell>
|
</Button>
|
||||||
<Cell stiff>{AnalogTypeNames[as.t - 1]} </Cell>
|
</HeaderCell>
|
||||||
{(as.t === AnalogType.DIGITAL_OUT &&
|
<HeaderCell resize>
|
||||||
as.g !== GPIO_25 &&
|
<Button
|
||||||
as.g !== GPIO_26) ||
|
fullWidth
|
||||||
as.t === AnalogType.DIGITAL_IN ||
|
style={{ fontSize: '14px', justifyContent: 'flex-start' }}
|
||||||
as.t === AnalogType.PULSE ? (
|
endIcon={getSortIcon(analog_sort.state, 'NAME')}
|
||||||
<Cell stiff>{as.v ? LL.ON() : LL.OFF()}</Cell>
|
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'NAME' })}
|
||||||
) : (
|
|
||||||
<Cell stiff>{formatValue(as.v, as.u)}</Cell>
|
|
||||||
)}
|
|
||||||
</Row>
|
|
||||||
))}
|
|
||||||
</Body>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Table>
|
|
||||||
),
|
|
||||||
[
|
|
||||||
analog_sort,
|
|
||||||
analog_theme,
|
|
||||||
getSortIcon,
|
|
||||||
sensorData.as,
|
|
||||||
LL,
|
|
||||||
updateAnalogSensor,
|
|
||||||
formatValue
|
|
||||||
]
|
|
||||||
);
|
|
||||||
|
|
||||||
const RenderTemperatureSensors = useMemo(
|
|
||||||
() => (
|
|
||||||
<Table
|
|
||||||
data={{ nodes: sensorData.ts }}
|
|
||||||
theme={temperature_theme}
|
|
||||||
sort={temperature_sort}
|
|
||||||
layout={{ custom: true }}
|
|
||||||
>
|
|
||||||
{(tableList: TemperatureSensor[]) => (
|
|
||||||
<>
|
|
||||||
<Header>
|
|
||||||
<HeaderRow>
|
|
||||||
<HeaderCell resize>
|
|
||||||
<Button
|
|
||||||
fullWidth
|
|
||||||
style={HEADER_BUTTON_STYLE}
|
|
||||||
endIcon={getSortIcon(temperature_sort.state, 'NAME')}
|
|
||||||
onClick={() =>
|
|
||||||
temperature_sort.fns.onToggleSort({ sortKey: 'NAME' })
|
|
||||||
}
|
|
||||||
>
|
|
||||||
{LL.NAME(0)}
|
|
||||||
</Button>
|
|
||||||
</HeaderCell>
|
|
||||||
<HeaderCell stiff>
|
|
||||||
<Button
|
|
||||||
fullWidth
|
|
||||||
style={HEADER_BUTTON_STYLE_END}
|
|
||||||
endIcon={getSortIcon(temperature_sort.state, 'VALUE')}
|
|
||||||
onClick={() =>
|
|
||||||
temperature_sort.fns.onToggleSort({ sortKey: 'VALUE' })
|
|
||||||
}
|
|
||||||
>
|
|
||||||
{LL.VALUE(0)}
|
|
||||||
</Button>
|
|
||||||
</HeaderCell>
|
|
||||||
</HeaderRow>
|
|
||||||
</Header>
|
|
||||||
<Body>
|
|
||||||
{tableList.map((ts: TemperatureSensor) => (
|
|
||||||
<Row
|
|
||||||
style={{ color: ts.s ? 'grey' : 'inherit' }}
|
|
||||||
key={ts.id}
|
|
||||||
item={ts}
|
|
||||||
onClick={() => updateTemperatureSensor(ts)}
|
|
||||||
>
|
>
|
||||||
<Cell>{ts.n}</Cell>
|
{LL.NAME(0)}
|
||||||
<Cell>{formatValue(ts.t, ts.u)}</Cell>
|
</Button>
|
||||||
</Row>
|
</HeaderCell>
|
||||||
))}
|
<HeaderCell stiff>
|
||||||
</Body>
|
<Button
|
||||||
</>
|
fullWidth
|
||||||
)}
|
style={{ fontSize: '14px', justifyContent: 'flex-start' }}
|
||||||
</Table>
|
endIcon={getSortIcon(analog_sort.state, 'TYPE')}
|
||||||
),
|
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'TYPE' })}
|
||||||
[
|
>
|
||||||
temperature_sort,
|
{LL.TYPE(0)}
|
||||||
temperature_theme,
|
</Button>
|
||||||
getSortIcon,
|
</HeaderCell>
|
||||||
sensorData.ts,
|
<HeaderCell stiff>
|
||||||
LL,
|
<Button
|
||||||
updateTemperatureSensor,
|
fullWidth
|
||||||
formatValue
|
style={{ fontSize: '14px', justifyContent: 'flex-end' }}
|
||||||
]
|
endIcon={getSortIcon(analog_sort.state, 'VALUE')}
|
||||||
|
onClick={() => analog_sort.fns.onToggleSort({ sortKey: 'VALUE' })}
|
||||||
|
>
|
||||||
|
{LL.VALUE(0)}
|
||||||
|
</Button>
|
||||||
|
</HeaderCell>
|
||||||
|
</HeaderRow>
|
||||||
|
</Header>
|
||||||
|
<Body>
|
||||||
|
{tableList.map((a: AnalogSensor) => (
|
||||||
|
<Row key={a.id} item={a} onClick={() => updateAnalogSensor(a)}>
|
||||||
|
<Cell stiff>{a.g}</Cell>
|
||||||
|
<Cell>{a.n}</Cell>
|
||||||
|
<Cell stiff>{AnalogTypeNames[a.t]} </Cell>
|
||||||
|
{(a.t === AnalogType.DIGITAL_OUT && a.g !== 25 && a.g !== 26) ||
|
||||||
|
a.t === AnalogType.DIGITAL_IN ? (
|
||||||
|
<Cell stiff>{a.v ? LL.ON() : LL.OFF()}</Cell>
|
||||||
|
) : (
|
||||||
|
<Cell stiff>{a.t ? formatValue(a.v, a.u) : ''}</Cell>
|
||||||
|
)}
|
||||||
|
</Row>
|
||||||
|
))}
|
||||||
|
</Body>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Table>
|
||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
<Typography sx={{ pb: 1 }} variant="h6" color="primary">
|
<Typography sx={{ pb: 1 }} variant="h6" color="secondary">
|
||||||
{LL.TEMP_SENSORS()}
|
{LL.TEMP_SENSORS()}
|
||||||
</Typography>
|
</Typography>
|
||||||
{RenderTemperatureSensors}
|
<RenderTemperatureSensors />
|
||||||
{selectedTemperatureSensor && (
|
{selectedTemperatureSensor && (
|
||||||
<DashboardSensorsTemperatureDialog
|
<DashboardSensorsTemperatureDialog
|
||||||
open={temperatureDialogOpen}
|
open={temperatureDialogOpen}
|
||||||
@@ -574,10 +470,10 @@ const Sensors = () => {
|
|||||||
)}
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<Typography sx={{ pt: 4, pb: 1 }} variant="h6" color="primary">
|
<Typography sx={{ pt: 4, pb: 1 }} variant="h6" color="secondary">
|
||||||
{LL.ANALOG_SENSORS()}
|
{LL.ANALOG_SENSORS()}
|
||||||
</Typography>
|
</Typography>
|
||||||
{RenderAnalogSensors}
|
<RenderAnalogSensors />
|
||||||
{selectedAnalogSensor && (
|
{selectedAnalogSensor && (
|
||||||
<DashboardSensorsAnalogDialog
|
<DashboardSensorsAnalogDialog
|
||||||
open={analogDialogOpen}
|
open={analogDialogOpen}
|
||||||
@@ -585,13 +481,16 @@ const Sensors = () => {
|
|||||||
onSave={onAnalogDialogSave}
|
onSave={onAnalogDialogSave}
|
||||||
creating={creating}
|
creating={creating}
|
||||||
selectedItem={selectedAnalogSensor}
|
selectedItem={selectedAnalogSensor}
|
||||||
analogGPIOList={sensorData.available_gpios}
|
validator={analogSensorItemValidation(
|
||||||
disabledTypeList={sensorData.exclude_types}
|
sensorData.as,
|
||||||
validator={analogSensorItemValidation(sensorData.as, selectedAnalogSensor)}
|
selectedAnalogSensor,
|
||||||
|
creating,
|
||||||
|
sensorData.platform
|
||||||
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{sensorData?.analog_enabled === true && me.admin && (
|
{sensorData?.analog_enabled === true && me.admin && (
|
||||||
<Box mt={2} display="flex" flexWrap="wrap" justifyContent="flex-end">
|
<Box mt={1} display="flex" flexWrap="wrap" justifyContent="flex-end">
|
||||||
<Button
|
<Button
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="primary"
|
color="primary"
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import { useCallback, useEffect, useMemo, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import DoneIcon from '@mui/icons-material/Done';
|
|
||||||
import RemoveIcon from '@mui/icons-material/RemoveCircleOutline';
|
import RemoveIcon from '@mui/icons-material/RemoveCircleOutline';
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
import {
|
import {
|
||||||
@@ -11,9 +10,10 @@ import {
|
|||||||
DialogActions,
|
DialogActions,
|
||||||
DialogContent,
|
DialogContent,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
Grid,
|
Grid2 as Grid,
|
||||||
InputAdornment,
|
InputAdornment,
|
||||||
MenuItem,
|
MenuItem,
|
||||||
|
TextField,
|
||||||
Typography
|
Typography
|
||||||
} from '@mui/material';
|
} from '@mui/material';
|
||||||
|
|
||||||
@@ -34,8 +34,6 @@ interface DashboardSensorsAnalogDialogProps {
|
|||||||
onSave: (as: AnalogSensor) => void;
|
onSave: (as: AnalogSensor) => void;
|
||||||
creating: boolean;
|
creating: boolean;
|
||||||
selectedItem: AnalogSensor;
|
selectedItem: AnalogSensor;
|
||||||
analogGPIOList: number[];
|
|
||||||
disabledTypeList: number[];
|
|
||||||
validator: Schema;
|
validator: Schema;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -45,111 +43,13 @@ const SensorsAnalogDialog = ({
|
|||||||
onSave,
|
onSave,
|
||||||
creating,
|
creating,
|
||||||
selectedItem,
|
selectedItem,
|
||||||
analogGPIOList,
|
|
||||||
disabledTypeList,
|
|
||||||
validator
|
validator
|
||||||
}: DashboardSensorsAnalogDialogProps) => {
|
}: DashboardSensorsAnalogDialogProps) => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
const [editItem, setEditItem] = useState<AnalogSensor>(selectedItem);
|
const [editItem, setEditItem] = useState<AnalogSensor>(selectedItem);
|
||||||
|
const updateFormValue = updateValue(setEditItem);
|
||||||
|
|
||||||
const updateFormValue = useMemo(
|
|
||||||
() =>
|
|
||||||
updateValue((updater) =>
|
|
||||||
setEditItem(
|
|
||||||
(prev) =>
|
|
||||||
updater(
|
|
||||||
prev as unknown as Record<string, unknown>
|
|
||||||
) as unknown as AnalogSensor
|
|
||||||
)
|
|
||||||
),
|
|
||||||
[setEditItem]
|
|
||||||
);
|
|
||||||
|
|
||||||
// Memoize helper functions to check sensor type conditions
|
|
||||||
const isCounterOrRate = useMemo(
|
|
||||||
() =>
|
|
||||||
editItem.t === AnalogType.COUNTER ||
|
|
||||||
editItem.t === AnalogType.RATE ||
|
|
||||||
(editItem.t >= AnalogType.CNT_0 && editItem.t <= AnalogType.CNT_2),
|
|
||||||
[editItem.t]
|
|
||||||
);
|
|
||||||
const isCounter = useMemo(
|
|
||||||
() =>
|
|
||||||
editItem.t === AnalogType.COUNTER ||
|
|
||||||
(editItem.t >= AnalogType.CNT_0 && editItem.t <= AnalogType.CNT_2),
|
|
||||||
[editItem.t]
|
|
||||||
);
|
|
||||||
const isFreqType = useMemo(
|
|
||||||
() => editItem.t >= AnalogType.FREQ_0 && editItem.t <= AnalogType.FREQ_2,
|
|
||||||
[editItem.t]
|
|
||||||
);
|
|
||||||
const isPWM = useMemo(
|
|
||||||
() =>
|
|
||||||
editItem.t === AnalogType.PWM_0 ||
|
|
||||||
editItem.t === AnalogType.PWM_1 ||
|
|
||||||
editItem.t === AnalogType.PWM_2,
|
|
||||||
[editItem.t]
|
|
||||||
);
|
|
||||||
const isDACOutGPIO = useMemo(
|
|
||||||
() =>
|
|
||||||
editItem.t === AnalogType.DIGITAL_OUT &&
|
|
||||||
(editItem.g === 25 || editItem.g === 26),
|
|
||||||
[editItem.t, editItem.g]
|
|
||||||
);
|
|
||||||
const isDigitalOutGPIO = useMemo(
|
|
||||||
() =>
|
|
||||||
editItem.t === AnalogType.DIGITAL_OUT &&
|
|
||||||
editItem.g !== 25 &&
|
|
||||||
editItem.g !== 26,
|
|
||||||
[editItem.t, editItem.g]
|
|
||||||
);
|
|
||||||
|
|
||||||
// Memoize menu items to avoid recreation on each render
|
|
||||||
const analogTypeMenuItems = useMemo(
|
|
||||||
() =>
|
|
||||||
AnalogTypeNames.map((val, i) => ({ name: val, value: i + 1 }))
|
|
||||||
.sort((a, b) => a.name.localeCompare(b.name))
|
|
||||||
.map(({ name, value }) => (
|
|
||||||
<MenuItem
|
|
||||||
key={name}
|
|
||||||
value={value}
|
|
||||||
disabled={disabledTypeList?.includes(value)}
|
|
||||||
>
|
|
||||||
{name}
|
|
||||||
</MenuItem>
|
|
||||||
)),
|
|
||||||
[disabledTypeList]
|
|
||||||
);
|
|
||||||
|
|
||||||
const uomMenuItems = useMemo(
|
|
||||||
() =>
|
|
||||||
DeviceValueUOM_s.map((val, i) => (
|
|
||||||
<MenuItem key={val} value={i}>
|
|
||||||
{val}
|
|
||||||
</MenuItem>
|
|
||||||
)),
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
const analogGPIOMenuItems = () =>
|
|
||||||
// add selectedItem.g to the list
|
|
||||||
[
|
|
||||||
...(analogGPIOList?.includes(selectedItem.g) || selectedItem.g === undefined
|
|
||||||
? analogGPIOList
|
|
||||||
: [selectedItem.g, ...analogGPIOList])
|
|
||||||
]
|
|
||||||
.filter((gpio, idx, arr) => arr.indexOf(gpio) === idx)
|
|
||||||
.sort((a, b) => a - b)
|
|
||||||
.map((gpio: number) => {
|
|
||||||
return (
|
|
||||||
<MenuItem key={gpio} value={gpio}>
|
|
||||||
{gpio}
|
|
||||||
</MenuItem>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
// Reset form when dialog opens or selectedItem changes
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
@@ -157,16 +57,13 @@ const SensorsAnalogDialog = ({
|
|||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const handleClose = useCallback(
|
const handleClose = (_event, reason: 'backdropClick' | 'escapeKeyDown') => {
|
||||||
(_event: React.SyntheticEvent, reason: 'backdropClick' | 'escapeKeyDown') => {
|
if (reason !== 'backdropClick') {
|
||||||
if (reason !== 'backdropClick') {
|
onClose();
|
||||||
onClose();
|
}
|
||||||
}
|
};
|
||||||
},
|
|
||||||
[onClose]
|
|
||||||
);
|
|
||||||
|
|
||||||
const save = useCallback(async () => {
|
const save = async () => {
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(validator, editItem);
|
await validate(validator, editItem);
|
||||||
@@ -174,84 +71,95 @@ const SensorsAnalogDialog = ({
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
}, [validator, editItem, onSave]);
|
};
|
||||||
|
|
||||||
const remove = useCallback(() => {
|
const remove = () => {
|
||||||
onSave({ ...editItem, d: true });
|
editItem.d = true;
|
||||||
}, [editItem, onSave]);
|
onSave(editItem);
|
||||||
|
};
|
||||||
const dialogTitle = useMemo(
|
|
||||||
() =>
|
|
||||||
`${creating ? LL.ADD(1) + ' ' + LL.NEW(0) : LL.EDIT()} ${LL.ANALOG_SENSOR(0)}`,
|
|
||||||
[creating, LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
||||||
<DialogTitle>{dialogTitle}</DialogTitle>
|
<DialogTitle>
|
||||||
|
{creating ? LL.ADD(1) + ' ' + LL.NEW(0) : LL.EDIT()}
|
||||||
|
{LL.ANALOG_SENSOR(0)}
|
||||||
|
</DialogTitle>
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<Grid container spacing={2}>
|
<Grid container spacing={2}>
|
||||||
<ValidatedTextField
|
|
||||||
name="g"
|
|
||||||
label="GPIO"
|
|
||||||
value={editItem.g}
|
|
||||||
sx={{ width: '9ch' }}
|
|
||||||
disabled={editItem.s || !creating}
|
|
||||||
select
|
|
||||||
onChange={updateFormValue}
|
|
||||||
>
|
|
||||||
{analogGPIOMenuItems()}
|
|
||||||
</ValidatedTextField>
|
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
|
fieldErrors={fieldErrors}
|
||||||
|
name="g"
|
||||||
|
label="GPIO"
|
||||||
|
sx={{ width: '11ch' }}
|
||||||
|
value={numberValue(editItem.g)}
|
||||||
|
type="number"
|
||||||
|
variant="outlined"
|
||||||
|
onChange={updateFormValue}
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
{creating && (
|
||||||
|
<Grid>
|
||||||
|
<Box color="warning.main" mt={2}>
|
||||||
|
<Typography variant="body2">{LL.WARN_GPIO()}</Typography>
|
||||||
|
</Box>
|
||||||
|
</Grid>
|
||||||
|
)}
|
||||||
|
<Grid>
|
||||||
|
<ValidatedTextField
|
||||||
|
fieldErrors={fieldErrors}
|
||||||
name="n"
|
name="n"
|
||||||
label={LL.NAME(0)}
|
label={LL.NAME(0)}
|
||||||
value={editItem.n}
|
value={editItem.n}
|
||||||
fullWidth
|
fullWidth
|
||||||
|
variant="outlined"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<TextField
|
||||||
name="t"
|
name="t"
|
||||||
label={LL.TYPE(0)}
|
label={LL.TYPE(0)}
|
||||||
value={editItem.t}
|
value={editItem.t}
|
||||||
fullWidth
|
fullWidth
|
||||||
select
|
select
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
disabled={editItem.s}
|
|
||||||
>
|
>
|
||||||
{analogTypeMenuItems}
|
{AnalogTypeNames.map((val, i) => (
|
||||||
</ValidatedTextField>
|
<MenuItem key={val} value={i}>
|
||||||
|
{val}
|
||||||
|
</MenuItem>
|
||||||
|
))}
|
||||||
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
{(isCounterOrRate ||
|
{editItem.t >= AnalogType.COUNTER && editItem.t <= AnalogType.RATE && (
|
||||||
isFreqType ||
|
|
||||||
editItem.t === AnalogType.ADC ||
|
|
||||||
editItem.t === AnalogType.TIMER) && (
|
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<TextField
|
||||||
name="u"
|
name="u"
|
||||||
label={LL.UNIT()}
|
label={LL.UNIT()}
|
||||||
value={editItem.u}
|
value={editItem.u}
|
||||||
sx={{ width: '15ch' }}
|
sx={{ width: '15ch' }}
|
||||||
select
|
select
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
disabled={editItem.s}
|
|
||||||
>
|
>
|
||||||
{uomMenuItems}
|
{DeviceValueUOM_s.map((val, i) => (
|
||||||
</ValidatedTextField>
|
<MenuItem key={val} value={i}>
|
||||||
|
{val}
|
||||||
|
</MenuItem>
|
||||||
|
))}
|
||||||
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
{editItem.t === AnalogType.ADC && (
|
{editItem.t === AnalogType.ADC && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<TextField
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.OFFSET()}
|
label={LL.OFFSET()}
|
||||||
value={numberValue(editItem.o)}
|
value={numberValue(editItem.o)}
|
||||||
type="number"
|
type="number"
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
|
variant="outlined"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
disabled={editItem.s}
|
|
||||||
slotProps={{
|
slotProps={{
|
||||||
input: {
|
input: {
|
||||||
startAdornment: (
|
startAdornment: (
|
||||||
@@ -263,152 +171,118 @@ const SensorsAnalogDialog = ({
|
|||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
{editItem.t === AnalogType.NTC && (
|
{editItem.t === AnalogType.COUNTER && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<TextField
|
||||||
name="o"
|
|
||||||
label={LL.OFFSET()}
|
|
||||||
value={numberValue(editItem.o)}
|
|
||||||
sx={{ width: '11ch' }}
|
|
||||||
type="number"
|
|
||||||
onChange={updateFormValue}
|
|
||||||
disabled={editItem.s}
|
|
||||||
slotProps={{
|
|
||||||
input: {
|
|
||||||
startAdornment: (
|
|
||||||
<InputAdornment position="start">°C</InputAdornment>
|
|
||||||
)
|
|
||||||
},
|
|
||||||
htmlInput: { min: '-20', max: '20', step: '0.1' }
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
)}
|
|
||||||
{isCounter && (
|
|
||||||
<Grid>
|
|
||||||
<ValidatedTextField
|
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.STARTVALUE()}
|
label={LL.STARTVALUE()}
|
||||||
value={numberValue(editItem.o)}
|
value={numberValue(editItem.o)}
|
||||||
type="number"
|
type="number"
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
|
variant="outlined"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
disabled={editItem.s}
|
|
||||||
slotProps={{
|
slotProps={{
|
||||||
htmlInput: { step: '0.001' }
|
htmlInput: { step: '0.001' }
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
{editItem.t === AnalogType.RGB && (
|
{editItem.t >= AnalogType.COUNTER && editItem.t <= AnalogType.RATE && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<TextField
|
||||||
name="o"
|
|
||||||
label={'RGB ' + LL.VALUE(0)}
|
|
||||||
value={numberValue(editItem.o)}
|
|
||||||
type="number"
|
|
||||||
sx={{ width: '11ch' }}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
disabled={editItem.s}
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
)}
|
|
||||||
{(isCounterOrRate ||
|
|
||||||
isFreqType ||
|
|
||||||
editItem.t === AnalogType.ADC ||
|
|
||||||
editItem.t === AnalogType.TIMER) && (
|
|
||||||
<Grid>
|
|
||||||
<ValidatedTextField
|
|
||||||
name="f"
|
name="f"
|
||||||
label={LL.FACTOR()}
|
label={LL.FACTOR()}
|
||||||
value={numberValue(editItem.f)}
|
value={numberValue(editItem.f)}
|
||||||
sx={{ width: '14ch' }}
|
sx={{ width: '11ch' }}
|
||||||
type="number"
|
type="number"
|
||||||
|
variant="outlined"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
disabled={editItem.s}
|
|
||||||
slotProps={{
|
slotProps={{
|
||||||
htmlInput: { step: '0.001' }
|
htmlInput: { step: '0.001' }
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
{isDACOutGPIO && (
|
{editItem.t === AnalogType.DIGITAL_OUT &&
|
||||||
<Grid>
|
(editItem.g === 25 || editItem.g === 26) && (
|
||||||
<ValidatedTextField
|
|
||||||
name="o"
|
|
||||||
label={LL.VALUE(0)}
|
|
||||||
value={numberValue(editItem.o)}
|
|
||||||
sx={{ width: '11ch' }}
|
|
||||||
type="number"
|
|
||||||
onChange={updateFormValue}
|
|
||||||
disabled={editItem.s}
|
|
||||||
slotProps={{
|
|
||||||
htmlInput: { min: '0', max: '255', step: '1' }
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
)}
|
|
||||||
{isDigitalOutGPIO && (
|
|
||||||
<>
|
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<TextField
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.VALUE(0)}
|
label={LL.VALUE(0)}
|
||||||
value={numberValue(editItem.o)}
|
value={numberValue(editItem.o)}
|
||||||
select
|
sx={{ width: '11ch' }}
|
||||||
|
type="number"
|
||||||
|
variant="outlined"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
disabled={editItem.s}
|
slotProps={{
|
||||||
>
|
htmlInput: { min: '0', max: '255', step: '1' }
|
||||||
<MenuItem value={0}>{LL.OFF()}</MenuItem>
|
}}
|
||||||
<MenuItem value={1}>{LL.ON()}</MenuItem>
|
/>
|
||||||
</ValidatedTextField>
|
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
)}
|
||||||
<ValidatedTextField
|
{editItem.t === AnalogType.DIGITAL_OUT &&
|
||||||
name="f"
|
editItem.g !== 25 &&
|
||||||
label={LL.POLARITY()}
|
editItem.g !== 26 && (
|
||||||
value={editItem.f}
|
<>
|
||||||
sx={{ width: '15ch' }}
|
<Grid>
|
||||||
select
|
<TextField
|
||||||
onChange={updateFormValue}
|
name="o"
|
||||||
disabled={editItem.s}
|
label={LL.VALUE(0)}
|
||||||
>
|
value={numberValue(editItem.o)}
|
||||||
<MenuItem value={1}>{LL.ACTIVEHIGH()}</MenuItem>
|
select
|
||||||
<MenuItem value={-1}>{LL.ACTIVELOW()}</MenuItem>
|
variant="outlined"
|
||||||
</ValidatedTextField>
|
onChange={updateFormValue}
|
||||||
</Grid>
|
>
|
||||||
<Grid>
|
<MenuItem value={0}>{LL.OFF()}</MenuItem>
|
||||||
<ValidatedTextField
|
<MenuItem value={1}>{LL.ON()}</MenuItem>
|
||||||
name="u"
|
</TextField>
|
||||||
label={LL.STARTVALUE()}
|
</Grid>
|
||||||
sx={{ width: '15ch' }}
|
<Grid>
|
||||||
value={editItem.u}
|
<TextField
|
||||||
select
|
name="f"
|
||||||
onChange={updateFormValue}
|
label={LL.POLARITY()}
|
||||||
disabled={editItem.s}
|
value={editItem.f}
|
||||||
>
|
sx={{ width: '15ch' }}
|
||||||
<MenuItem value={0}>{LL.UNCHANGED()}</MenuItem>
|
select
|
||||||
<MenuItem value={1}>
|
onChange={updateFormValue}
|
||||||
{LL.ALWAYS()} {LL.OFF()}
|
>
|
||||||
</MenuItem>
|
<MenuItem value={1}>{LL.ACTIVEHIGH()}</MenuItem>
|
||||||
<MenuItem value={2}>
|
<MenuItem value={-1}>{LL.ACTIVELOW()}</MenuItem>
|
||||||
{LL.ALWAYS()} {LL.ON()}
|
</TextField>
|
||||||
</MenuItem>
|
</Grid>
|
||||||
</ValidatedTextField>
|
<Grid>
|
||||||
</Grid>
|
<TextField
|
||||||
</>
|
name="u"
|
||||||
)}
|
label={LL.STARTVALUE()}
|
||||||
{isPWM && (
|
sx={{ width: '15ch' }}
|
||||||
|
value={editItem.u}
|
||||||
|
select
|
||||||
|
onChange={updateFormValue}
|
||||||
|
>
|
||||||
|
<MenuItem value={0}>{LL.UNCHANGED()}</MenuItem>
|
||||||
|
<MenuItem value={1}>
|
||||||
|
{LL.ALWAYS()} {LL.OFF()}
|
||||||
|
</MenuItem>
|
||||||
|
<MenuItem value={2}>
|
||||||
|
{LL.ALWAYS()} {LL.ON()}
|
||||||
|
</MenuItem>
|
||||||
|
</TextField>
|
||||||
|
</Grid>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{(editItem.t === AnalogType.PWM_0 ||
|
||||||
|
editItem.t === AnalogType.PWM_1 ||
|
||||||
|
editItem.t === AnalogType.PWM_2) && (
|
||||||
<>
|
<>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<TextField
|
||||||
name="f"
|
name="f"
|
||||||
label={LL.FREQ()}
|
label={LL.FREQ()}
|
||||||
value={numberValue(editItem.f)}
|
value={numberValue(editItem.f)}
|
||||||
type="number"
|
type="number"
|
||||||
|
variant="outlined"
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
disabled={editItem.s}
|
|
||||||
slotProps={{
|
slotProps={{
|
||||||
input: {
|
input: {
|
||||||
startAdornment: (
|
startAdornment: (
|
||||||
@@ -420,14 +294,14 @@ const SensorsAnalogDialog = ({
|
|||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<TextField
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.DUTY_CYCLE()}
|
label={LL.DUTY_CYCLE()}
|
||||||
value={numberValue(editItem.o)}
|
value={numberValue(editItem.o)}
|
||||||
type="number"
|
type="number"
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
|
variant="outlined"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
disabled={editItem.s}
|
|
||||||
slotProps={{
|
slotProps={{
|
||||||
input: {
|
input: {
|
||||||
startAdornment: (
|
startAdornment: (
|
||||||
@@ -440,81 +314,13 @@ const SensorsAnalogDialog = ({
|
|||||||
</Grid>
|
</Grid>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
{editItem.t === AnalogType.PULSE && (
|
|
||||||
<>
|
|
||||||
<Grid>
|
|
||||||
<ValidatedTextField
|
|
||||||
name="o"
|
|
||||||
label={LL.POLARITY()}
|
|
||||||
value={editItem.o}
|
|
||||||
sx={{ width: '11ch' }}
|
|
||||||
select
|
|
||||||
onChange={updateFormValue}
|
|
||||||
disabled={editItem.s}
|
|
||||||
>
|
|
||||||
<MenuItem value={0}>{LL.ACTIVEHIGH()}</MenuItem>
|
|
||||||
<MenuItem value={1}>{LL.ACTIVELOW()}</MenuItem>
|
|
||||||
</ValidatedTextField>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
|
||||||
<ValidatedTextField
|
|
||||||
name="f"
|
|
||||||
label="Pulse"
|
|
||||||
value={numberValue(editItem.f)}
|
|
||||||
type="number"
|
|
||||||
sx={{ width: '15ch' }}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
disabled={editItem.s}
|
|
||||||
slotProps={{
|
|
||||||
input: {
|
|
||||||
startAdornment: (
|
|
||||||
<InputAdornment position="start">s</InputAdornment>
|
|
||||||
)
|
|
||||||
},
|
|
||||||
htmlInput: { min: '0', max: '10000', step: '0.1' }
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Grid>
|
</Grid>
|
||||||
{fieldErrors && Object.keys(fieldErrors).length > 0 && (
|
|
||||||
<Box mt={1}>
|
|
||||||
{Object.values(fieldErrors).map((errArr, idx) =>
|
|
||||||
Array.isArray(errArr)
|
|
||||||
? errArr.map((err, j) => (
|
|
||||||
<Typography
|
|
||||||
key={`${idx}-${j}`}
|
|
||||||
color="error"
|
|
||||||
variant="caption"
|
|
||||||
display="block"
|
|
||||||
>
|
|
||||||
{err.message}
|
|
||||||
</Typography>
|
|
||||||
))
|
|
||||||
: null
|
|
||||||
)}
|
|
||||||
</Box>
|
|
||||||
)}
|
|
||||||
{editItem.s && (
|
|
||||||
<Grid>
|
|
||||||
<Typography mt={1} color="warning.main" variant="body2">
|
|
||||||
<WarningIcon
|
|
||||||
fontSize="small"
|
|
||||||
sx={{ mr: 1, verticalAlign: 'middle' }}
|
|
||||||
color="warning"
|
|
||||||
/>
|
|
||||||
{LL.SYSTEM(0)} {LL.SENSOR(0)}
|
|
||||||
</Typography>
|
|
||||||
</Grid>
|
|
||||||
)}
|
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
<DialogActions>
|
<DialogActions>
|
||||||
{!creating && (
|
{!creating && (
|
||||||
<Box flexGrow={1} sx={{ '& button': { mt: 0 } }}>
|
<Box flexGrow={1} sx={{ '& button': { mt: 0 } }}>
|
||||||
<Button
|
<Button
|
||||||
startIcon={<RemoveIcon />}
|
startIcon={<RemoveIcon />}
|
||||||
disabled={editItem.s}
|
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="warning"
|
color="warning"
|
||||||
onClick={remove}
|
onClick={remove}
|
||||||
@@ -532,7 +338,7 @@ const SensorsAnalogDialog = ({
|
|||||||
{LL.CANCEL()}
|
{LL.CANCEL()}
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
startIcon={<DoneIcon />}
|
startIcon={<WarningIcon color="warning" />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={save}
|
onClick={save}
|
||||||
color="primary"
|
color="primary"
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import { useCallback, useEffect, useMemo, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import DoneIcon from '@mui/icons-material/Done';
|
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
import {
|
import {
|
||||||
Box,
|
Box,
|
||||||
@@ -10,7 +9,7 @@ import {
|
|||||||
DialogActions,
|
DialogActions,
|
||||||
DialogContent,
|
DialogContent,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
Grid,
|
Grid2 as Grid,
|
||||||
InputAdornment,
|
InputAdornment,
|
||||||
TextField,
|
TextField,
|
||||||
Typography
|
Typography
|
||||||
@@ -34,12 +33,6 @@ interface SensorsTemperatureDialogProps {
|
|||||||
validator: Schema;
|
validator: Schema;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Constants
|
|
||||||
const OFFSET_MIN = -5;
|
|
||||||
const OFFSET_MAX = 5;
|
|
||||||
const OFFSET_STEP = 0.1;
|
|
||||||
const TEMP_UNIT = '°C';
|
|
||||||
|
|
||||||
const SensorsTemperatureDialog = ({
|
const SensorsTemperatureDialog = ({
|
||||||
open,
|
open,
|
||||||
onClose,
|
onClose,
|
||||||
@@ -50,18 +43,7 @@ const SensorsTemperatureDialog = ({
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
const [editItem, setEditItem] = useState<TemperatureSensor>(selectedItem);
|
const [editItem, setEditItem] = useState<TemperatureSensor>(selectedItem);
|
||||||
|
const updateFormValue = updateValue(setEditItem);
|
||||||
const updateFormValue = useMemo(
|
|
||||||
() =>
|
|
||||||
updateValue(
|
|
||||||
setEditItem as unknown as (
|
|
||||||
updater: (
|
|
||||||
prevState: Readonly<Record<string, unknown>>
|
|
||||||
) => Record<string, unknown>
|
|
||||||
) => void
|
|
||||||
),
|
|
||||||
[setEditItem]
|
|
||||||
);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (open) {
|
if (open) {
|
||||||
@@ -70,16 +52,13 @@ const SensorsTemperatureDialog = ({
|
|||||||
}
|
}
|
||||||
}, [open, selectedItem]);
|
}, [open, selectedItem]);
|
||||||
|
|
||||||
const handleClose = useCallback(
|
const handleClose = (_event, reason: 'backdropClick' | 'escapeKeyDown') => {
|
||||||
(_event: React.SyntheticEvent, reason?: string) => {
|
if (reason !== 'backdropClick') {
|
||||||
if (reason !== 'backdropClick') {
|
onClose();
|
||||||
onClose();
|
}
|
||||||
}
|
};
|
||||||
},
|
|
||||||
[onClose]
|
|
||||||
);
|
|
||||||
|
|
||||||
const save = useCallback(async () => {
|
const save = async () => {
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
await validate(validator, editItem);
|
await validate(validator, editItem);
|
||||||
@@ -87,31 +66,15 @@ const SensorsTemperatureDialog = ({
|
|||||||
} catch (error) {
|
} catch (error) {
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
}, [validator, editItem, onSave]);
|
};
|
||||||
|
|
||||||
const dialogTitle = useMemo(() => `${LL.EDIT()} ${LL.TEMP_SENSOR()}`, [LL]);
|
|
||||||
|
|
||||||
const offsetValue = useMemo(() => numberValue(editItem.o), [editItem.o]);
|
|
||||||
|
|
||||||
const slotProps = useMemo(
|
|
||||||
() => ({
|
|
||||||
input: {
|
|
||||||
startAdornment: <InputAdornment position="start">{TEMP_UNIT}</InputAdornment>
|
|
||||||
},
|
|
||||||
htmlInput: {
|
|
||||||
min: OFFSET_MIN,
|
|
||||||
max: OFFSET_MAX,
|
|
||||||
step: OFFSET_STEP
|
|
||||||
}
|
|
||||||
}),
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
<Dialog sx={dialogStyle} open={open} onClose={handleClose}>
|
||||||
<DialogTitle>{dialogTitle}</DialogTitle>
|
<DialogTitle>
|
||||||
|
{LL.EDIT()} {LL.TEMP_SENSOR()}
|
||||||
|
</DialogTitle>
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<Box color="warning.main" mb={2}>
|
<Box color="warning.main" p={0} pl={0} pr={0} mt={0} mb={2}>
|
||||||
<Typography variant="body2">
|
<Typography variant="body2">
|
||||||
{LL.ID_OF(LL.SENSOR(0))}: {editItem.id}
|
{LL.ID_OF(LL.SENSOR(0))}: {editItem.id}
|
||||||
</Typography>
|
</Typography>
|
||||||
@@ -119,7 +82,7 @@ const SensorsTemperatureDialog = ({
|
|||||||
<Grid container spacing={2}>
|
<Grid container spacing={2}>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors ?? {}}
|
fieldErrors={fieldErrors}
|
||||||
name="n"
|
name="n"
|
||||||
label={LL.NAME(0)}
|
label={LL.NAME(0)}
|
||||||
value={editItem.n}
|
value={editItem.n}
|
||||||
@@ -131,27 +94,22 @@ const SensorsTemperatureDialog = ({
|
|||||||
<TextField
|
<TextField
|
||||||
name="o"
|
name="o"
|
||||||
label={LL.OFFSET()}
|
label={LL.OFFSET()}
|
||||||
value={offsetValue}
|
value={numberValue(editItem.o)}
|
||||||
sx={{ width: '11ch' }}
|
sx={{ width: '11ch' }}
|
||||||
type="number"
|
type="number"
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
slotProps={slotProps}
|
slotProps={{
|
||||||
|
input: {
|
||||||
|
startAdornment: (
|
||||||
|
<InputAdornment position="start">°C</InputAdornment>
|
||||||
|
)
|
||||||
|
},
|
||||||
|
htmlInput: { min: '-5', max: '5', step: '0.1' }
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
{editItem.s && (
|
|
||||||
<Grid>
|
|
||||||
<Typography mt={1} color="warning.main" variant="body2">
|
|
||||||
<WarningIcon
|
|
||||||
fontSize="small"
|
|
||||||
sx={{ mr: 1, verticalAlign: 'middle' }}
|
|
||||||
color="warning"
|
|
||||||
/>
|
|
||||||
{LL.SYSTEM(0)} {LL.SENSOR(0)}
|
|
||||||
</Typography>
|
|
||||||
</Grid>
|
|
||||||
)}
|
|
||||||
</DialogContent>
|
</DialogContent>
|
||||||
<DialogActions>
|
<DialogActions>
|
||||||
<Button
|
<Button
|
||||||
@@ -163,7 +121,7 @@ const SensorsTemperatureDialog = ({
|
|||||||
{LL.CANCEL()}
|
{LL.CANCEL()}
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
startIcon={<DoneIcon />}
|
startIcon={<WarningIcon color="warning" />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={save}
|
onClick={save}
|
||||||
color="primary"
|
color="primary"
|
||||||
|
|||||||
@@ -1,65 +0,0 @@
|
|||||||
import { memo, useCallback, useContext } from 'react';
|
|
||||||
|
|
||||||
import PersonIcon from '@mui/icons-material/Person';
|
|
||||||
import {
|
|
||||||
Avatar,
|
|
||||||
Box,
|
|
||||||
Button,
|
|
||||||
Divider,
|
|
||||||
List,
|
|
||||||
ListItem,
|
|
||||||
ListItemText,
|
|
||||||
Typography
|
|
||||||
} from '@mui/material';
|
|
||||||
|
|
||||||
import { AuthenticatedContext } from '@/contexts/authentication';
|
|
||||||
import { SectionContent, useLayoutTitle } from 'components';
|
|
||||||
import { LanguageSelector } from 'components/inputs';
|
|
||||||
import { useI18nContext } from 'i18n/i18n-react';
|
|
||||||
|
|
||||||
const UserProfileComponent = () => {
|
|
||||||
const { LL } = useI18nContext();
|
|
||||||
const { me, signOut } = useContext(AuthenticatedContext);
|
|
||||||
|
|
||||||
useLayoutTitle(LL.USER_PROFILE());
|
|
||||||
|
|
||||||
const handleSignOut = useCallback(() => {
|
|
||||||
signOut(true);
|
|
||||||
}, [signOut]);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<SectionContent>
|
|
||||||
<List sx={{ flexGrow: 1 }}>
|
|
||||||
<ListItem disablePadding>
|
|
||||||
<Avatar sx={{ bgcolor: '#9e9e9e', color: 'white' }}>
|
|
||||||
<PersonIcon />
|
|
||||||
</Avatar>
|
|
||||||
<ListItemText
|
|
||||||
sx={{ pl: 2, color: '#2196f3' }}
|
|
||||||
primary={me.username}
|
|
||||||
secondary={'(' + (me.admin ? LL.ADMINISTRATOR() : LL.GUEST()) + ')'}
|
|
||||||
/>
|
|
||||||
</ListItem>
|
|
||||||
</List>
|
|
||||||
<Box mt={2} mb={2} display="flex" alignItems="center">
|
|
||||||
<Typography mr={2} variant="body1" align="center">
|
|
||||||
{LL.LANGUAGE()}:
|
|
||||||
</Typography>
|
|
||||||
<LanguageSelector />
|
|
||||||
</Box>
|
|
||||||
<Divider />
|
|
||||||
<Button
|
|
||||||
sx={{ mt: 2 }}
|
|
||||||
variant="outlined"
|
|
||||||
color="primary"
|
|
||||||
onClick={handleSignOut}
|
|
||||||
>
|
|
||||||
{LL.SIGN_OUT()}
|
|
||||||
</Button>
|
|
||||||
</SectionContent>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
const UserProfile = memo(UserProfileComponent);
|
|
||||||
|
|
||||||
export default UserProfile;
|
|
||||||
@@ -2,30 +2,27 @@ import type { TranslationFunctions } from 'i18n/i18n-types';
|
|||||||
|
|
||||||
import { DeviceValueUOM, DeviceValueUOM_s } from './types';
|
import { DeviceValueUOM, DeviceValueUOM_s } from './types';
|
||||||
|
|
||||||
// Cache NumberFormat instances for better performance
|
|
||||||
const numberFormatter = new Intl.NumberFormat();
|
|
||||||
const numberFormatterWithDecimal = new Intl.NumberFormat(undefined, {
|
|
||||||
minimumFractionDigits: 1
|
|
||||||
});
|
|
||||||
|
|
||||||
const formatDurationMin = (LL: TranslationFunctions, duration_min: number) => {
|
const formatDurationMin = (LL: TranslationFunctions, duration_min: number) => {
|
||||||
const totalMs = duration_min * 60000;
|
const days = Math.trunc((duration_min * 60000) / 86400000);
|
||||||
const days = Math.trunc(totalMs / 86400000);
|
const hours = Math.trunc((duration_min * 60000) / 3600000) % 24;
|
||||||
const hours = Math.trunc(totalMs / 3600000) % 24;
|
const minutes = Math.trunc((duration_min * 60000) / 60000) % 60;
|
||||||
const minutes = Math.trunc(duration_min) % 60;
|
|
||||||
|
|
||||||
const parts: string[] = [];
|
let formatted = '';
|
||||||
if (days) {
|
if (days) {
|
||||||
parts.push(LL.NUM_DAYS({ num: days }));
|
formatted += LL.NUM_DAYS({ num: days });
|
||||||
}
|
|
||||||
if (hours) {
|
|
||||||
parts.push(LL.NUM_HOURS({ num: hours }));
|
|
||||||
}
|
|
||||||
if (minutes) {
|
|
||||||
parts.push(LL.NUM_MINUTES({ num: minutes }));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return parts.join(' ');
|
if (hours) {
|
||||||
|
if (formatted) formatted += ' ';
|
||||||
|
formatted += LL.NUM_HOURS({ num: hours });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (minutes) {
|
||||||
|
if (formatted) formatted += ' ';
|
||||||
|
formatted += LL.NUM_MINUTES({ num: minutes });
|
||||||
|
}
|
||||||
|
|
||||||
|
return formatted;
|
||||||
};
|
};
|
||||||
|
|
||||||
export function formatValue(
|
export function formatValue(
|
||||||
@@ -33,21 +30,18 @@ export function formatValue(
|
|||||||
value?: unknown,
|
value?: unknown,
|
||||||
uom?: DeviceValueUOM
|
uom?: DeviceValueUOM
|
||||||
) {
|
) {
|
||||||
// Handle non-numeric values or missing data
|
|
||||||
if (typeof value !== 'number' || uom === undefined || value === undefined) {
|
if (typeof value !== 'number' || uom === undefined || value === undefined) {
|
||||||
if (value === undefined || typeof value === 'boolean') {
|
if (value === undefined || typeof value === 'boolean') {
|
||||||
return '';
|
return '';
|
||||||
}
|
}
|
||||||
// Type assertion is safe here since we know it's not a number, boolean, or undefined
|
|
||||||
return (
|
return (
|
||||||
(value as string) +
|
(value as string) +
|
||||||
(value === '' || uom === undefined || uom === DeviceValueUOM.NONE
|
(value === '' || uom === undefined || uom === 0
|
||||||
? ''
|
? ''
|
||||||
: ' ' + DeviceValueUOM_s[uom])
|
: ' ' + DeviceValueUOM_s[uom])
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handle numeric values
|
|
||||||
switch (uom) {
|
switch (uom) {
|
||||||
case DeviceValueUOM.HOURS:
|
case DeviceValueUOM.HOURS:
|
||||||
return value ? formatDurationMin(LL, value * 60) : LL.NUM_HOURS({ num: 0 });
|
return value ? formatDurationMin(LL, value * 60) : LL.NUM_HOURS({ num: 0 });
|
||||||
@@ -56,12 +50,18 @@ export function formatValue(
|
|||||||
case DeviceValueUOM.SECONDS:
|
case DeviceValueUOM.SECONDS:
|
||||||
return LL.NUM_SECONDS({ num: value });
|
return LL.NUM_SECONDS({ num: value });
|
||||||
case DeviceValueUOM.NONE:
|
case DeviceValueUOM.NONE:
|
||||||
return numberFormatter.format(value);
|
return new Intl.NumberFormat().format(value);
|
||||||
case DeviceValueUOM.DEGREES:
|
case DeviceValueUOM.DEGREES:
|
||||||
case DeviceValueUOM.DEGREES_R:
|
case DeviceValueUOM.DEGREES_R:
|
||||||
case DeviceValueUOM.FAHRENHEIT:
|
case DeviceValueUOM.FAHRENHEIT:
|
||||||
return numberFormatterWithDecimal.format(value) + ' ' + DeviceValueUOM_s[uom];
|
return (
|
||||||
|
new Intl.NumberFormat(undefined, {
|
||||||
|
minimumFractionDigits: 1
|
||||||
|
}).format(value) +
|
||||||
|
' ' +
|
||||||
|
DeviceValueUOM_s[uom]
|
||||||
|
);
|
||||||
default:
|
default:
|
||||||
return numberFormatter.format(value) + ' ' + DeviceValueUOM_s[uom];
|
return new Intl.NumberFormat().format(value) + ' ' + DeviceValueUOM_s[uom];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -43,16 +43,6 @@ export interface Settings {
|
|||||||
modbus_port: number;
|
modbus_port: number;
|
||||||
modbus_max_clients: number;
|
modbus_max_clients: number;
|
||||||
modbus_timeout: number;
|
modbus_timeout: number;
|
||||||
email_enabled: boolean;
|
|
||||||
email_ssl?: boolean;
|
|
||||||
email_starttls?: boolean;
|
|
||||||
email_server: string;
|
|
||||||
email_port: number;
|
|
||||||
email_login: string;
|
|
||||||
email_pass: string;
|
|
||||||
email_sender: string;
|
|
||||||
email_recp: string;
|
|
||||||
email_subject: string;
|
|
||||||
developer_mode: boolean;
|
developer_mode: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -70,7 +60,7 @@ export interface Stat {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface Activity {
|
export interface Activity {
|
||||||
readonly stats: readonly Stat[];
|
stats: Stat[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Device {
|
export interface Device {
|
||||||
@@ -92,43 +82,38 @@ export interface TemperatureSensor {
|
|||||||
t?: number; // temp, optional
|
t?: number; // temp, optional
|
||||||
o: number; // offset
|
o: number; // offset
|
||||||
u: number; // uom
|
u: number; // uom
|
||||||
s: boolean; // system sensor flag
|
|
||||||
o_n?: string;
|
o_n?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface AnalogSensor {
|
export interface AnalogSensor {
|
||||||
id: number;
|
id: number;
|
||||||
g: number; // GPIO
|
g: number; // GPIO
|
||||||
n: string; // name
|
n: string;
|
||||||
v: number; // value
|
v: number;
|
||||||
u: number; // uom
|
u: number;
|
||||||
o: number; // offset
|
o: number;
|
||||||
f: number; // factor
|
f: number;
|
||||||
t: number; // type
|
t: number;
|
||||||
d: boolean; // deleted flag
|
d: boolean; // deleted flag
|
||||||
s: boolean; // system sensor flag
|
o_n?: string;
|
||||||
o_n?: string; // original name
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface WriteTemperatureSensor {
|
export interface WriteTemperatureSensor {
|
||||||
id: string;
|
id: string;
|
||||||
name: string;
|
name: string;
|
||||||
offset: number;
|
offset: number;
|
||||||
is_system: boolean;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface SensorData {
|
export interface SensorData {
|
||||||
ts: TemperatureSensor[];
|
ts: TemperatureSensor[];
|
||||||
as: AnalogSensor[];
|
as: AnalogSensor[];
|
||||||
analog_enabled: boolean;
|
analog_enabled: boolean;
|
||||||
available_gpios: number[];
|
|
||||||
exclude_types: number[];
|
|
||||||
platform: string;
|
platform: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface CoreData {
|
export interface CoreData {
|
||||||
readonly connected: boolean;
|
connected: boolean;
|
||||||
readonly devices: readonly Device[];
|
devices: Device[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DashboardItem {
|
export interface DashboardItem {
|
||||||
@@ -137,12 +122,11 @@ export interface DashboardItem {
|
|||||||
n?: string; // name, optional
|
n?: string; // name, optional
|
||||||
dv?: DeviceValue; // device value, optional
|
dv?: DeviceValue; // device value, optional
|
||||||
nodes?: DashboardItem[]; // children nodes, optional
|
nodes?: DashboardItem[]; // children nodes, optional
|
||||||
parentNode: DashboardItem; // to stop lint errors
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DashboardData {
|
export interface DashboardData {
|
||||||
readonly connected: boolean; // true if connected to EMS bus
|
connected: boolean; // true if connected to EMS bus
|
||||||
readonly nodes: readonly DashboardItem[];
|
nodes: DashboardItem[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DeviceValue {
|
export interface DeviceValue {
|
||||||
@@ -155,11 +139,10 @@ export interface DeviceValue {
|
|||||||
s?: string; // steps for up/down, optional
|
s?: string; // steps for up/down, optional
|
||||||
m?: number; // min, optional
|
m?: number; // min, optional
|
||||||
x?: number; // max, optional
|
x?: number; // max, optional
|
||||||
[key: string]: unknown;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DeviceData {
|
export interface DeviceData {
|
||||||
readonly nodes: readonly DeviceValue[];
|
nodes: DeviceValue[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface DeviceEntity {
|
export interface DeviceEntity {
|
||||||
@@ -205,14 +188,13 @@ export enum DeviceValueUOM {
|
|||||||
VOLTS,
|
VOLTS,
|
||||||
MBAR,
|
MBAR,
|
||||||
LH,
|
LH,
|
||||||
CTKWH,
|
CTKWH
|
||||||
HERTZ
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const DeviceValueUOM_s = [
|
export const DeviceValueUOM_s = [
|
||||||
'',
|
'',
|
||||||
'°C',
|
'°C',
|
||||||
'°C Rel',
|
'°C',
|
||||||
'%',
|
'%',
|
||||||
'l/min',
|
'l/min',
|
||||||
'kWh',
|
'kWh',
|
||||||
@@ -236,12 +218,12 @@ export const DeviceValueUOM_s = [
|
|||||||
'V',
|
'V',
|
||||||
'mbar',
|
'mbar',
|
||||||
'l/h',
|
'l/h',
|
||||||
'ct/kWh',
|
'ct/kWh'
|
||||||
'Hz'
|
];
|
||||||
] as const;
|
|
||||||
|
|
||||||
export enum AnalogType {
|
export enum AnalogType {
|
||||||
REMOVED = -1,
|
REMOVED = -1,
|
||||||
|
NOTUSED = 0,
|
||||||
DIGITAL_IN = 1,
|
DIGITAL_IN = 1,
|
||||||
COUNTER = 2,
|
COUNTER = 2,
|
||||||
ADC = 3,
|
ADC = 3,
|
||||||
@@ -250,45 +232,29 @@ export enum AnalogType {
|
|||||||
DIGITAL_OUT = 6,
|
DIGITAL_OUT = 6,
|
||||||
PWM_0 = 7,
|
PWM_0 = 7,
|
||||||
PWM_1 = 8,
|
PWM_1 = 8,
|
||||||
PWM_2 = 9,
|
PWM_2 = 9
|
||||||
NTC = 10,
|
|
||||||
RGB = 11,
|
|
||||||
PULSE = 12,
|
|
||||||
FREQ_0 = 13,
|
|
||||||
FREQ_1 = 14,
|
|
||||||
FREQ_2 = 15,
|
|
||||||
CNT_0 = 16,
|
|
||||||
CNT_1 = 17,
|
|
||||||
CNT_2 = 18
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export const AnalogTypeNames = [
|
export const AnalogTypeNames = [
|
||||||
'Digital In', // 1
|
'(disabled)',
|
||||||
'Counter', // 2
|
'Digital In',
|
||||||
'ADC In', // 3
|
'Counter',
|
||||||
'Timer', // 4
|
'ADC',
|
||||||
'Rate', // 5
|
'Timer',
|
||||||
'Digital Out', // 6
|
'Rate',
|
||||||
'PWM 0', // 7
|
'Digital Out',
|
||||||
'PWM 1', // 8
|
'PWM 0',
|
||||||
'PWM 2', // 9
|
'PWM 1',
|
||||||
'NTC Temp', // 10
|
'PWM 2'
|
||||||
'RGB Led', // 11
|
];
|
||||||
'Pulse', // 12
|
|
||||||
'Freq 0', // 13
|
|
||||||
'Freq 1', // 14
|
|
||||||
'Freq 2', // 15
|
|
||||||
'Counter 0', // 16
|
|
||||||
'Counter 1', // 17
|
|
||||||
'Counter 2' // 18
|
|
||||||
] as const;
|
|
||||||
|
|
||||||
export const BOARD_PROFILES = {
|
type BoardProfiles = Record<string, string>;
|
||||||
|
|
||||||
|
export const BOARD_PROFILES: BoardProfiles = {
|
||||||
S32: 'BBQKees Gateway S32',
|
S32: 'BBQKees Gateway S32',
|
||||||
S32S3: 'BBQKees Gateway S3',
|
S32S3: 'BBQKees Gateway S3',
|
||||||
E32: 'BBQKees Gateway E32',
|
E32: 'BBQKees Gateway E32',
|
||||||
E32V2: 'BBQKees Gateway E32 V2',
|
E32V2: 'BBQKees Gateway E32 V2',
|
||||||
E32V2_2: 'BBQKees Gateway E32 V2.2',
|
|
||||||
NODEMCU: 'NodeMCU 32S',
|
NODEMCU: 'NodeMCU 32S',
|
||||||
'MH-ET': 'MH-ET Live D1 Mini',
|
'MH-ET': 'MH-ET Live D1 Mini',
|
||||||
LOLIN: 'Lolin D32',
|
LOLIN: 'Lolin D32',
|
||||||
@@ -297,9 +263,7 @@ export const BOARD_PROFILES = {
|
|||||||
C3MINI: 'Wemos C3 Mini',
|
C3MINI: 'Wemos C3 Mini',
|
||||||
S2MINI: 'Wemos S2 Mini',
|
S2MINI: 'Wemos S2 Mini',
|
||||||
S3MINI: 'Liligo S3'
|
S3MINI: 'Liligo S3'
|
||||||
} as const;
|
};
|
||||||
|
|
||||||
export type BoardProfileKey = keyof typeof BOARD_PROFILES;
|
|
||||||
|
|
||||||
export interface BoardProfile {
|
export interface BoardProfile {
|
||||||
board_profile: string;
|
board_profile: string;
|
||||||
@@ -336,7 +300,6 @@ export interface WriteAnalogSensor {
|
|||||||
uom: number;
|
uom: number;
|
||||||
type: number;
|
type: number;
|
||||||
deleted: boolean;
|
deleted: boolean;
|
||||||
is_system: boolean;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum DeviceEntityMask {
|
export enum DeviceEntityMask {
|
||||||
@@ -368,7 +331,7 @@ export interface ScheduleItem {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface Schedule {
|
export interface Schedule {
|
||||||
readonly schedule: readonly ScheduleItem[];
|
schedule: ScheduleItem[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ModuleItem {
|
export interface ModuleItem {
|
||||||
@@ -386,7 +349,7 @@ export interface ModuleItem {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface Modules {
|
export interface Modules {
|
||||||
readonly modules: readonly ModuleItem[];
|
modules: ModuleItem[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export enum ScheduleFlag {
|
export enum ScheduleFlag {
|
||||||
@@ -417,7 +380,6 @@ export interface EntityItem {
|
|||||||
value_type: number;
|
value_type: number;
|
||||||
value?: unknown;
|
value?: unknown;
|
||||||
writeable: boolean;
|
writeable: boolean;
|
||||||
hide: boolean;
|
|
||||||
deleted?: boolean;
|
deleted?: boolean;
|
||||||
o_id?: number;
|
o_id?: number;
|
||||||
o_ram?: number;
|
o_ram?: number;
|
||||||
@@ -431,11 +393,10 @@ export interface EntityItem {
|
|||||||
o_deleted?: boolean;
|
o_deleted?: boolean;
|
||||||
o_writeable?: boolean;
|
o_writeable?: boolean;
|
||||||
o_value?: unknown;
|
o_value?: unknown;
|
||||||
o_hide?: boolean;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Entities {
|
export interface Entities {
|
||||||
readonly entities: readonly EntityItem[];
|
entities: EntityItem[];
|
||||||
}
|
}
|
||||||
|
|
||||||
// matches emsdevice.h DeviceType
|
// matches emsdevice.h DeviceType
|
||||||
@@ -491,4 +452,4 @@ export const DeviceValueTypeNames = [
|
|||||||
'ENUM',
|
'ENUM',
|
||||||
'RAW',
|
'RAW',
|
||||||
'CMD'
|
'CMD'
|
||||||
] as const;
|
];
|
||||||
|
|||||||
@@ -11,273 +11,376 @@ import type {
|
|||||||
TemperatureSensor
|
TemperatureSensor
|
||||||
} from './types';
|
} from './types';
|
||||||
|
|
||||||
// Constants
|
export const GPIO_VALIDATOR = {
|
||||||
const ERROR_MESSAGES = {
|
validator(
|
||||||
GPIO_INVALID: 'Must be an valid GPIO port',
|
rule: InternalRuleItem,
|
||||||
NAME_DUPLICATE: 'Name already in use',
|
value: number,
|
||||||
GPIO_DUPLICATE: 'GPIO already in use',
|
callback: (error?: string) => void
|
||||||
VALUE_OUT_OF_RANGE: 'Value out of range',
|
) {
|
||||||
HEX_REQUIRED: 'Is required and must be in hex format'
|
if (
|
||||||
} as const;
|
value &&
|
||||||
|
(value === 1 ||
|
||||||
const VALIDATION_LIMITS = {
|
(value >= 6 && value <= 11) ||
|
||||||
PORT_MIN: 0,
|
value === 20 ||
|
||||||
PORT_MAX: 65535,
|
value === 24 ||
|
||||||
MODBUS_MAX_CLIENTS_MIN: 0,
|
(value >= 28 && value <= 31) ||
|
||||||
MODBUS_MAX_CLIENTS_MAX: 50,
|
value > 40 ||
|
||||||
MODBUS_TIMEOUT_MIN: 100,
|
value < 0)
|
||||||
MODBUS_TIMEOUT_MAX: 20000,
|
) {
|
||||||
SYSLOG_MARK_INTERVAL_MIN: 0,
|
callback('Must be an valid GPIO port');
|
||||||
SYSLOG_MARK_INTERVAL_MAX: 3600,
|
} else {
|
||||||
SHOWER_MIN_DURATION_MIN: 10,
|
callback();
|
||||||
SHOWER_MIN_DURATION_MAX: 360,
|
}
|
||||||
SHOWER_ALERT_TRIGGER_MIN: 1,
|
|
||||||
SHOWER_ALERT_TRIGGER_MAX: 20,
|
|
||||||
SHOWER_ALERT_COLDSHOT_MIN: 1,
|
|
||||||
SHOWER_ALERT_COLDSHOT_MAX: 10,
|
|
||||||
REMOTE_TIMEOUT_MIN: 1,
|
|
||||||
REMOTE_TIMEOUT_MAX: 240,
|
|
||||||
OFFSET_MIN: 0,
|
|
||||||
OFFSET_MAX: 255,
|
|
||||||
COMMAND_MIN: 1,
|
|
||||||
COMMAND_MAX: 300,
|
|
||||||
NAME_MAX_LENGTH: 19,
|
|
||||||
HEX_BASE: 16
|
|
||||||
} as const;
|
|
||||||
|
|
||||||
type ValidationRules = Array<{
|
|
||||||
required?: boolean;
|
|
||||||
message?: string;
|
|
||||||
[key: string]: unknown;
|
|
||||||
}>;
|
|
||||||
|
|
||||||
export const createSettingsValidator = (settings: Settings) => {
|
|
||||||
const schema: Record<string, ValidationRules> = {};
|
|
||||||
|
|
||||||
// Syslog validations
|
|
||||||
if (settings.syslog_enabled) {
|
|
||||||
schema.syslog_host = [
|
|
||||||
{ required: true, message: 'Host is required' },
|
|
||||||
IP_OR_HOSTNAME_VALIDATOR
|
|
||||||
];
|
|
||||||
schema.syslog_port = [
|
|
||||||
{ required: true, message: 'Port is required' },
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: VALIDATION_LIMITS.PORT_MIN,
|
|
||||||
max: VALIDATION_LIMITS.PORT_MAX,
|
|
||||||
message: 'Invalid Port'
|
|
||||||
}
|
|
||||||
];
|
|
||||||
schema.syslog_mark_interval = [
|
|
||||||
{ required: true, message: 'Mark interval is required' },
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: VALIDATION_LIMITS.SYSLOG_MARK_INTERVAL_MIN,
|
|
||||||
max: VALIDATION_LIMITS.SYSLOG_MARK_INTERVAL_MAX,
|
|
||||||
message: `Must be between ${VALIDATION_LIMITS.SYSLOG_MARK_INTERVAL_MIN} and ${VALIDATION_LIMITS.SYSLOG_MARK_INTERVAL_MAX}`
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Modbus validations
|
|
||||||
if (settings.modbus_enabled) {
|
|
||||||
schema.modbus_max_clients = [
|
|
||||||
{ required: true, message: 'Max clients is required' },
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: VALIDATION_LIMITS.MODBUS_MAX_CLIENTS_MIN,
|
|
||||||
max: VALIDATION_LIMITS.MODBUS_MAX_CLIENTS_MAX,
|
|
||||||
message: 'Invalid number'
|
|
||||||
}
|
|
||||||
];
|
|
||||||
schema.modbus_port = [
|
|
||||||
{ required: true, message: 'Port is required' },
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: VALIDATION_LIMITS.PORT_MIN,
|
|
||||||
max: VALIDATION_LIMITS.PORT_MAX,
|
|
||||||
message: 'Invalid Port'
|
|
||||||
}
|
|
||||||
];
|
|
||||||
schema.modbus_timeout = [
|
|
||||||
{ required: true, message: 'Timeout is required' },
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: VALIDATION_LIMITS.MODBUS_TIMEOUT_MIN,
|
|
||||||
max: VALIDATION_LIMITS.MODBUS_TIMEOUT_MAX,
|
|
||||||
message: `Must be between ${VALIDATION_LIMITS.MODBUS_TIMEOUT_MIN} and ${VALIDATION_LIMITS.MODBUS_TIMEOUT_MAX}`
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
// Shower timer validations
|
|
||||||
if (settings.shower_timer) {
|
|
||||||
schema.shower_min_duration = [
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: VALIDATION_LIMITS.SHOWER_MIN_DURATION_MIN,
|
|
||||||
max: VALIDATION_LIMITS.SHOWER_MIN_DURATION_MAX,
|
|
||||||
message: `Time must be between ${VALIDATION_LIMITS.SHOWER_MIN_DURATION_MIN} and ${VALIDATION_LIMITS.SHOWER_MIN_DURATION_MAX} seconds`
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
// Shower alert validations
|
|
||||||
if (settings.shower_alert) {
|
|
||||||
schema.shower_alert_trigger = [
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: VALIDATION_LIMITS.SHOWER_ALERT_TRIGGER_MIN,
|
|
||||||
max: VALIDATION_LIMITS.SHOWER_ALERT_TRIGGER_MAX,
|
|
||||||
message: `Time must be between ${VALIDATION_LIMITS.SHOWER_ALERT_TRIGGER_MIN} and ${VALIDATION_LIMITS.SHOWER_ALERT_TRIGGER_MAX} minutes`
|
|
||||||
}
|
|
||||||
];
|
|
||||||
schema.shower_alert_coldshot = [
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: VALIDATION_LIMITS.SHOWER_ALERT_COLDSHOT_MIN,
|
|
||||||
max: VALIDATION_LIMITS.SHOWER_ALERT_COLDSHOT_MAX,
|
|
||||||
message: `Time must be between ${VALIDATION_LIMITS.SHOWER_ALERT_COLDSHOT_MIN} and ${VALIDATION_LIMITS.SHOWER_ALERT_COLDSHOT_MAX} seconds`
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remote timeout validations
|
|
||||||
if (settings.remote_timeout_en) {
|
|
||||||
schema.remote_timeout = [
|
|
||||||
{
|
|
||||||
type: 'number',
|
|
||||||
min: VALIDATION_LIMITS.REMOTE_TIMEOUT_MIN,
|
|
||||||
max: VALIDATION_LIMITS.REMOTE_TIMEOUT_MAX,
|
|
||||||
message: `Timeout must be between ${VALIDATION_LIMITS.REMOTE_TIMEOUT_MIN} and ${VALIDATION_LIMITS.REMOTE_TIMEOUT_MAX} hours`
|
|
||||||
}
|
|
||||||
];
|
|
||||||
}
|
|
||||||
|
|
||||||
return new Schema(schema);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// Generic unique name validator factory
|
export const GPIO_VALIDATORR = {
|
||||||
const createUniqueNameValidator = <T extends { name: string }>(
|
|
||||||
items: T[],
|
|
||||||
originalName?: string
|
|
||||||
) => ({
|
|
||||||
validator(
|
validator(
|
||||||
_rule: InternalRuleItem,
|
rule: InternalRuleItem,
|
||||||
|
value: number,
|
||||||
|
callback: (error?: string) => void
|
||||||
|
) {
|
||||||
|
if (
|
||||||
|
value &&
|
||||||
|
(value === 1 ||
|
||||||
|
(value >= 6 && value <= 11) ||
|
||||||
|
(value >= 16 && value <= 17) ||
|
||||||
|
value === 20 ||
|
||||||
|
value === 24 ||
|
||||||
|
(value >= 28 && value <= 31) ||
|
||||||
|
value > 40 ||
|
||||||
|
value < 0)
|
||||||
|
) {
|
||||||
|
callback('Must be an valid GPIO port');
|
||||||
|
} else {
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const GPIO_VALIDATORC3 = {
|
||||||
|
validator(
|
||||||
|
rule: InternalRuleItem,
|
||||||
|
value: number,
|
||||||
|
callback: (error?: string) => void
|
||||||
|
) {
|
||||||
|
if (value && ((value >= 11 && value <= 19) || value > 21 || value < 0)) {
|
||||||
|
callback('Must be an valid GPIO port');
|
||||||
|
} else {
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const GPIO_VALIDATORS2 = {
|
||||||
|
validator(
|
||||||
|
rule: InternalRuleItem,
|
||||||
|
value: number,
|
||||||
|
callback: (error?: string) => void
|
||||||
|
) {
|
||||||
|
if (
|
||||||
|
value &&
|
||||||
|
((value >= 19 && value <= 20) ||
|
||||||
|
(value >= 22 && value <= 32) ||
|
||||||
|
value > 40 ||
|
||||||
|
value < 0)
|
||||||
|
) {
|
||||||
|
callback('Must be an valid GPIO port');
|
||||||
|
} else {
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const GPIO_VALIDATORS3 = {
|
||||||
|
validator(
|
||||||
|
rule: InternalRuleItem,
|
||||||
|
value: number,
|
||||||
|
callback: (error?: string) => void
|
||||||
|
) {
|
||||||
|
if (
|
||||||
|
value &&
|
||||||
|
((value >= 19 && value <= 20) ||
|
||||||
|
(value >= 22 && value <= 37) ||
|
||||||
|
(value >= 39 && value <= 42) ||
|
||||||
|
value > 48 ||
|
||||||
|
value < 0)
|
||||||
|
) {
|
||||||
|
callback('Must be an valid GPIO port');
|
||||||
|
} else {
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createSettingsValidator = (settings: Settings) =>
|
||||||
|
new Schema({
|
||||||
|
...(settings.board_profile === 'CUSTOM' &&
|
||||||
|
settings.platform === 'ESP32' && {
|
||||||
|
led_gpio: [
|
||||||
|
{ required: true, message: 'LED GPIO is required' },
|
||||||
|
GPIO_VALIDATOR
|
||||||
|
],
|
||||||
|
dallas_gpio: [
|
||||||
|
{ required: true, message: 'GPIO is required' },
|
||||||
|
GPIO_VALIDATOR
|
||||||
|
],
|
||||||
|
pbutton_gpio: [
|
||||||
|
{ required: true, message: 'Button GPIO is required' },
|
||||||
|
GPIO_VALIDATOR
|
||||||
|
],
|
||||||
|
tx_gpio: [
|
||||||
|
{ required: true, message: 'Tx GPIO is required' },
|
||||||
|
GPIO_VALIDATOR
|
||||||
|
],
|
||||||
|
rx_gpio: [{ required: true, message: 'Rx GPIO is required' }, GPIO_VALIDATOR]
|
||||||
|
}),
|
||||||
|
...(settings.board_profile === 'CUSTOM' &&
|
||||||
|
settings.platform === 'ESP32C3' && {
|
||||||
|
led_gpio: [
|
||||||
|
{ required: true, message: 'LED GPIO is required' },
|
||||||
|
GPIO_VALIDATORC3
|
||||||
|
],
|
||||||
|
dallas_gpio: [
|
||||||
|
{ required: true, message: 'GPIO is required' },
|
||||||
|
GPIO_VALIDATORC3
|
||||||
|
],
|
||||||
|
pbutton_gpio: [
|
||||||
|
{ required: true, message: 'Button GPIO is required' },
|
||||||
|
GPIO_VALIDATORC3
|
||||||
|
],
|
||||||
|
tx_gpio: [
|
||||||
|
{ required: true, message: 'Tx GPIO is required' },
|
||||||
|
GPIO_VALIDATORC3
|
||||||
|
],
|
||||||
|
rx_gpio: [
|
||||||
|
{ required: true, message: 'Rx GPIO is required' },
|
||||||
|
GPIO_VALIDATORC3
|
||||||
|
]
|
||||||
|
}),
|
||||||
|
...(settings.board_profile === 'CUSTOM' &&
|
||||||
|
settings.platform === 'ESP32S2' && {
|
||||||
|
led_gpio: [
|
||||||
|
{ required: true, message: 'LED GPIO is required' },
|
||||||
|
GPIO_VALIDATORS2
|
||||||
|
],
|
||||||
|
dallas_gpio: [
|
||||||
|
{ required: true, message: 'GPIO is required' },
|
||||||
|
GPIO_VALIDATORS2
|
||||||
|
],
|
||||||
|
pbutton_gpio: [
|
||||||
|
{ required: true, message: 'Button GPIO is required' },
|
||||||
|
GPIO_VALIDATORS2
|
||||||
|
],
|
||||||
|
tx_gpio: [
|
||||||
|
{ required: true, message: 'Tx GPIO is required' },
|
||||||
|
GPIO_VALIDATORS2
|
||||||
|
],
|
||||||
|
rx_gpio: [
|
||||||
|
{ required: true, message: 'Rx GPIO is required' },
|
||||||
|
GPIO_VALIDATORS2
|
||||||
|
]
|
||||||
|
}),
|
||||||
|
...(settings.board_profile === 'CUSTOM' &&
|
||||||
|
settings.platform === 'ESP32S3' && {
|
||||||
|
led_gpio: [
|
||||||
|
{ required: true, message: 'LED GPIO is required' },
|
||||||
|
GPIO_VALIDATORS3
|
||||||
|
],
|
||||||
|
dallas_gpio: [
|
||||||
|
{ required: true, message: 'GPIO is required' },
|
||||||
|
GPIO_VALIDATORS3
|
||||||
|
],
|
||||||
|
pbutton_gpio: [
|
||||||
|
{ required: true, message: 'Button GPIO is required' },
|
||||||
|
GPIO_VALIDATORS3
|
||||||
|
],
|
||||||
|
tx_gpio: [
|
||||||
|
{ required: true, message: 'Tx GPIO is required' },
|
||||||
|
GPIO_VALIDATORS3
|
||||||
|
],
|
||||||
|
rx_gpio: [
|
||||||
|
{ required: true, message: 'Rx GPIO is required' },
|
||||||
|
GPIO_VALIDATORS3
|
||||||
|
]
|
||||||
|
}),
|
||||||
|
...(settings.syslog_enabled && {
|
||||||
|
syslog_host: [
|
||||||
|
{ required: true, message: 'Host is required' },
|
||||||
|
IP_OR_HOSTNAME_VALIDATOR
|
||||||
|
],
|
||||||
|
syslog_port: [
|
||||||
|
{ required: true, message: 'Port is required' },
|
||||||
|
{ type: 'number', min: 0, max: 65535, message: 'Invalid Port' }
|
||||||
|
],
|
||||||
|
syslog_mark_interval: [
|
||||||
|
{ required: true, message: 'Mark interval is required' },
|
||||||
|
{ type: 'number', min: 0, max: 10, message: 'Must be between 0 and 10' }
|
||||||
|
]
|
||||||
|
}),
|
||||||
|
...(settings.modbus_enabled && {
|
||||||
|
modbus_max_clients: [
|
||||||
|
{ required: true, message: 'Max clients is required' },
|
||||||
|
{ type: 'number', min: 0, max: 50, message: 'Invalid number' }
|
||||||
|
],
|
||||||
|
modbus_port: [
|
||||||
|
{ required: true, message: 'Port is required' },
|
||||||
|
{ type: 'number', min: 0, max: 65535, message: 'Invalid Port' }
|
||||||
|
],
|
||||||
|
modbus_timeout: [
|
||||||
|
{ required: true, message: 'Timeout is required' },
|
||||||
|
{
|
||||||
|
type: 'number',
|
||||||
|
min: 100,
|
||||||
|
max: 20000,
|
||||||
|
message: 'Must be between 100 and 20000'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}),
|
||||||
|
...(settings.shower_timer && {
|
||||||
|
shower_min_duration: [
|
||||||
|
{
|
||||||
|
type: 'number',
|
||||||
|
min: 10,
|
||||||
|
max: 360,
|
||||||
|
message: 'Time must be between 10 and 360 seconds'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}),
|
||||||
|
...(settings.shower_alert && {
|
||||||
|
shower_alert_trigger: [
|
||||||
|
{
|
||||||
|
type: 'number',
|
||||||
|
min: 1,
|
||||||
|
max: 20,
|
||||||
|
message: 'Time must be between 1 and 20 minutes'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
shower_alert_coldshot: [
|
||||||
|
{
|
||||||
|
type: 'number',
|
||||||
|
min: 1,
|
||||||
|
max: 10,
|
||||||
|
message: 'Time must be between 1 and 10 seconds'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}),
|
||||||
|
...(settings.remote_timeout_en && {
|
||||||
|
remote_timeout: [
|
||||||
|
{
|
||||||
|
type: 'number',
|
||||||
|
min: 1,
|
||||||
|
max: 240,
|
||||||
|
message: 'Timeout must be between 1 and 240 hours'
|
||||||
|
}
|
||||||
|
]
|
||||||
|
})
|
||||||
|
});
|
||||||
|
|
||||||
|
export const uniqueNameValidator = (schedule: ScheduleItem[], o_name?: string) => ({
|
||||||
|
validator(
|
||||||
|
rule: InternalRuleItem,
|
||||||
name: string,
|
name: string,
|
||||||
callback: (error?: string) => void
|
callback: (error?: string) => void
|
||||||
) {
|
) {
|
||||||
if (
|
if (
|
||||||
name !== '' &&
|
name !== '' &&
|
||||||
(originalName === undefined ||
|
(o_name === undefined || o_name.toLowerCase() !== name.toLowerCase()) &&
|
||||||
originalName.toLowerCase() !== name.toLowerCase()) &&
|
schedule.find((si) => si.name.toLowerCase() === name.toLowerCase())
|
||||||
items.find((item) => item.name.toLowerCase() === name.toLowerCase())
|
|
||||||
) {
|
) {
|
||||||
callback(ERROR_MESSAGES.NAME_DUPLICATE);
|
callback('Name already in use');
|
||||||
return;
|
} else {
|
||||||
|
callback();
|
||||||
}
|
}
|
||||||
callback();
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Generic field name validator (for cases where the name field has different property names)
|
|
||||||
const createUniqueFieldNameValidator = <T>(
|
|
||||||
items: T[],
|
|
||||||
getName: (item: T) => string,
|
|
||||||
originalName?: string
|
|
||||||
) => ({
|
|
||||||
validator(
|
|
||||||
_rule: InternalRuleItem,
|
|
||||||
name: string,
|
|
||||||
callback: (error?: string) => void
|
|
||||||
) {
|
|
||||||
if (
|
|
||||||
name !== '' &&
|
|
||||||
(originalName === undefined ||
|
|
||||||
originalName.toLowerCase() !== name.toLowerCase()) &&
|
|
||||||
items.find((item) => getName(item).toLowerCase() === name.toLowerCase())
|
|
||||||
) {
|
|
||||||
callback(ERROR_MESSAGES.NAME_DUPLICATE);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
callback();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
const NAME_PATTERN_BASE = '[a-zA-Z0-9_]';
|
|
||||||
const NAME_PATTERN_MESSAGE = `Must be <${VALIDATION_LIMITS.NAME_MAX_LENGTH + 1} characters: alphanumeric or '_'`;
|
|
||||||
|
|
||||||
const NAME_PATTERN = {
|
|
||||||
type: 'string' as const,
|
|
||||||
pattern: new RegExp(
|
|
||||||
`^${NAME_PATTERN_BASE}{0,${VALIDATION_LIMITS.NAME_MAX_LENGTH}}$`
|
|
||||||
),
|
|
||||||
message: NAME_PATTERN_MESSAGE
|
|
||||||
};
|
|
||||||
|
|
||||||
const NAME_PATTERN_REQUIRED = {
|
|
||||||
type: 'string' as const,
|
|
||||||
pattern: new RegExp(
|
|
||||||
`^${NAME_PATTERN_BASE}{1,${VALIDATION_LIMITS.NAME_MAX_LENGTH}}$`
|
|
||||||
),
|
|
||||||
message: NAME_PATTERN_MESSAGE
|
|
||||||
};
|
|
||||||
|
|
||||||
export const uniqueNameValidator = (schedule: ScheduleItem[], o_name?: string) =>
|
|
||||||
createUniqueNameValidator(schedule, o_name);
|
|
||||||
|
|
||||||
export const schedulerItemValidation = (
|
export const schedulerItemValidation = (
|
||||||
schedule: ScheduleItem[],
|
schedule: ScheduleItem[],
|
||||||
scheduleItem: ScheduleItem
|
scheduleItem: ScheduleItem
|
||||||
) =>
|
) =>
|
||||||
new Schema({
|
new Schema({
|
||||||
name: [NAME_PATTERN, uniqueNameValidator(schedule, scheduleItem.o_name)],
|
name: [
|
||||||
|
{
|
||||||
|
type: 'string',
|
||||||
|
pattern: /^[a-zA-Z0-9_]{0,19}$/,
|
||||||
|
message: "Must be <20 characters: alphanumeric or '_'"
|
||||||
|
},
|
||||||
|
...[uniqueNameValidator(schedule, scheduleItem.o_name)]
|
||||||
|
],
|
||||||
cmd: [
|
cmd: [
|
||||||
{ required: true, message: 'Command is required' },
|
{ required: true, message: 'Command is required' },
|
||||||
{
|
{
|
||||||
type: 'string',
|
type: 'string',
|
||||||
min: VALIDATION_LIMITS.COMMAND_MIN,
|
min: 1,
|
||||||
max: VALIDATION_LIMITS.COMMAND_MAX,
|
max: 300,
|
||||||
message: `Command must be ${VALIDATION_LIMITS.COMMAND_MIN}-${VALIDATION_LIMITS.COMMAND_MAX} characters`
|
message: 'Command must be 1-300 characters'
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
});
|
});
|
||||||
|
|
||||||
export const uniqueCustomNameValidator = (entity: EntityItem[], o_name?: string) =>
|
export const uniqueCustomNameValidator = (
|
||||||
createUniqueNameValidator(entity, o_name);
|
entity: EntityItem[],
|
||||||
|
o_name?: string
|
||||||
const hexValidator = {
|
) => ({
|
||||||
validator(
|
validator(
|
||||||
_rule: InternalRuleItem,
|
rule: InternalRuleItem,
|
||||||
value: string,
|
name: string,
|
||||||
callback: (error?: string) => void
|
callback: (error?: string) => void
|
||||||
) {
|
) {
|
||||||
if (!value || Number.isNaN(Number.parseInt(value, VALIDATION_LIMITS.HEX_BASE))) {
|
if (
|
||||||
callback(ERROR_MESSAGES.HEX_REQUIRED);
|
(o_name === undefined || o_name.toLowerCase() !== name.toLowerCase()) &&
|
||||||
return;
|
entity.find((ei) => ei.name.toLowerCase() === name.toLowerCase())
|
||||||
|
) {
|
||||||
|
callback('Name already in use');
|
||||||
|
} else {
|
||||||
|
callback();
|
||||||
}
|
}
|
||||||
callback();
|
|
||||||
}
|
}
|
||||||
};
|
});
|
||||||
|
|
||||||
export const entityItemValidation = (entity: EntityItem[], entityItem: EntityItem) =>
|
export const entityItemValidation = (entity: EntityItem[], entityItem: EntityItem) =>
|
||||||
new Schema({
|
new Schema({
|
||||||
name: [
|
name: [
|
||||||
{ required: true, message: 'Name is required' },
|
{ required: true, message: 'Name is required' },
|
||||||
NAME_PATTERN_REQUIRED,
|
{
|
||||||
uniqueCustomNameValidator(entity, entityItem.o_name)
|
type: 'string',
|
||||||
|
pattern: /^[a-zA-Z0-9_]{1,19}$/,
|
||||||
|
message: "Must be <20 characters: alphanumeric or '_'"
|
||||||
|
},
|
||||||
|
...[uniqueCustomNameValidator(entity, entityItem.o_name)]
|
||||||
|
],
|
||||||
|
device_id: [
|
||||||
|
{
|
||||||
|
validator(
|
||||||
|
rule: InternalRuleItem,
|
||||||
|
value: string,
|
||||||
|
callback: (error?: string) => void
|
||||||
|
) {
|
||||||
|
if (isNaN(parseInt(value, 16))) {
|
||||||
|
callback('Is required and must be in hex format');
|
||||||
|
}
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
type_id: [
|
||||||
|
{
|
||||||
|
validator(
|
||||||
|
rule: InternalRuleItem,
|
||||||
|
value: string,
|
||||||
|
callback: (error?: string) => void
|
||||||
|
) {
|
||||||
|
if (isNaN(parseInt(value, 16))) {
|
||||||
|
callback('Is required and must be in hex format');
|
||||||
|
}
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
}
|
||||||
],
|
],
|
||||||
device_id: [hexValidator],
|
|
||||||
type_id: [hexValidator],
|
|
||||||
offset: [
|
offset: [
|
||||||
{ required: true, message: 'Offset is required' },
|
{ required: true, message: 'Offset is required' },
|
||||||
{
|
{ type: 'number', min: 0, max: 255, message: 'Must be between 0 and 255' }
|
||||||
type: 'number',
|
|
||||||
min: VALIDATION_LIMITS.OFFSET_MIN,
|
|
||||||
max: VALIDATION_LIMITS.OFFSET_MAX,
|
|
||||||
message: `Must be between ${VALIDATION_LIMITS.OFFSET_MIN} and ${VALIDATION_LIMITS.OFFSET_MAX}`
|
|
||||||
}
|
|
||||||
],
|
],
|
||||||
factor: [{ required: true, message: 'is required' }]
|
factor: [{ required: true, message: 'is required' }]
|
||||||
});
|
});
|
||||||
@@ -285,34 +388,93 @@ export const entityItemValidation = (entity: EntityItem[], entityItem: EntityIte
|
|||||||
export const uniqueTemperatureNameValidator = (
|
export const uniqueTemperatureNameValidator = (
|
||||||
sensors: TemperatureSensor[],
|
sensors: TemperatureSensor[],
|
||||||
o_name?: string
|
o_name?: string
|
||||||
) => createUniqueFieldNameValidator(sensors, (s) => s.n, o_name);
|
) => ({
|
||||||
|
validator(rule: InternalRuleItem, n: string, callback: (error?: string) => void) {
|
||||||
|
if (
|
||||||
|
(o_name === undefined || o_name.toLowerCase() !== n.toLowerCase()) &&
|
||||||
|
n !== '' &&
|
||||||
|
sensors.find((ts) => ts.n.toLowerCase() === n.toLowerCase())
|
||||||
|
) {
|
||||||
|
callback('Name already in use');
|
||||||
|
} else {
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
export const temperatureSensorItemValidation = (
|
export const temperatureSensorItemValidation = (
|
||||||
sensors: TemperatureSensor[],
|
sensors: TemperatureSensor[],
|
||||||
sensor: TemperatureSensor
|
sensor: TemperatureSensor
|
||||||
) =>
|
) =>
|
||||||
new Schema({
|
new Schema({
|
||||||
n: [NAME_PATTERN, uniqueTemperatureNameValidator(sensors, sensor.o_n)]
|
n: [
|
||||||
|
{
|
||||||
|
type: 'string',
|
||||||
|
pattern: /^[a-zA-Z0-9_]{0,19}$/,
|
||||||
|
message: "Must be <20 characters: alphanumeric or '_'"
|
||||||
|
},
|
||||||
|
...[uniqueTemperatureNameValidator(sensors, sensor.o_n)]
|
||||||
|
]
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export const isGPIOUniqueValidator = (sensors: AnalogSensor[]) => ({
|
||||||
|
validator(
|
||||||
|
rule: InternalRuleItem,
|
||||||
|
gpio: number,
|
||||||
|
callback: (error?: string) => void
|
||||||
|
) {
|
||||||
|
if (sensors.find((as) => as.g === gpio)) {
|
||||||
|
callback('GPIO already in use');
|
||||||
|
} else {
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
export const uniqueAnalogNameValidator = (
|
export const uniqueAnalogNameValidator = (
|
||||||
sensors: AnalogSensor[],
|
sensors: AnalogSensor[],
|
||||||
o_name?: string
|
o_name?: string
|
||||||
) => createUniqueFieldNameValidator(sensors, (s) => s.n, o_name);
|
) => ({
|
||||||
|
validator(rule: InternalRuleItem, n: string, callback: (error?: string) => void) {
|
||||||
|
if (
|
||||||
|
(o_name === undefined || o_name.toLowerCase() !== n.toLowerCase()) &&
|
||||||
|
n !== '' &&
|
||||||
|
sensors.find((as) => as.n.toLowerCase() === n.toLowerCase())
|
||||||
|
) {
|
||||||
|
callback('Name already in use');
|
||||||
|
} else {
|
||||||
|
callback();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
export const analogSensorItemValidation = (
|
export const analogSensorItemValidation = (
|
||||||
sensors: AnalogSensor[],
|
sensors: AnalogSensor[],
|
||||||
sensor: AnalogSensor
|
sensor: AnalogSensor,
|
||||||
) => {
|
creating: boolean,
|
||||||
return new Schema({
|
platform: string
|
||||||
// name is required and must be unique
|
) =>
|
||||||
|
new Schema({
|
||||||
n: [
|
n: [
|
||||||
{ required: true, message: 'Name is required' },
|
{
|
||||||
NAME_PATTERN,
|
type: 'string',
|
||||||
uniqueAnalogNameValidator(sensors, sensor.o_n)
|
pattern: /^[a-zA-Z0-9_]{0,19}$/,
|
||||||
|
message: "Must be <20 characters: alphanumeric or '_'"
|
||||||
|
},
|
||||||
|
...[uniqueAnalogNameValidator(sensors, sensor.o_n)]
|
||||||
|
],
|
||||||
|
g: [
|
||||||
|
{ required: true, message: 'GPIO is required' },
|
||||||
|
platform === 'ESP32S3'
|
||||||
|
? GPIO_VALIDATORS3
|
||||||
|
: platform === 'ESP32S2'
|
||||||
|
? GPIO_VALIDATORS2
|
||||||
|
: platform === 'ESP32C3'
|
||||||
|
? GPIO_VALIDATORC3
|
||||||
|
: GPIO_VALIDATOR,
|
||||||
|
...(creating ? [isGPIOUniqueValidator(sensors)] : [])
|
||||||
]
|
]
|
||||||
});
|
});
|
||||||
};
|
|
||||||
|
|
||||||
export const deviceValueItemValidation = (dv: DeviceValue) =>
|
export const deviceValueItemValidation = (dv: DeviceValue) =>
|
||||||
new Schema({
|
new Schema({
|
||||||
@@ -320,18 +482,17 @@ export const deviceValueItemValidation = (dv: DeviceValue) =>
|
|||||||
{ required: true, message: 'Value is required' },
|
{ required: true, message: 'Value is required' },
|
||||||
{
|
{
|
||||||
validator(
|
validator(
|
||||||
_rule: InternalRuleItem,
|
rule: InternalRuleItem,
|
||||||
value: unknown,
|
value: unknown,
|
||||||
callback: (error?: string) => void
|
callback: (error?: string) => void
|
||||||
) {
|
) {
|
||||||
if (
|
if (
|
||||||
typeof value === 'number' &&
|
typeof value === 'number' &&
|
||||||
dv.m !== undefined &&
|
dv.m &&
|
||||||
dv.x !== undefined &&
|
dv.x &&
|
||||||
(value < dv.m || value > dv.x)
|
(value < dv.m || value > dv.x)
|
||||||
) {
|
) {
|
||||||
callback(ERROR_MESSAGES.VALUE_OUT_OF_RANGE);
|
callback('Value out of range');
|
||||||
return;
|
|
||||||
}
|
}
|
||||||
callback();
|
callback();
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useMemo, useState } from 'react';
|
import { useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
@@ -27,19 +27,6 @@ export const isAPEnabled = ({ provision_mode }: APSettingsType) =>
|
|||||||
provision_mode === APProvisionMode.AP_MODE_ALWAYS ||
|
provision_mode === APProvisionMode.AP_MODE_ALWAYS ||
|
||||||
provision_mode === APProvisionMode.AP_MODE_DISCONNECTED;
|
provision_mode === APProvisionMode.AP_MODE_DISCONNECTED;
|
||||||
|
|
||||||
// Efficient range function without recursion
|
|
||||||
const createRange = (start: number, end: number): number[] => {
|
|
||||||
const result: number[] = [];
|
|
||||||
for (let i = start; i <= end; i++) {
|
|
||||||
result.push(i);
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Pre-computed ranges for better performance
|
|
||||||
const CHANNEL_RANGE = createRange(1, 14);
|
|
||||||
const MAX_CLIENTS_RANGE = createRange(1, 9);
|
|
||||||
|
|
||||||
const APSettings = () => {
|
const APSettings = () => {
|
||||||
const {
|
const {
|
||||||
loadData,
|
loadData,
|
||||||
@@ -63,42 +50,37 @@ const APSettings = () => {
|
|||||||
|
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
|
|
||||||
const updateFormValue = useMemo(
|
const updateFormValue = updateValueDirty(
|
||||||
() =>
|
origData,
|
||||||
updateValueDirty(
|
dirtyFlags,
|
||||||
origData as unknown as Record<string, unknown>,
|
setDirtyFlags,
|
||||||
dirtyFlags,
|
updateDataValue
|
||||||
setDirtyFlags,
|
|
||||||
updateDataValue as (value: unknown) => void
|
|
||||||
),
|
|
||||||
[origData, dirtyFlags, setDirtyFlags, updateDataValue]
|
|
||||||
);
|
);
|
||||||
|
|
||||||
// Memoize AP enabled state
|
|
||||||
const apEnabled = useMemo(() => (data ? isAPEnabled(data) : false), [data]);
|
|
||||||
|
|
||||||
// Memoize validation and submit handler
|
|
||||||
const validateAndSubmit = useCallback(async () => {
|
|
||||||
if (!data) return;
|
|
||||||
|
|
||||||
try {
|
|
||||||
setFieldErrors(undefined);
|
|
||||||
await validate(createAPSettingsValidator(data), data);
|
|
||||||
await saveData();
|
|
||||||
} catch (error) {
|
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
|
||||||
}
|
|
||||||
}, [data, saveData]);
|
|
||||||
|
|
||||||
const content = () => {
|
const content = () => {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
const validateAndSubmit = async () => {
|
||||||
|
try {
|
||||||
|
setFieldErrors(undefined);
|
||||||
|
await validate(createAPSettingsValidator(data), data);
|
||||||
|
await saveData();
|
||||||
|
} catch (error) {
|
||||||
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// no lodash - https://asleepace.com/blog/typescript-range-without-a-loop/
|
||||||
|
function range(a: number, b: number): number[] {
|
||||||
|
return a < b ? [a, ...range(a + 1, b)] : [b];
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="provision_mode"
|
name="provision_mode"
|
||||||
label={LL.AP_PROVIDE() + '...'}
|
label={LL.AP_PROVIDE() + '...'}
|
||||||
value={data.provision_mode}
|
value={data.provision_mode}
|
||||||
@@ -118,10 +100,10 @@ const APSettings = () => {
|
|||||||
{LL.AP_PROVIDE_TEXT_3()}
|
{LL.AP_PROVIDE_TEXT_3()}
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
</ValidatedTextField>
|
</ValidatedTextField>
|
||||||
{apEnabled && (
|
{isAPEnabled(data) && (
|
||||||
<>
|
<>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="ssid"
|
name="ssid"
|
||||||
label={LL.ACCESS_POINT(2) + ' SSID'}
|
label={LL.ACCESS_POINT(2) + ' SSID'}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -131,7 +113,7 @@ const APSettings = () => {
|
|||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
<ValidatedPasswordField
|
<ValidatedPasswordField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="password"
|
name="password"
|
||||||
label={LL.ACCESS_POINT(2) + ' ' + LL.PASSWORD()}
|
label={LL.ACCESS_POINT(2) + ' ' + LL.PASSWORD()}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -141,7 +123,7 @@ const APSettings = () => {
|
|||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="channel"
|
name="channel"
|
||||||
label={LL.AP_PREFERRED_CHANNEL()}
|
label={LL.AP_PREFERRED_CHANNEL()}
|
||||||
value={numberValue(data.channel)}
|
value={numberValue(data.channel)}
|
||||||
@@ -152,7 +134,7 @@ const APSettings = () => {
|
|||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
>
|
>
|
||||||
{CHANNEL_RANGE.map((i) => (
|
{range(1, 14).map((i) => (
|
||||||
<MenuItem key={i} value={i}>
|
<MenuItem key={i} value={i}>
|
||||||
{i}
|
{i}
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
@@ -169,7 +151,7 @@ const APSettings = () => {
|
|||||||
label={LL.AP_HIDE_SSID()}
|
label={LL.AP_HIDE_SSID()}
|
||||||
/>
|
/>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="max_clients"
|
name="max_clients"
|
||||||
label={LL.AP_MAX_CLIENTS()}
|
label={LL.AP_MAX_CLIENTS()}
|
||||||
value={numberValue(data.max_clients)}
|
value={numberValue(data.max_clients)}
|
||||||
@@ -180,14 +162,14 @@ const APSettings = () => {
|
|||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
>
|
>
|
||||||
{MAX_CLIENTS_RANGE.map((i) => (
|
{range(1, 9).map((i) => (
|
||||||
<MenuItem key={i} value={i}>
|
<MenuItem key={i} value={i}>
|
||||||
{i}
|
{i}
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
))}
|
))}
|
||||||
</ValidatedTextField>
|
</ValidatedTextField>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="local_ip"
|
name="local_ip"
|
||||||
label={LL.AP_LOCAL_IP()}
|
label={LL.AP_LOCAL_IP()}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -197,7 +179,7 @@ const APSettings = () => {
|
|||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="gateway_ip"
|
name="gateway_ip"
|
||||||
label={LL.NETWORK_GATEWAY()}
|
label={LL.NETWORK_GATEWAY()}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -207,7 +189,7 @@ const APSettings = () => {
|
|||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="subnet_mask"
|
name="subnet_mask"
|
||||||
label={LL.NETWORK_SUBNET()}
|
label={LL.NETWORK_SUBNET()}
|
||||||
fullWidth
|
fullWidth
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useMemo, useState } from 'react';
|
import { useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -9,7 +9,7 @@ import {
|
|||||||
Button,
|
Button,
|
||||||
Checkbox,
|
Checkbox,
|
||||||
Divider,
|
Divider,
|
||||||
Grid,
|
Grid2 as Grid,
|
||||||
InputAdornment,
|
InputAdornment,
|
||||||
MenuItem,
|
MenuItem,
|
||||||
TextField,
|
TextField,
|
||||||
@@ -28,7 +28,6 @@ import {
|
|||||||
FormLoader,
|
FormLoader,
|
||||||
MessageBox,
|
MessageBox,
|
||||||
SectionContent,
|
SectionContent,
|
||||||
ValidatedPasswordField,
|
|
||||||
ValidatedTextField,
|
ValidatedTextField,
|
||||||
useLayoutTitle
|
useLayoutTitle
|
||||||
} from 'components';
|
} from 'components';
|
||||||
@@ -38,13 +37,13 @@ import { validate } from 'validators';
|
|||||||
|
|
||||||
import { API, getBoardProfile, readSettings, writeSettings } from '../../api/app';
|
import { API, getBoardProfile, readSettings, writeSettings } from '../../api/app';
|
||||||
import { BOARD_PROFILES } from '../main/types';
|
import { BOARD_PROFILES } from '../main/types';
|
||||||
import type { APIcall, BoardProfileKey, Settings } from '../main/types';
|
import type { APIcall, Settings } from '../main/types';
|
||||||
import { createSettingsValidator } from '../main/validators';
|
import { createSettingsValidator } from '../main/validators';
|
||||||
|
|
||||||
export function boardProfileSelectItems() {
|
export function boardProfileSelectItems() {
|
||||||
return Object.keys(BOARD_PROFILES).map((code) => (
|
return Object.keys(BOARD_PROFILES).map((code) => (
|
||||||
<MenuItem key={code} value={code}>
|
<MenuItem key={code} value={code}>
|
||||||
{BOARD_PROFILES[code as BoardProfileKey]}
|
{BOARD_PROFILES[code]}
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
));
|
));
|
||||||
}
|
}
|
||||||
@@ -73,10 +72,10 @@ const ApplicationSettings = () => {
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
const updateFormValue = updateValueDirty(
|
const updateFormValue = updateValueDirty(
|
||||||
origData as unknown as Record<string, unknown>,
|
origData,
|
||||||
dirtyFlags,
|
dirtyFlags,
|
||||||
setDirtyFlags,
|
setDirtyFlags,
|
||||||
updateDataValue as (value: unknown) => void
|
updateDataValue
|
||||||
);
|
);
|
||||||
|
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
@@ -107,61 +106,50 @@ const ApplicationSettings = () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// Memoized input props to prevent recreation on every render
|
const doRestart = async () => {
|
||||||
const SecondsInputProps = useMemo(
|
|
||||||
() => ({
|
|
||||||
endAdornment: <InputAdornment position="end">{LL.SECONDS()}</InputAdornment>
|
|
||||||
}),
|
|
||||||
[LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const MinutesInputProps = useMemo(
|
|
||||||
() => ({
|
|
||||||
endAdornment: <InputAdornment position="end">{LL.MINUTES()}</InputAdornment>
|
|
||||||
}),
|
|
||||||
[LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const HoursInputProps = useMemo(
|
|
||||||
() => ({
|
|
||||||
endAdornment: <InputAdornment position="end">{LL.HOURS()}</InputAdornment>
|
|
||||||
}),
|
|
||||||
[LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const doRestart = useCallback(async () => {
|
|
||||||
setRestarting(true);
|
setRestarting(true);
|
||||||
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
||||||
(error: Error) => {
|
(error: Error) => {
|
||||||
toast.error(error.message);
|
toast.error(error.message);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}, [sendAPI]);
|
};
|
||||||
|
|
||||||
const updateBoardProfile = useCallback(
|
const updateBoardProfile = async (board_profile: string) => {
|
||||||
async (board_profile: string) => {
|
await readBoardProfile(board_profile).catch((error: Error) => {
|
||||||
await readBoardProfile(board_profile).catch((error: Error) => {
|
toast.error(error.message);
|
||||||
toast.error(error.message);
|
});
|
||||||
});
|
};
|
||||||
},
|
|
||||||
[readBoardProfile]
|
|
||||||
);
|
|
||||||
|
|
||||||
useLayoutTitle(LL.APPLICATION());
|
useLayoutTitle(LL.APPLICATION());
|
||||||
|
|
||||||
const validateAndSubmit = useCallback(async () => {
|
const SecondsInputProps = {
|
||||||
try {
|
endAdornment: <InputAdornment position="end">{LL.SECONDS()}</InputAdornment>
|
||||||
setFieldErrors(undefined);
|
};
|
||||||
await validate(createSettingsValidator(data), data);
|
const MinutesInputProps = {
|
||||||
} catch (error) {
|
endAdornment: <InputAdornment position="end">{LL.MINUTES()}</InputAdornment>
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
};
|
||||||
} finally {
|
const HoursInputProps = {
|
||||||
await saveData();
|
endAdornment: <InputAdornment position="end">{LL.HOURS()}</InputAdornment>
|
||||||
}
|
};
|
||||||
}, [data, saveData]);
|
|
||||||
|
|
||||||
const changeBoardProfile = useCallback(
|
const content = () => {
|
||||||
(event: React.ChangeEvent<HTMLInputElement>) => {
|
if (!data || !hardwareData) {
|
||||||
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
const validateAndSubmit = async () => {
|
||||||
|
try {
|
||||||
|
setFieldErrors(undefined);
|
||||||
|
await validate(createSettingsValidator(data), data);
|
||||||
|
} catch (error) {
|
||||||
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
|
} finally {
|
||||||
|
await saveData();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const changeBoardProfile = (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
const boardProfile = event.target.value;
|
const boardProfile = event.target.value;
|
||||||
updateFormValue(event);
|
updateFormValue(event);
|
||||||
if (boardProfile === 'CUSTOM') {
|
if (boardProfile === 'CUSTOM') {
|
||||||
@@ -172,22 +160,12 @@ const ApplicationSettings = () => {
|
|||||||
} else {
|
} else {
|
||||||
void updateBoardProfile(boardProfile);
|
void updateBoardProfile(boardProfile);
|
||||||
}
|
}
|
||||||
},
|
};
|
||||||
[data, updateBoardProfile, updateFormValue, updateDataValue]
|
|
||||||
);
|
|
||||||
|
|
||||||
const restart = useCallback(async () => {
|
const restart = async () => {
|
||||||
await validateAndSubmit();
|
await validateAndSubmit();
|
||||||
await doRestart();
|
await doRestart();
|
||||||
}, [validateAndSubmit, doRestart]);
|
};
|
||||||
|
|
||||||
// Memoize board profile select items to prevent recreation
|
|
||||||
const boardProfileItems = useMemo(() => boardProfileSelectItems(), []);
|
|
||||||
|
|
||||||
const content = () => {
|
|
||||||
if (!data || !hardwareData) {
|
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -241,7 +219,7 @@ const ApplicationSettings = () => {
|
|||||||
<Grid container spacing={2} rowSpacing={0}>
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="modbus_max_clients"
|
name="modbus_max_clients"
|
||||||
label={LL.AP_MAX_CLIENTS()}
|
label={LL.AP_MAX_CLIENTS()}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
@@ -253,7 +231,7 @@ const ApplicationSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="modbus_port"
|
name="modbus_port"
|
||||||
label="Port"
|
label="Port"
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
@@ -265,7 +243,7 @@ const ApplicationSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="modbus_timeout"
|
name="modbus_timeout"
|
||||||
label="Timeout"
|
label="Timeout"
|
||||||
slotProps={{
|
slotProps={{
|
||||||
@@ -295,7 +273,7 @@ const ApplicationSettings = () => {
|
|||||||
<Grid container spacing={2} rowSpacing={0}>
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="syslog_host"
|
name="syslog_host"
|
||||||
label="Host"
|
label="Host"
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
@@ -306,7 +284,7 @@ const ApplicationSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="syslog_port"
|
name="syslog_port"
|
||||||
label="Port"
|
label="Port"
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
@@ -329,15 +307,15 @@ const ApplicationSettings = () => {
|
|||||||
>
|
>
|
||||||
<MenuItem value={-1}>OFF</MenuItem>
|
<MenuItem value={-1}>OFF</MenuItem>
|
||||||
<MenuItem value={3}>ERR</MenuItem>
|
<MenuItem value={3}>ERR</MenuItem>
|
||||||
<MenuItem value={4}>WARN</MenuItem>
|
|
||||||
<MenuItem value={5}>NOTICE</MenuItem>
|
<MenuItem value={5}>NOTICE</MenuItem>
|
||||||
<MenuItem value={6}>INFO</MenuItem>
|
<MenuItem value={6}>INFO</MenuItem>
|
||||||
|
<MenuItem value={7}>DEBUG</MenuItem>
|
||||||
<MenuItem value={9}>ALL</MenuItem>
|
<MenuItem value={9}>ALL</MenuItem>
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="syslog_mark_interval"
|
name="syslog_mark_interval"
|
||||||
label={LL.MARK_INTERVAL()}
|
label={LL.MARK_INTERVAL()}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
@@ -352,156 +330,6 @@ const ApplicationSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
<Typography color="secondary">eMail</Typography>
|
|
||||||
<BlockFormControlLabel
|
|
||||||
control={
|
|
||||||
<Checkbox
|
|
||||||
checked={data.email_enabled}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
name="email_enabled"
|
|
||||||
disabled={!hardwareData.psram}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
label={
|
|
||||||
<Typography color={!hardwareData.psram ? 'grey' : 'default'}>
|
|
||||||
Enable eMail notification
|
|
||||||
{!hardwareData.psram && (
|
|
||||||
<Typography variant="caption">
|
|
||||||
({LL.IS_REQUIRED('PSRAM')})
|
|
||||||
</Typography>
|
|
||||||
)}
|
|
||||||
</Typography>
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
{data.email_enabled && (
|
|
||||||
<>
|
|
||||||
<Grid
|
|
||||||
container
|
|
||||||
spacing={2}
|
|
||||||
direction="row"
|
|
||||||
justifyContent="flex-start"
|
|
||||||
alignItems="flex-start"
|
|
||||||
>
|
|
||||||
<Grid>
|
|
||||||
<ValidatedTextField
|
|
||||||
fieldErrors={fieldErrors || {}}
|
|
||||||
name="email_server"
|
|
||||||
label="SMTP Server"
|
|
||||||
variant="outlined"
|
|
||||||
value={data.email_server}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
|
||||||
<ValidatedTextField
|
|
||||||
fieldErrors={fieldErrors || {}}
|
|
||||||
sx={{ width: '12ch' }}
|
|
||||||
name="email_port"
|
|
||||||
variant="outlined"
|
|
||||||
label="Port"
|
|
||||||
value={numberValue(data.email_port)}
|
|
||||||
type="number"
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
<Grid size={4} mt={!data.email_ssl && !data.email_starttls ? 0 : 3}>
|
|
||||||
{!data.email_starttls && (
|
|
||||||
<BlockFormControlLabel
|
|
||||||
sx={{ width: '12ch' }}
|
|
||||||
control={
|
|
||||||
<Checkbox
|
|
||||||
checked={data.email_ssl}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
name="email_ssl"
|
|
||||||
disabled={
|
|
||||||
data.email_starttls || data.email_ssl === undefined
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
label="SSL/TLS"
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
{!data.email_ssl && (
|
|
||||||
<BlockFormControlLabel
|
|
||||||
sx={{ width: '12ch' }}
|
|
||||||
control={
|
|
||||||
<Checkbox
|
|
||||||
checked={data.email_starttls}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
name="email_starttls"
|
|
||||||
disabled={
|
|
||||||
data.email_ssl || data.email_starttls === undefined
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
label="STARTTLS"
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</Grid>
|
|
||||||
</Grid>
|
|
||||||
<Grid container spacing={2} rowSpacing={0}>
|
|
||||||
<Grid>
|
|
||||||
<ValidatedTextField
|
|
||||||
fieldErrors={fieldErrors || {}}
|
|
||||||
name="email_login"
|
|
||||||
label="Login"
|
|
||||||
variant="outlined"
|
|
||||||
value={data.email_login}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
|
||||||
<ValidatedPasswordField
|
|
||||||
fieldErrors={fieldErrors || {}}
|
|
||||||
name="email_pass"
|
|
||||||
label="Password"
|
|
||||||
variant="outlined"
|
|
||||||
value={data.email_pass}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
</Grid>
|
|
||||||
<Grid container spacing={2} rowSpacing={0}>
|
|
||||||
<Grid>
|
|
||||||
<ValidatedTextField
|
|
||||||
fieldErrors={fieldErrors || {}}
|
|
||||||
name="email_sender"
|
|
||||||
label="From"
|
|
||||||
variant="outlined"
|
|
||||||
value={data.email_sender}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
|
||||||
<ValidatedTextField
|
|
||||||
fieldErrors={fieldErrors || {}}
|
|
||||||
name="email_recp"
|
|
||||||
label="To"
|
|
||||||
variant="outlined"
|
|
||||||
value={data.email_recp}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
|
||||||
<ValidatedTextField
|
|
||||||
fieldErrors={fieldErrors || {}}
|
|
||||||
name="email_subject"
|
|
||||||
label="Subject"
|
|
||||||
variant="outlined"
|
|
||||||
value={data.email_subject}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
</Grid>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
<Typography sx={{ pb: 1, pt: 2 }} variant="h6" color="primary">
|
<Typography sx={{ pb: 1, pt: 2 }} variant="h6" color="primary">
|
||||||
{LL.SENSORS()}
|
{LL.SENSORS()}
|
||||||
</Typography>
|
</Typography>
|
||||||
@@ -640,32 +468,24 @@ const ApplicationSettings = () => {
|
|||||||
name="board_profile"
|
name="board_profile"
|
||||||
label={LL.BOARD_PROFILE()}
|
label={LL.BOARD_PROFILE()}
|
||||||
value={data.board_profile}
|
value={data.board_profile}
|
||||||
disabled={processingBoard}
|
disabled={processingBoard || hardwareData.model.startsWith('BBQKees')}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onChange={changeBoardProfile}
|
onChange={changeBoardProfile}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
select
|
select
|
||||||
>
|
>
|
||||||
{hardwareData.model.startsWith('BBQKees') ? (
|
{boardProfileSelectItems()}
|
||||||
<MenuItem key={hardwareData.board} value={hardwareData.board}>
|
<Divider />
|
||||||
{BOARD_PROFILES[hardwareData.board as BoardProfileKey]}
|
<MenuItem key={'CUSTOM'} value={'CUSTOM'}>
|
||||||
</MenuItem>
|
{LL.CUSTOM()}…
|
||||||
) : (
|
</MenuItem>
|
||||||
boardProfileItems
|
|
||||||
)}
|
|
||||||
{(data.board_profile === 'CUSTOM' || data.developer_mode) && <Divider />}
|
|
||||||
{(data.board_profile === 'CUSTOM' || data.developer_mode) && (
|
|
||||||
<MenuItem key={'CUSTOM'} value={'CUSTOM'}>
|
|
||||||
{LL.CUSTOM()}…
|
|
||||||
</MenuItem>
|
|
||||||
)}
|
|
||||||
</TextField>
|
</TextField>
|
||||||
{data.board_profile === 'CUSTOM' && (
|
{data.board_profile === 'CUSTOM' && (
|
||||||
<>
|
<>
|
||||||
<Grid container spacing={2} rowSpacing={0}>
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="rx_gpio"
|
name="rx_gpio"
|
||||||
label={LL.GPIO_OF('Rx')}
|
label={LL.GPIO_OF('Rx')}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -678,7 +498,7 @@ const ApplicationSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="tx_gpio"
|
name="tx_gpio"
|
||||||
label={LL.GPIO_OF('Tx')}
|
label={LL.GPIO_OF('Tx')}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -691,7 +511,7 @@ const ApplicationSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="pbutton_gpio"
|
name="pbutton_gpio"
|
||||||
label={LL.GPIO_OF(LL.BUTTON())}
|
label={LL.GPIO_OF(LL.BUTTON())}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -704,7 +524,7 @@ const ApplicationSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="dallas_gpio"
|
name="dallas_gpio"
|
||||||
label={
|
label={
|
||||||
LL.GPIO_OF(LL.TEMPERATURE()) + ' (0=' + LL.DISABLED(1) + ')'
|
LL.GPIO_OF(LL.TEMPERATURE()) + ' (0=' + LL.DISABLED(1) + ')'
|
||||||
@@ -719,7 +539,7 @@ const ApplicationSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="led_gpio"
|
name="led_gpio"
|
||||||
label={LL.GPIO_OF('LED') + ' (0=' + LL.DISABLED(1) + ')'}
|
label={LL.GPIO_OF('LED') + ' (0=' + LL.DISABLED(1) + ')'}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -734,7 +554,7 @@ const ApplicationSettings = () => {
|
|||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<TextField
|
||||||
name="led_type"
|
name="led_type"
|
||||||
label={'LED ' + LL.TYPE(0)}
|
label={'LED ' + LL.TYPE()}
|
||||||
value={data.led_type}
|
value={data.led_type}
|
||||||
fullWidth
|
fullWidth
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
@@ -761,7 +581,6 @@ const ApplicationSettings = () => {
|
|||||||
<MenuItem value={0}>{LL.DISABLED(1)}</MenuItem>
|
<MenuItem value={0}>{LL.DISABLED(1)}</MenuItem>
|
||||||
<MenuItem value={1}>LAN8720</MenuItem>
|
<MenuItem value={1}>LAN8720</MenuItem>
|
||||||
<MenuItem value={2}>TLK110</MenuItem>
|
<MenuItem value={2}>TLK110</MenuItem>
|
||||||
<MenuItem value={3}>RTL8201</MenuItem>
|
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
@@ -924,7 +743,7 @@ const ApplicationSettings = () => {
|
|||||||
{data.remote_timeout_en && (
|
{data.remote_timeout_en && (
|
||||||
<Box mt={2}>
|
<Box mt={2}>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="remote_timeout"
|
name="remote_timeout"
|
||||||
label={LL.REMOTE_TIMEOUT()}
|
label={LL.REMOTE_TIMEOUT()}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
@@ -964,7 +783,7 @@ const ApplicationSettings = () => {
|
|||||||
{data.shower_timer && (
|
{data.shower_timer && (
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="shower_min_duration"
|
name="shower_min_duration"
|
||||||
label={LL.MIN_DURATION()}
|
label={LL.MIN_DURATION()}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
@@ -982,7 +801,7 @@ const ApplicationSettings = () => {
|
|||||||
<>
|
<>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="shower_alert_trigger"
|
name="shower_alert_trigger"
|
||||||
label={LL.TRIGGER_TIME()}
|
label={LL.TRIGGER_TIME()}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
@@ -998,7 +817,7 @@ const ApplicationSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="shower_alert_coldshot"
|
name="shower_alert_coldshot"
|
||||||
label={LL.COLD_SHOT_DURATION()}
|
label={LL.COLD_SHOT_DURATION()}
|
||||||
slotProps={{
|
slotProps={{
|
||||||
@@ -1017,9 +836,8 @@ const ApplicationSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
{restartNeeded && (
|
{restartNeeded && (
|
||||||
<MessageBox level="warning" message={LL.RESTART_TEXT(0)}>
|
<MessageBox my={2} level="warning" message={LL.RESTART_TEXT(0)}>
|
||||||
<Button
|
<Button
|
||||||
sx={{ ml: 2 }}
|
|
||||||
startIcon={<PowerSettingsNewIcon />}
|
startIcon={<PowerSettingsNewIcon />}
|
||||||
variant="contained"
|
variant="contained"
|
||||||
color="error"
|
color="error"
|
||||||
@@ -1055,12 +873,10 @@ const ApplicationSettings = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
return restarting ? (
|
return (
|
||||||
<SystemMonitor />
|
|
||||||
) : (
|
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
{content()}
|
{restarting ? <SystemMonitor /> : content()}
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import { useCallback, useMemo, useState } from 'react';
|
import { useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import DownloadIcon from '@mui/icons-material/GetApp';
|
import DownloadIcon from '@mui/icons-material/GetApp';
|
||||||
import { Box, Button, Grid, Typography } from '@mui/material';
|
import { Box, Button, Grid2 as Grid, Typography } from '@mui/material';
|
||||||
|
|
||||||
import * as SystemApi from 'api/system';
|
import * as SystemApi from 'api/system';
|
||||||
import { API, callAction } from 'api/app';
|
import { API, callAction } from 'api/app';
|
||||||
@@ -19,12 +19,6 @@ import {
|
|||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
import { saveFile } from 'utils';
|
import { saveFile } from 'utils';
|
||||||
|
|
||||||
interface DownloadButton {
|
|
||||||
type: string;
|
|
||||||
label: string | number;
|
|
||||||
isGridButton: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
const DownloadUpload = () => {
|
const DownloadUpload = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
@@ -41,7 +35,7 @@ const DownloadUpload = () => {
|
|||||||
toast.info(LL.DOWNLOAD_SUCCESSFUL());
|
toast.info(LL.DOWNLOAD_SUCCESSFUL());
|
||||||
})
|
})
|
||||||
.onError((error) => {
|
.onError((error) => {
|
||||||
toast.error(String(error.error?.message || 'An error occurred'));
|
toast.error(error.message);
|
||||||
});
|
});
|
||||||
|
|
||||||
const { send: sendAPI } = useRequest((data: APIcall) => API(data), {
|
const { send: sendAPI } = useRequest((data: APIcall) => API(data), {
|
||||||
@@ -50,126 +44,95 @@ const DownloadUpload = () => {
|
|||||||
|
|
||||||
const { data, send: loadData, error } = useRequest(SystemApi.readSystemStatus);
|
const { data, send: loadData, error } = useRequest(SystemApi.readSystemStatus);
|
||||||
|
|
||||||
const doRestart = useCallback(async () => {
|
const doRestart = async () => {
|
||||||
setRestarting(true);
|
setRestarting(true);
|
||||||
try {
|
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
||||||
await sendAPI({ device: 'system', cmd: 'restart', id: 0 });
|
(error: Error) => {
|
||||||
} catch (error) {
|
toast.error(error.message);
|
||||||
toast.error((error as Error).message);
|
}
|
||||||
setRestarting(false);
|
);
|
||||||
}
|
};
|
||||||
}, [sendAPI]);
|
|
||||||
|
|
||||||
useLayoutTitle(LL.DOWNLOAD_UPLOAD());
|
useLayoutTitle(LL.DOWNLOAD_UPLOAD());
|
||||||
|
|
||||||
const downloadButtons: DownloadButton[] = useMemo(
|
const content = () => {
|
||||||
() => [
|
if (!data) {
|
||||||
{
|
return <FormLoader onRetry={loadData} errorMessage={error?.message} />;
|
||||||
type: 'settings',
|
}
|
||||||
label: LL.SETTINGS_OF(LL.APPLICATION()),
|
|
||||||
isGridButton: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'customizations',
|
|
||||||
label: LL.CUSTOMIZATIONS(),
|
|
||||||
isGridButton: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'entities',
|
|
||||||
label: LL.CUSTOM_ENTITIES(0),
|
|
||||||
isGridButton: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'schedule',
|
|
||||||
label: LL.SCHEDULE(0),
|
|
||||||
isGridButton: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'systembackup',
|
|
||||||
label: LL.DOWNLOAD_SYSTEM_BACKUP(),
|
|
||||||
isGridButton: true
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: 'allvalues',
|
|
||||||
label: LL.ALLVALUES(),
|
|
||||||
isGridButton: false
|
|
||||||
}
|
|
||||||
],
|
|
||||||
[LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const handleDownload = useCallback(
|
|
||||||
(type: string) => () => {
|
|
||||||
void sendExportData(type);
|
|
||||||
},
|
|
||||||
[sendExportData]
|
|
||||||
);
|
|
||||||
|
|
||||||
if (restarting) {
|
|
||||||
return <SystemMonitor />;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!data) {
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<>
|
||||||
<FormLoader onRetry={loadData} errorMessage={error?.message || ''} />
|
<Typography sx={{ pb: 2 }} variant="h6" color="primary">
|
||||||
</SectionContent>
|
{LL.DOWNLOAD(0)}
|
||||||
);
|
</Typography>
|
||||||
}
|
|
||||||
|
|
||||||
const gridButtons = downloadButtons.filter((btn) => btn.isGridButton);
|
<Typography mb={1} variant="body1" color="warning">
|
||||||
const standaloneButton = downloadButtons.find((btn) => !btn.isGridButton);
|
{LL.DOWNLOAD_SETTINGS_TEXT()}.
|
||||||
|
</Typography>
|
||||||
|
<Grid container spacing={1}>
|
||||||
|
<Button
|
||||||
|
sx={{ ml: 2 }}
|
||||||
|
startIcon={<DownloadIcon />}
|
||||||
|
variant="outlined"
|
||||||
|
color="primary"
|
||||||
|
onClick={() => sendExportData('settings')}
|
||||||
|
>
|
||||||
|
{LL.SETTINGS_OF(LL.APPLICATION())}
|
||||||
|
</Button>
|
||||||
|
|
||||||
return (
|
<Button
|
||||||
<SectionContent>
|
sx={{ ml: 2 }}
|
||||||
<Typography sx={{ pb: 2 }} variant="h6" color="primary">
|
startIcon={<DownloadIcon />}
|
||||||
{LL.DOWNLOAD(0)}
|
variant="outlined"
|
||||||
</Typography>
|
color="primary"
|
||||||
|
onClick={() => sendExportData('customizations')}
|
||||||
<Typography mb={1} variant="body1" color="warning">
|
>
|
||||||
{LL.DOWNLOAD_SETTINGS_TEXT()}.
|
{LL.CUSTOMIZATIONS()}
|
||||||
</Typography>
|
</Button>
|
||||||
|
<Button
|
||||||
<Grid container spacing={2}>
|
sx={{ ml: 2 }}
|
||||||
{gridButtons.map((button) => (
|
startIcon={<DownloadIcon />}
|
||||||
<Grid key={button.type}>
|
variant="outlined"
|
||||||
<Button
|
color="primary"
|
||||||
startIcon={<DownloadIcon />}
|
onClick={() => sendExportData('entities')}
|
||||||
variant="outlined"
|
>
|
||||||
color="primary"
|
{LL.CUSTOM_ENTITIES(0)}
|
||||||
onClick={handleDownload(button.type)}
|
</Button>
|
||||||
>
|
<Button
|
||||||
{button.label}
|
sx={{ ml: 2 }}
|
||||||
</Button>
|
startIcon={<DownloadIcon />}
|
||||||
</Grid>
|
variant="outlined"
|
||||||
))}
|
color="primary"
|
||||||
</Grid>
|
onClick={() => sendExportData('schedule')}
|
||||||
|
>
|
||||||
<Typography mt={2} mb={1} variant="body1" color="warning">
|
{LL.SCHEDULE(0)}
|
||||||
{LL.DOWNLOAD_SETTINGS_TEXT2()}.
|
</Button>
|
||||||
</Typography>
|
</Grid>
|
||||||
|
|
||||||
{standaloneButton && (
|
|
||||||
<Button
|
<Button
|
||||||
|
sx={{ ml: 2, mt: 2 }}
|
||||||
startIcon={<DownloadIcon />}
|
startIcon={<DownloadIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="primary"
|
color="primary"
|
||||||
onClick={handleDownload(standaloneButton.type)}
|
onClick={() => sendExportData('allvalues')}
|
||||||
>
|
>
|
||||||
{standaloneButton.label}
|
{LL.ALLVALUES()}
|
||||||
</Button>
|
</Button>
|
||||||
)}
|
|
||||||
|
|
||||||
<Typography sx={{ pt: 2, pb: 2 }} variant="h6" color="primary">
|
<Typography sx={{ pt: 2, pb: 2 }} variant="h6" color="primary">
|
||||||
{LL.UPLOAD()}
|
{LL.UPLOAD()}
|
||||||
</Typography>
|
</Typography>
|
||||||
|
|
||||||
<Box color="warning.main" sx={{ pb: 2 }}>
|
<Box color="warning.main" sx={{ pb: 2 }}>
|
||||||
<Typography variant="body1">{LL.UPLOAD_TEXT()}.</Typography>
|
<Typography variant="body1">{LL.UPLOAD_TEXT()}.</Typography>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
<SingleUpload text={LL.UPLOAD_DRAG()} doRestart={doRestart} />
|
<SingleUpload text={LL.UPLOAD_DRAG()} doRestart={doRestart} />
|
||||||
</SectionContent>
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SectionContent>{restarting ? <SystemMonitor /> : content()}</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -1,14 +1,11 @@
|
|||||||
import { useCallback, useMemo, useState } from 'react';
|
import { useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import SettingsBackupRestoreIcon from '@mui/icons-material/SettingsBackupRestore';
|
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
import {
|
import {
|
||||||
Box,
|
|
||||||
Button,
|
Button,
|
||||||
Checkbox,
|
Checkbox,
|
||||||
Grid,
|
Grid2 as Grid,
|
||||||
InputAdornment,
|
InputAdornment,
|
||||||
MenuItem,
|
MenuItem,
|
||||||
TextField,
|
TextField,
|
||||||
@@ -33,8 +30,6 @@ import type { MqttSettingsType } from 'types';
|
|||||||
import { numberValue, updateValueDirty, useRest } from 'utils';
|
import { numberValue, updateValueDirty, useRest } from 'utils';
|
||||||
import { createMqttSettingsValidator, validate } from 'validators';
|
import { createMqttSettingsValidator, validate } from 'validators';
|
||||||
|
|
||||||
import { callAction } from '../../api/app';
|
|
||||||
|
|
||||||
const MqttSettings = () => {
|
const MqttSettings = () => {
|
||||||
const {
|
const {
|
||||||
loadData,
|
loadData,
|
||||||
@@ -57,104 +52,48 @@ const MqttSettings = () => {
|
|||||||
|
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
|
|
||||||
const sendResetMQTT = useCallback(() => {
|
const updateFormValue = updateValueDirty(
|
||||||
void callAction({ action: 'resetMQTT' })
|
origData,
|
||||||
.then(() => {
|
dirtyFlags,
|
||||||
toast.success('MQTT ' + LL.REFRESH() + ' successful');
|
setDirtyFlags,
|
||||||
})
|
updateDataValue
|
||||||
.catch((error) => {
|
|
||||||
toast.error(String(error.error?.message || 'An error occurred'));
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const updateFormValue = useMemo(
|
|
||||||
() =>
|
|
||||||
updateValueDirty(
|
|
||||||
origData as unknown as Record<string, unknown>,
|
|
||||||
dirtyFlags,
|
|
||||||
setDirtyFlags,
|
|
||||||
updateDataValue as (value: unknown) => void
|
|
||||||
),
|
|
||||||
[origData, dirtyFlags, setDirtyFlags, updateDataValue]
|
|
||||||
);
|
);
|
||||||
|
|
||||||
const SecondsInputProps = useMemo(
|
const SecondsInputProps = {
|
||||||
() => ({
|
endAdornment: <InputAdornment position="end">{LL.SECONDS()}</InputAdornment>
|
||||||
endAdornment: <InputAdornment position="end">{LL.SECONDS()}</InputAdornment>
|
};
|
||||||
}),
|
|
||||||
[LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const emptyFieldErrors = useMemo(() => ({}), []);
|
const content = () => {
|
||||||
|
if (!data) {
|
||||||
const validateAndSubmit = useCallback(async () => {
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage} />;
|
||||||
if (!data) return;
|
|
||||||
try {
|
|
||||||
setFieldErrors(undefined);
|
|
||||||
await validate(createMqttSettingsValidator(data), data);
|
|
||||||
await saveData();
|
|
||||||
} catch (error) {
|
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
|
||||||
}
|
}
|
||||||
}, [data, saveData]);
|
|
||||||
|
|
||||||
const publishIntervalFields = useMemo(
|
const validateAndSubmit = async () => {
|
||||||
() => [
|
try {
|
||||||
{ name: 'publish_time_heartbeat', label: 'Heartbeat', validated: true },
|
setFieldErrors(undefined);
|
||||||
{ name: 'publish_time_boiler', label: LL.MQTT_INT_BOILER(), validated: false },
|
await validate(createMqttSettingsValidator(data), data);
|
||||||
{
|
await saveData();
|
||||||
name: 'publish_time_thermostat',
|
} catch (error) {
|
||||||
label: LL.MQTT_INT_THERMOSTATS(),
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
validated: false
|
}
|
||||||
},
|
};
|
||||||
{ name: 'publish_time_solar', label: LL.MQTT_INT_SOLAR(), validated: false },
|
|
||||||
{ name: 'publish_time_mixer', label: LL.MQTT_INT_MIXER(), validated: false },
|
|
||||||
{ name: 'publish_time_water', label: LL.MQTT_INT_WATER(), validated: false },
|
|
||||||
{ name: 'publish_time_sensor', label: LL.SENSORS(), validated: false },
|
|
||||||
{ name: 'publish_time_other', label: LL.DEFAULT(0), validated: false }
|
|
||||||
],
|
|
||||||
[LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!data) {
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
|
||||||
<FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />
|
|
||||||
</SectionContent>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<SectionContent>
|
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
|
||||||
<>
|
<>
|
||||||
<Box display="flex" gap={2} mb={1}>
|
<BlockFormControlLabel
|
||||||
<BlockFormControlLabel
|
control={
|
||||||
control={
|
<Checkbox
|
||||||
<Checkbox
|
name="enabled"
|
||||||
name="enabled"
|
checked={data.enabled}
|
||||||
checked={data.enabled}
|
onChange={updateFormValue}
|
||||||
onChange={updateFormValue}
|
/>
|
||||||
/>
|
}
|
||||||
}
|
label={LL.ENABLE_MQTT()}
|
||||||
label={LL.ENABLE_MQTT()}
|
/>
|
||||||
/>
|
|
||||||
{data.enabled && (
|
|
||||||
<Button
|
|
||||||
startIcon={<SettingsBackupRestoreIcon />}
|
|
||||||
color="secondary"
|
|
||||||
variant="outlined"
|
|
||||||
onClick={sendResetMQTT}
|
|
||||||
>
|
|
||||||
{LL.REFRESH() + ' MQTT'}
|
|
||||||
</Button>
|
|
||||||
)}
|
|
||||||
</Box>
|
|
||||||
<Grid container spacing={2} rowSpacing={0}>
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors ?? emptyFieldErrors}
|
fieldErrors={fieldErrors}
|
||||||
name="host"
|
name="host"
|
||||||
label={LL.ADDRESS_OF(LL.BROKER())}
|
label={LL.ADDRESS_OF(LL.BROKER())}
|
||||||
multiline
|
multiline
|
||||||
@@ -166,7 +105,7 @@ const MqttSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors ?? emptyFieldErrors}
|
fieldErrors={fieldErrors}
|
||||||
name="port"
|
name="port"
|
||||||
label="Port"
|
label="Port"
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
@@ -178,7 +117,7 @@ const MqttSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors ?? emptyFieldErrors}
|
fieldErrors={fieldErrors}
|
||||||
name="base"
|
name="base"
|
||||||
label={LL.BASE_TOPIC()}
|
label={LL.BASE_TOPIC()}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
@@ -190,7 +129,7 @@ const MqttSettings = () => {
|
|||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<TextField
|
||||||
name="client_id"
|
name="client_id"
|
||||||
label={`${LL.ID_OF(LL.CLIENT())} (${LL.OPTIONAL()})`}
|
label={LL.ID_OF(LL.CLIENT()) + ' (' + LL.OPTIONAL() + ')'}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
value={data.client_id}
|
value={data.client_id}
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
@@ -219,7 +158,7 @@ const MqttSettings = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
<Grid>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors ?? emptyFieldErrors}
|
fieldErrors={fieldErrors}
|
||||||
name="keep_alive"
|
name="keep_alive"
|
||||||
label="Keep Alive"
|
label="Keep Alive"
|
||||||
slotProps={{
|
slotProps={{
|
||||||
@@ -266,7 +205,6 @@ const MqttSettings = () => {
|
|||||||
label={LL.CERT()}
|
label={LL.CERT()}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
value={data.rootCA}
|
value={data.rootCA}
|
||||||
sx={{ width: '50ch' }}
|
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
@@ -316,160 +254,219 @@ const MqttSettings = () => {
|
|||||||
}
|
}
|
||||||
label={LL.MQTT_RESPONSE()}
|
label={LL.MQTT_RESPONSE()}
|
||||||
/>
|
/>
|
||||||
<Grid container spacing={2} rowSpacing={0}>
|
{!data.ha_enabled && (
|
||||||
<Grid>
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
<BlockFormControlLabel
|
|
||||||
control={
|
|
||||||
<Checkbox
|
|
||||||
name="publish_single"
|
|
||||||
checked={data.publish_single}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
disabled={data.ha_enabled}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
label={LL.MQTT_PUBLISH_TEXT_1()}
|
|
||||||
/>
|
|
||||||
</Grid>
|
|
||||||
{data.publish_single && (
|
|
||||||
<Grid>
|
<Grid>
|
||||||
<BlockFormControlLabel
|
<BlockFormControlLabel
|
||||||
control={
|
control={
|
||||||
<Checkbox
|
<Checkbox
|
||||||
name="publish_single2cmd"
|
name="publish_single"
|
||||||
checked={data.publish_single2cmd}
|
checked={data.publish_single}
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
/>
|
/>
|
||||||
}
|
}
|
||||||
label={LL.MQTT_PUBLISH_TEXT_2()}
|
label={LL.MQTT_PUBLISH_TEXT_1()}
|
||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
{data.publish_single && (
|
||||||
</Grid>
|
<Grid>
|
||||||
{/* <Grid container spacing={2} rowSpacing={0}> */}
|
<BlockFormControlLabel
|
||||||
<Grid>
|
control={
|
||||||
<BlockFormControlLabel
|
<Checkbox
|
||||||
control={
|
name="publish_single2cmd"
|
||||||
<Checkbox
|
checked={data.publish_single2cmd}
|
||||||
name="ha_enabled"
|
onChange={updateFormValue}
|
||||||
checked={data.ha_enabled}
|
/>
|
||||||
onChange={updateFormValue}
|
}
|
||||||
disabled={data.publish_single}
|
label={LL.MQTT_PUBLISH_TEXT_2()}
|
||||||
/>
|
/>
|
||||||
}
|
</Grid>
|
||||||
label={LL.MQTT_PUBLISH_TEXT_3()}
|
)}
|
||||||
/>
|
</Grid>
|
||||||
</Grid>
|
)}
|
||||||
{data.ha_enabled && (
|
{!data.publish_single && (
|
||||||
<Grid container spacing={2} rowSpacing={0}>
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
<Grid>
|
<Grid>
|
||||||
<TextField
|
<BlockFormControlLabel
|
||||||
name="discovery_type"
|
control={
|
||||||
label={LL.MQTT_PUBLISH_TEXT_5()}
|
<Checkbox
|
||||||
value={data.discovery_type}
|
name="ha_enabled"
|
||||||
variant="outlined"
|
checked={data.ha_enabled}
|
||||||
onChange={updateFormValue}
|
onChange={updateFormValue}
|
||||||
margin="normal"
|
/>
|
||||||
select
|
}
|
||||||
>
|
label={LL.MQTT_PUBLISH_TEXT_3()}
|
||||||
<MenuItem value={0}>Home Assistant</MenuItem>
|
|
||||||
<MenuItem value={1}>Domoticz</MenuItem>
|
|
||||||
<MenuItem value={2}>Domoticz (latest)</MenuItem>
|
|
||||||
</TextField>
|
|
||||||
</Grid>
|
|
||||||
<Grid>
|
|
||||||
<TextField
|
|
||||||
name="discovery_prefix"
|
|
||||||
label={LL.MQTT_PUBLISH_TEXT_4()}
|
|
||||||
variant="outlined"
|
|
||||||
value={data.discovery_prefix}
|
|
||||||
onChange={updateFormValue}
|
|
||||||
margin="normal"
|
|
||||||
/>
|
/>
|
||||||
</Grid>
|
</Grid>
|
||||||
<Grid>
|
{data.ha_enabled && (
|
||||||
<TextField
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
name="entity_format"
|
<Grid>
|
||||||
label={LL.MQTT_ENTITY_FORMAT()}
|
<TextField
|
||||||
value={data.entity_format}
|
name="discovery_type"
|
||||||
variant="outlined"
|
label={LL.MQTT_PUBLISH_TEXT_5()}
|
||||||
onChange={updateFormValue}
|
value={data.discovery_type}
|
||||||
margin="normal"
|
variant="outlined"
|
||||||
select
|
onChange={updateFormValue}
|
||||||
>
|
margin="normal"
|
||||||
<MenuItem value={0}>{LL.MQTT_ENTITY_FORMAT_0()}</MenuItem>
|
select
|
||||||
<MenuItem value={3}>
|
>
|
||||||
{LL.MQTT_ENTITY_FORMAT_1()} (v3.5)
|
<MenuItem value={0}>Home Assistant</MenuItem>
|
||||||
</MenuItem>
|
<MenuItem value={1}>Domoticz</MenuItem>
|
||||||
<MenuItem value={4}>
|
<MenuItem value={2}>Domoticz (latest)</MenuItem>
|
||||||
{LL.MQTT_ENTITY_FORMAT_2()} (v3.5)
|
</TextField>
|
||||||
</MenuItem>
|
</Grid>
|
||||||
<MenuItem value={1}>
|
<Grid>
|
||||||
{LL.MQTT_ENTITY_FORMAT_1()} (latest)
|
<TextField
|
||||||
</MenuItem>
|
name="discovery_prefix"
|
||||||
<MenuItem value={2}>
|
label={LL.MQTT_PUBLISH_TEXT_4()}
|
||||||
{LL.MQTT_ENTITY_FORMAT_2()} (latest)
|
variant="outlined"
|
||||||
</MenuItem>
|
value={data.discovery_prefix}
|
||||||
</TextField>
|
onChange={updateFormValue}
|
||||||
</Grid>
|
margin="normal"
|
||||||
<Grid>
|
/>
|
||||||
{data.discovery_type === 0 && (
|
</Grid>
|
||||||
<TextField
|
<Grid>
|
||||||
name="ha_number_mode"
|
<TextField
|
||||||
label={LL.MQTT_INPUT_NUMBER_FORMAT()}
|
name="entity_format"
|
||||||
value={data.ha_number_mode}
|
label={LL.MQTT_ENTITY_FORMAT()}
|
||||||
variant="outlined"
|
value={data.entity_format}
|
||||||
onChange={updateFormValue}
|
variant="outlined"
|
||||||
sx={{ width: '20ch' }}
|
onChange={updateFormValue}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
select
|
select
|
||||||
>
|
>
|
||||||
<MenuItem value={0}>Box</MenuItem>
|
<MenuItem value={0}>{LL.MQTT_ENTITY_FORMAT_0()}</MenuItem>
|
||||||
<MenuItem value={1}>Slider</MenuItem>
|
<MenuItem value={3}>
|
||||||
</TextField>
|
{LL.MQTT_ENTITY_FORMAT_1()} (v3.6)
|
||||||
)}
|
</MenuItem>
|
||||||
</Grid>
|
<MenuItem value={4}>
|
||||||
|
{LL.MQTT_ENTITY_FORMAT_2()} (v3.6)
|
||||||
|
</MenuItem>
|
||||||
|
<MenuItem value={1}>{LL.MQTT_ENTITY_FORMAT_1()}</MenuItem>
|
||||||
|
<MenuItem value={2}>{LL.MQTT_ENTITY_FORMAT_2()}</MenuItem>
|
||||||
|
</TextField>
|
||||||
|
</Grid>
|
||||||
|
</Grid>
|
||||||
|
)}
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
<Typography sx={{ pt: 2 }} variant="h6" color="primary">
|
<Typography sx={{ pt: 2 }} variant="h6" color="primary">
|
||||||
{LL.MQTT_PUBLISH_INTERVALS()} (0=auto)
|
{LL.MQTT_PUBLISH_INTERVALS()} (0=auto)
|
||||||
</Typography>
|
</Typography>
|
||||||
<Grid container spacing={2} rowSpacing={0}>
|
<Grid container spacing={2} rowSpacing={0}>
|
||||||
{publishIntervalFields.map((field) => (
|
<Grid>
|
||||||
<Grid key={field.name}>
|
<ValidatedTextField
|
||||||
{field.validated ? (
|
fieldErrors={fieldErrors}
|
||||||
<ValidatedTextField
|
name="publish_time_heartbeat"
|
||||||
fieldErrors={fieldErrors ?? emptyFieldErrors}
|
label="Heartbeat"
|
||||||
name={field.name}
|
slotProps={{
|
||||||
label={field.label}
|
input: SecondsInputProps
|
||||||
slotProps={{
|
}}
|
||||||
input: SecondsInputProps
|
variant="outlined"
|
||||||
}}
|
value={numberValue(data.publish_time_heartbeat)}
|
||||||
variant="outlined"
|
type="number"
|
||||||
value={numberValue(
|
onChange={updateFormValue}
|
||||||
data[field.name as keyof MqttSettingsType] as number
|
margin="normal"
|
||||||
)}
|
/>
|
||||||
type="number"
|
</Grid>
|
||||||
onChange={updateFormValue}
|
<Grid>
|
||||||
margin="normal"
|
<TextField
|
||||||
/>
|
name="publish_time_boiler"
|
||||||
) : (
|
label={LL.MQTT_INT_BOILER()}
|
||||||
<TextField
|
variant="outlined"
|
||||||
name={field.name}
|
value={numberValue(data.publish_time_boiler)}
|
||||||
label={field.label}
|
type="number"
|
||||||
variant="outlined"
|
onChange={updateFormValue}
|
||||||
value={numberValue(
|
margin="normal"
|
||||||
data[field.name as keyof MqttSettingsType] as number
|
slotProps={{
|
||||||
)}
|
input: SecondsInputProps
|
||||||
type="number"
|
}}
|
||||||
onChange={updateFormValue}
|
/>
|
||||||
margin="normal"
|
</Grid>
|
||||||
slotProps={{
|
<Grid>
|
||||||
input: SecondsInputProps
|
<TextField
|
||||||
}}
|
name="publish_time_thermostat"
|
||||||
/>
|
label={LL.MQTT_INT_THERMOSTATS()}
|
||||||
)}
|
variant="outlined"
|
||||||
</Grid>
|
value={numberValue(data.publish_time_thermostat)}
|
||||||
))}
|
type="number"
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
slotProps={{
|
||||||
|
input: SecondsInputProps
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
<Grid>
|
||||||
|
<TextField
|
||||||
|
name="publish_time_solar"
|
||||||
|
label={LL.MQTT_INT_SOLAR()}
|
||||||
|
variant="outlined"
|
||||||
|
value={numberValue(data.publish_time_solar)}
|
||||||
|
type="number"
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
slotProps={{
|
||||||
|
input: SecondsInputProps
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
<Grid>
|
||||||
|
<TextField
|
||||||
|
name="publish_time_mixer"
|
||||||
|
label={LL.MQTT_INT_MIXER()}
|
||||||
|
variant="outlined"
|
||||||
|
value={numberValue(data.publish_time_mixer)}
|
||||||
|
type="number"
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
slotProps={{
|
||||||
|
input: SecondsInputProps
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
<Grid>
|
||||||
|
<TextField
|
||||||
|
name="publish_time_water"
|
||||||
|
label={LL.MQTT_INT_WATER()}
|
||||||
|
variant="outlined"
|
||||||
|
value={numberValue(data.publish_time_water)}
|
||||||
|
type="number"
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
slotProps={{
|
||||||
|
input: SecondsInputProps
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
<Grid>
|
||||||
|
<TextField
|
||||||
|
name="publish_time_sensor"
|
||||||
|
label={LL.TEMP_SENSORS()}
|
||||||
|
variant="outlined"
|
||||||
|
value={numberValue(data.publish_time_sensor)}
|
||||||
|
type="number"
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
slotProps={{
|
||||||
|
input: SecondsInputProps
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
|
<Grid>
|
||||||
|
<TextField
|
||||||
|
name="publish_time_other"
|
||||||
|
label={LL.DEFAULT(0)}
|
||||||
|
variant="outlined"
|
||||||
|
value={numberValue(data.publish_time_other)}
|
||||||
|
type="number"
|
||||||
|
onChange={updateFormValue}
|
||||||
|
margin="normal"
|
||||||
|
slotProps={{
|
||||||
|
input: SecondsInputProps
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</Grid>
|
||||||
</Grid>
|
</Grid>
|
||||||
{dirtyFlags && dirtyFlags.length !== 0 && (
|
{dirtyFlags && dirtyFlags.length !== 0 && (
|
||||||
<ButtonRow>
|
<ButtonRow>
|
||||||
@@ -496,6 +493,13 @@ const MqttSettings = () => {
|
|||||||
</ButtonRow>
|
</ButtonRow>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SectionContent>
|
||||||
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
|
{content()}
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,27 +1,12 @@
|
|||||||
import { useCallback, useMemo, useState } from 'react';
|
import { useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
|
||||||
|
|
||||||
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
import {
|
import { Button, Checkbox, MenuItem } from '@mui/material';
|
||||||
Box,
|
|
||||||
Button,
|
|
||||||
Checkbox,
|
|
||||||
Dialog,
|
|
||||||
DialogActions,
|
|
||||||
DialogContent,
|
|
||||||
DialogTitle,
|
|
||||||
MenuItem,
|
|
||||||
TextField,
|
|
||||||
Typography
|
|
||||||
} from '@mui/material';
|
|
||||||
|
|
||||||
import * as NTPApi from 'api/ntp';
|
import * as NTPApi from 'api/ntp';
|
||||||
import { readNTPSettings } from 'api/ntp';
|
import { readNTPSettings } from 'api/ntp';
|
||||||
|
|
||||||
import { dialogStyle } from 'CustomTheme';
|
|
||||||
import { useRequest } from 'alova/client';
|
|
||||||
import { updateState } from 'alova/client';
|
import { updateState } from 'alova/client';
|
||||||
import type { ValidateFieldsError } from 'async-validator';
|
import type { ValidateFieldsError } from 'async-validator';
|
||||||
import {
|
import {
|
||||||
@@ -34,12 +19,12 @@ import {
|
|||||||
useLayoutTitle
|
useLayoutTitle
|
||||||
} from 'components';
|
} from 'components';
|
||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
import type { NTPSettingsType, Time } from 'types';
|
import type { NTPSettingsType } from 'types';
|
||||||
import { formatLocalDateTime, updateValueDirty, useRest } from 'utils';
|
import { updateValueDirty, useRest } from 'utils';
|
||||||
import { validate } from 'validators';
|
import { validate } from 'validators';
|
||||||
import { NTP_SETTINGS_VALIDATOR } from 'validators/ntp';
|
import { NTP_SETTINGS_VALIDATOR } from 'validators/ntp';
|
||||||
|
|
||||||
import { TIME_ZONES, selectedTimeZone, useTimeZoneSelectItems } from './TZ';
|
import { TIME_ZONES, selectedTimeZone, timeZoneSelectItems } from './TZ';
|
||||||
|
|
||||||
const NTPSettings = () => {
|
const NTPSettings = () => {
|
||||||
const {
|
const {
|
||||||
@@ -61,100 +46,38 @@ const NTPSettings = () => {
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
useLayoutTitle('NTP');
|
useLayoutTitle('NTP');
|
||||||
|
|
||||||
// Memoized timezone select items for better performance
|
const updateFormValue = updateValueDirty(
|
||||||
const timeZoneItems = useTimeZoneSelectItems();
|
origData,
|
||||||
|
dirtyFlags,
|
||||||
// Memoized selected timezone value
|
setDirtyFlags,
|
||||||
const selectedTzValue = useMemo(
|
updateDataValue
|
||||||
() => (data ? selectedTimeZone(data.tz_label, data.tz_format) : undefined),
|
|
||||||
[data?.tz_label, data?.tz_format]
|
|
||||||
);
|
);
|
||||||
|
|
||||||
const [localTime, setLocalTime] = useState<string>('');
|
|
||||||
const [settingTime, setSettingTime] = useState<boolean>(false);
|
|
||||||
const [processing, setProcessing] = useState<boolean>(false);
|
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
|
|
||||||
const { send: updateTime } = useRequest(
|
const content = () => {
|
||||||
(local_time: Time) => NTPApi.updateTime(local_time),
|
if (!data) {
|
||||||
{
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage} />;
|
||||||
immediate: false
|
|
||||||
}
|
}
|
||||||
);
|
|
||||||
|
|
||||||
// Memoize updateFormValue to prevent recreation on every render
|
const validateAndSubmit = async () => {
|
||||||
const updateFormValue = useMemo(
|
try {
|
||||||
() =>
|
setFieldErrors(undefined);
|
||||||
updateValueDirty(
|
await validate(NTP_SETTINGS_VALIDATOR, data);
|
||||||
origData as unknown as Record<string, unknown>,
|
await saveData();
|
||||||
dirtyFlags,
|
} catch (error) {
|
||||||
setDirtyFlags,
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
updateDataValue as (value: unknown) => void
|
}
|
||||||
),
|
};
|
||||||
[origData, dirtyFlags, setDirtyFlags, updateDataValue]
|
|
||||||
);
|
|
||||||
|
|
||||||
// Memoize updateLocalTime handler
|
const changeTimeZone = (event: React.ChangeEvent<HTMLInputElement>) => {
|
||||||
const updateLocalTime = useCallback(
|
|
||||||
(event: React.ChangeEvent<HTMLInputElement>) => setLocalTime(event.target.value),
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
// Memoize openSetTime handler
|
|
||||||
const openSetTime = useCallback(() => {
|
|
||||||
setLocalTime(formatLocalDateTime(new Date()));
|
|
||||||
setSettingTime(true);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
// Memoize configureTime handler
|
|
||||||
const configureTime = useCallback(async () => {
|
|
||||||
setProcessing(true);
|
|
||||||
|
|
||||||
try {
|
|
||||||
await updateTime({ local_time: formatLocalDateTime(new Date(localTime)) });
|
|
||||||
toast.success(LL.TIME_SET());
|
|
||||||
setSettingTime(false);
|
|
||||||
await loadData();
|
|
||||||
} catch {
|
|
||||||
toast.error(LL.PROBLEM_UPDATING());
|
|
||||||
} finally {
|
|
||||||
setProcessing(false);
|
|
||||||
}
|
|
||||||
}, [localTime, updateTime, LL, loadData]);
|
|
||||||
|
|
||||||
// Memoize close dialog handler
|
|
||||||
const handleCloseSetTime = useCallback(() => setSettingTime(false), []);
|
|
||||||
|
|
||||||
// Memoize validate and submit handler
|
|
||||||
const validateAndSubmit = useCallback(async () => {
|
|
||||||
if (!data) return;
|
|
||||||
try {
|
|
||||||
setFieldErrors(undefined);
|
|
||||||
await validate(NTP_SETTINGS_VALIDATOR, data);
|
|
||||||
await saveData();
|
|
||||||
} catch (error) {
|
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
|
||||||
}
|
|
||||||
}, [data, saveData]);
|
|
||||||
|
|
||||||
// Memoize timezone change handler
|
|
||||||
const changeTimeZone = useCallback(
|
|
||||||
(event: React.ChangeEvent<HTMLInputElement>) => {
|
|
||||||
void updateState(readNTPSettings(), (settings: NTPSettingsType) => ({
|
void updateState(readNTPSettings(), (settings: NTPSettingsType) => ({
|
||||||
...settings,
|
...settings,
|
||||||
tz_label: event.target.value,
|
tz_label: event.target.value,
|
||||||
tz_format: TIME_ZONES[event.target.value]
|
tz_format: TIME_ZONES[event.target.value]
|
||||||
}));
|
}));
|
||||||
updateFormValue(event);
|
updateFormValue(event);
|
||||||
},
|
};
|
||||||
[updateFormValue]
|
|
||||||
);
|
|
||||||
|
|
||||||
// Memoize render content to prevent unnecessary re-renders
|
|
||||||
const renderContent = useMemo(() => {
|
|
||||||
if (!data) {
|
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -169,7 +92,7 @@ const NTPSettings = () => {
|
|||||||
label={LL.ENABLE_NTP()}
|
label={LL.ENABLE_NTP()}
|
||||||
/>
|
/>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="server"
|
name="server"
|
||||||
label={LL.NTP_SERVER()}
|
label={LL.NTP_SERVER()}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -179,37 +102,19 @@ const NTPSettings = () => {
|
|||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="tz_label"
|
name="tz_label"
|
||||||
label={LL.TIME_ZONE()}
|
label={LL.TIME_ZONE()}
|
||||||
fullWidth
|
fullWidth
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
value={selectedTzValue}
|
value={selectedTimeZone(data.tz_label, data.tz_format)}
|
||||||
onChange={changeTimeZone}
|
onChange={changeTimeZone}
|
||||||
margin="normal"
|
margin="normal"
|
||||||
select
|
select
|
||||||
>
|
>
|
||||||
<MenuItem disabled>{LL.TIME_ZONE()}...</MenuItem>
|
<MenuItem disabled>{LL.TIME_ZONE()}...</MenuItem>
|
||||||
{timeZoneItems}
|
{timeZoneSelectItems()}
|
||||||
</ValidatedTextField>
|
</ValidatedTextField>
|
||||||
|
|
||||||
<Box display="flex" flexWrap="wrap">
|
|
||||||
{!data.enabled && !dirtyFlags.length && (
|
|
||||||
<Box flexWrap="nowrap" whiteSpace="nowrap">
|
|
||||||
<ButtonRow>
|
|
||||||
<Button
|
|
||||||
onClick={openSetTime}
|
|
||||||
variant="outlined"
|
|
||||||
color="primary"
|
|
||||||
startIcon={<AccessTimeIcon />}
|
|
||||||
>
|
|
||||||
{LL.SET_TIME(0)}
|
|
||||||
</Button>
|
|
||||||
</ButtonRow>
|
|
||||||
</Box>
|
|
||||||
)}
|
|
||||||
</Box>
|
|
||||||
|
|
||||||
{dirtyFlags && dirtyFlags.length !== 0 && (
|
{dirtyFlags && dirtyFlags.length !== 0 && (
|
||||||
<ButtonRow>
|
<ButtonRow>
|
||||||
<Button
|
<Button
|
||||||
@@ -236,66 +141,12 @@ const NTPSettings = () => {
|
|||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}, [
|
};
|
||||||
data,
|
|
||||||
errorMessage,
|
|
||||||
loadData,
|
|
||||||
updateFormValue,
|
|
||||||
fieldErrors,
|
|
||||||
selectedTzValue,
|
|
||||||
changeTimeZone,
|
|
||||||
timeZoneItems,
|
|
||||||
dirtyFlags,
|
|
||||||
openSetTime,
|
|
||||||
saving,
|
|
||||||
validateAndSubmit,
|
|
||||||
LL
|
|
||||||
]);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
{renderContent}
|
{content()}
|
||||||
<Dialog sx={dialogStyle} open={settingTime} onClose={handleCloseSetTime}>
|
|
||||||
<DialogTitle>{LL.SET_TIME(1)}</DialogTitle>
|
|
||||||
<DialogContent dividers>
|
|
||||||
<Box color="warning.main" p={0} pl={0} pr={0} mt={0} mb={2}>
|
|
||||||
<Typography variant="body2">{LL.SET_TIME_TEXT()}</Typography>
|
|
||||||
</Box>
|
|
||||||
<TextField
|
|
||||||
label={LL.LOCAL_TIME(0)}
|
|
||||||
type="datetime-local"
|
|
||||||
value={localTime}
|
|
||||||
onChange={updateLocalTime}
|
|
||||||
disabled={processing}
|
|
||||||
fullWidth
|
|
||||||
slotProps={{
|
|
||||||
inputLabel: {
|
|
||||||
shrink: true
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</DialogContent>
|
|
||||||
<DialogActions>
|
|
||||||
<Button
|
|
||||||
startIcon={<CancelIcon />}
|
|
||||||
variant="outlined"
|
|
||||||
onClick={handleCloseSetTime}
|
|
||||||
color="secondary"
|
|
||||||
>
|
|
||||||
{LL.CANCEL()}
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
startIcon={<AccessTimeIcon />}
|
|
||||||
variant="outlined"
|
|
||||||
onClick={configureTime}
|
|
||||||
disabled={processing}
|
|
||||||
color="primary"
|
|
||||||
>
|
|
||||||
{LL.UPDATE()}
|
|
||||||
</Button>
|
|
||||||
</DialogActions>
|
|
||||||
</Dialog>
|
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useMemo, useState } from 'react';
|
import { useState } from 'react';
|
||||||
|
|
||||||
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -17,7 +17,6 @@ import {
|
|||||||
DialogActions,
|
DialogActions,
|
||||||
DialogContent,
|
DialogContent,
|
||||||
DialogTitle,
|
DialogTitle,
|
||||||
Divider,
|
|
||||||
List
|
List
|
||||||
} from '@mui/material';
|
} from '@mui/material';
|
||||||
|
|
||||||
@@ -30,159 +29,134 @@ import { SectionContent, useLayoutTitle } from 'components';
|
|||||||
import ListMenuItem from 'components/layout/ListMenuItem';
|
import ListMenuItem from 'components/layout/ListMenuItem';
|
||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
|
|
||||||
import SystemMonitor from '../status/SystemMonitor';
|
|
||||||
|
|
||||||
const Settings = () => {
|
const Settings = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
useLayoutTitle(LL.SETTINGS(0));
|
useLayoutTitle(LL.SETTINGS(0));
|
||||||
|
|
||||||
const [confirmFactoryReset, setConfirmFactoryReset] = useState(false);
|
const [confirmFactoryReset, setConfirmFactoryReset] = useState<boolean>(false);
|
||||||
const [restarting, setRestarting] = useState<boolean>();
|
|
||||||
|
|
||||||
const { send: sendAPI } = useRequest((data: APIcall) => API(data), {
|
const { send: sendAPI } = useRequest((data: APIcall) => API(data), {
|
||||||
immediate: false
|
immediate: false
|
||||||
});
|
});
|
||||||
|
|
||||||
const doFormat = useCallback(async () => {
|
const doFormat = async () => {
|
||||||
await sendAPI({ device: 'system', cmd: 'format', id: 0 }).then(() => {
|
await sendAPI({ device: 'system', cmd: 'format', id: 0 }).then(() => {
|
||||||
setRestarting(true);
|
|
||||||
setConfirmFactoryReset(false);
|
setConfirmFactoryReset(false);
|
||||||
});
|
});
|
||||||
}, [sendAPI]);
|
};
|
||||||
|
|
||||||
const handleFactoryResetClose = useCallback(() => {
|
const renderFactoryResetDialog = () => (
|
||||||
setConfirmFactoryReset(false);
|
<Dialog
|
||||||
}, []);
|
sx={dialogStyle}
|
||||||
|
open={confirmFactoryReset}
|
||||||
const handleFactoryResetClick = useCallback(() => {
|
onClose={() => setConfirmFactoryReset(false)}
|
||||||
setConfirmFactoryReset(true);
|
>
|
||||||
}, []);
|
<DialogTitle>{LL.FACTORY_RESET()}</DialogTitle>
|
||||||
|
<DialogContent dividers>{LL.SYSTEM_FACTORY_TEXT_DIALOG()}</DialogContent>
|
||||||
const content = useMemo(() => {
|
<DialogActions>
|
||||||
return (
|
<Button
|
||||||
<>
|
startIcon={<CancelIcon />}
|
||||||
<List>
|
variant="outlined"
|
||||||
<ListMenuItem
|
onClick={() => setConfirmFactoryReset(false)}
|
||||||
icon={TuneIcon}
|
color="secondary"
|
||||||
bgcolor="#134ba2"
|
|
||||||
label={LL.APPLICATION()}
|
|
||||||
text={LL.APPLICATION_SETTINGS_1()}
|
|
||||||
to="application"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<ListMenuItem
|
|
||||||
icon={SettingsEthernetIcon}
|
|
||||||
bgcolor="#40828f"
|
|
||||||
label={LL.NETWORK(0)}
|
|
||||||
text={LL.CONFIGURE(LL.SETTINGS_OF(LL.NETWORK(1)))}
|
|
||||||
to="network"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<ListMenuItem
|
|
||||||
icon={SettingsInputAntennaIcon}
|
|
||||||
bgcolor="#5f9a5f"
|
|
||||||
label={LL.ACCESS_POINT(0)}
|
|
||||||
text={LL.CONFIGURE(LL.ACCESS_POINT(1))}
|
|
||||||
to="ap"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<ListMenuItem
|
|
||||||
icon={AccessTimeIcon}
|
|
||||||
bgcolor="#c5572c"
|
|
||||||
label="NTP"
|
|
||||||
text={LL.CONFIGURE(LL.LOCAL_TIME(1))}
|
|
||||||
to="ntp"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<ListMenuItem
|
|
||||||
icon={DeviceHubIcon}
|
|
||||||
bgcolor="#68374d"
|
|
||||||
label="MQTT"
|
|
||||||
text={LL.CONFIGURE('MQTT')}
|
|
||||||
to="mqtt"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<ListMenuItem
|
|
||||||
icon={LockIcon}
|
|
||||||
label={LL.SECURITY(0)}
|
|
||||||
text={LL.SECURITY_1()}
|
|
||||||
to="security"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<ListMenuItem
|
|
||||||
icon={ViewModuleIcon}
|
|
||||||
bgcolor="#efc34b"
|
|
||||||
label={LL.MODULES()}
|
|
||||||
text={LL.MODULES_1()}
|
|
||||||
to="modules"
|
|
||||||
/>
|
|
||||||
|
|
||||||
<ListMenuItem
|
|
||||||
icon={ImportExportIcon}
|
|
||||||
bgcolor="#5d89f7"
|
|
||||||
label={LL.DOWNLOAD_UPLOAD()}
|
|
||||||
text={LL.DOWNLOAD_UPLOAD_1()}
|
|
||||||
to="downloadUpload"
|
|
||||||
/>
|
|
||||||
</List>
|
|
||||||
|
|
||||||
<Dialog
|
|
||||||
sx={dialogStyle}
|
|
||||||
open={confirmFactoryReset}
|
|
||||||
onClose={handleFactoryResetClose}
|
|
||||||
>
|
>
|
||||||
<DialogTitle>{LL.FACTORY_RESET()}</DialogTitle>
|
{LL.CANCEL()}
|
||||||
<DialogContent dividers>{LL.SYSTEM_FACTORY_TEXT_DIALOG()}</DialogContent>
|
</Button>
|
||||||
<DialogActions>
|
<Button
|
||||||
<Button
|
startIcon={<SettingsBackupRestoreIcon />}
|
||||||
startIcon={<CancelIcon />}
|
variant="outlined"
|
||||||
variant="outlined"
|
onClick={doFormat}
|
||||||
onClick={handleFactoryResetClose}
|
color="error"
|
||||||
color="secondary"
|
|
||||||
>
|
|
||||||
{LL.CANCEL()}
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
startIcon={<SettingsBackupRestoreIcon />}
|
|
||||||
variant="outlined"
|
|
||||||
onClick={doFormat}
|
|
||||||
color="error"
|
|
||||||
>
|
|
||||||
{LL.FACTORY_RESET()}
|
|
||||||
</Button>
|
|
||||||
</DialogActions>
|
|
||||||
</Dialog>
|
|
||||||
|
|
||||||
<Divider />
|
|
||||||
|
|
||||||
<Box
|
|
||||||
mt={2}
|
|
||||||
display="flex"
|
|
||||||
justifyContent="flex-end"
|
|
||||||
flexWrap="nowrap"
|
|
||||||
whiteSpace="nowrap"
|
|
||||||
>
|
>
|
||||||
<Button
|
{LL.FACTORY_RESET()}
|
||||||
startIcon={<SettingsBackupRestoreIcon />}
|
</Button>
|
||||||
variant="outlined"
|
</DialogActions>
|
||||||
onClick={handleFactoryResetClick}
|
</Dialog>
|
||||||
color="error"
|
);
|
||||||
>
|
|
||||||
{LL.FACTORY_RESET()}
|
|
||||||
</Button>
|
|
||||||
</Box>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
}, [
|
|
||||||
LL,
|
|
||||||
handleFactoryResetClick,
|
|
||||||
handleFactoryResetClose,
|
|
||||||
doFormat,
|
|
||||||
confirmFactoryReset,
|
|
||||||
restarting
|
|
||||||
]);
|
|
||||||
|
|
||||||
return restarting ? <SystemMonitor /> : <SectionContent>{content}</SectionContent>;
|
const content = () => (
|
||||||
|
<>
|
||||||
|
<List sx={{ borderRadius: 3, border: '2px solid grey' }}>
|
||||||
|
<ListMenuItem
|
||||||
|
icon={TuneIcon}
|
||||||
|
bgcolor="#134ba2"
|
||||||
|
label={LL.APPLICATION()}
|
||||||
|
text={LL.APPLICATION_SETTINGS_1()}
|
||||||
|
to="application"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ListMenuItem
|
||||||
|
icon={SettingsEthernetIcon}
|
||||||
|
bgcolor="#40828f"
|
||||||
|
label={LL.NETWORK(0)}
|
||||||
|
text={LL.CONFIGURE(LL.SETTINGS_OF(LL.NETWORK(1)))}
|
||||||
|
to="network"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ListMenuItem
|
||||||
|
icon={SettingsInputAntennaIcon}
|
||||||
|
bgcolor="#5f9a5f"
|
||||||
|
label={LL.ACCESS_POINT(0)}
|
||||||
|
text={LL.CONFIGURE(LL.ACCESS_POINT(1))}
|
||||||
|
to="ap"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ListMenuItem
|
||||||
|
icon={AccessTimeIcon}
|
||||||
|
bgcolor="#c5572c"
|
||||||
|
label="NTP"
|
||||||
|
text={LL.CONFIGURE(LL.LOCAL_TIME(1))}
|
||||||
|
to="ntp"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ListMenuItem
|
||||||
|
icon={DeviceHubIcon}
|
||||||
|
bgcolor="#68374d"
|
||||||
|
label="MQTT"
|
||||||
|
text={LL.CONFIGURE('MQTT')}
|
||||||
|
to="mqtt"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ListMenuItem
|
||||||
|
icon={LockIcon}
|
||||||
|
label={LL.SECURITY(0)}
|
||||||
|
text={LL.SECURITY_1()}
|
||||||
|
to="security"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ListMenuItem
|
||||||
|
icon={ViewModuleIcon}
|
||||||
|
bgcolor="#efc34b"
|
||||||
|
label={LL.MODULES()}
|
||||||
|
text={LL.MODULES_1()}
|
||||||
|
to="modules"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<ListMenuItem
|
||||||
|
icon={ImportExportIcon}
|
||||||
|
bgcolor="#5d89f7"
|
||||||
|
label={LL.DOWNLOAD_UPLOAD()}
|
||||||
|
text={LL.DOWNLOAD_UPLOAD_1()}
|
||||||
|
to="downloadUpload"
|
||||||
|
/>
|
||||||
|
</List>
|
||||||
|
|
||||||
|
{renderFactoryResetDialog()}
|
||||||
|
|
||||||
|
<Box mt={2} display="flex" flexWrap="wrap">
|
||||||
|
<Button
|
||||||
|
startIcon={<SettingsBackupRestoreIcon />}
|
||||||
|
variant="outlined"
|
||||||
|
onClick={() => setConfirmFactoryReset(true)}
|
||||||
|
color="error"
|
||||||
|
>
|
||||||
|
{LL.FACTORY_RESET()}
|
||||||
|
</Button>
|
||||||
|
</Box>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
|
||||||
|
return <SectionContent>{content()}</SectionContent>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default Settings;
|
export default Settings;
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import { useMemo } from 'react';
|
|
||||||
|
|
||||||
import { MenuItem } from '@mui/material';
|
import { MenuItem } from '@mui/material';
|
||||||
|
|
||||||
export const TIME_ZONES: Record<string, string> = {
|
type TimeZones = Record<string, string>;
|
||||||
|
|
||||||
|
export const TIME_ZONES: TimeZones = {
|
||||||
'Africa/Abidjan': 'GMT0',
|
'Africa/Abidjan': 'GMT0',
|
||||||
'Africa/Accra': 'GMT0',
|
'Africa/Accra': 'GMT0',
|
||||||
'Africa/Addis_Ababa': 'EAT-3',
|
'Africa/Addis_Ababa': 'EAT-3',
|
||||||
@@ -465,33 +465,14 @@ export const TIME_ZONES: Record<string, string> = {
|
|||||||
'Pacific/Wallis': 'UNK-12'
|
'Pacific/Wallis': 'UNK-12'
|
||||||
};
|
};
|
||||||
|
|
||||||
// Pre-compute sorted timezone labels for better performance
|
|
||||||
export const TIME_ZONE_LABELS = Object.keys(TIME_ZONES).sort();
|
|
||||||
|
|
||||||
export function selectedTimeZone(label: string, format: string) {
|
export function selectedTimeZone(label: string, format: string) {
|
||||||
return TIME_ZONES[label] === format ? label : undefined;
|
return TIME_ZONES[label] === format ? label : undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Memoized version for use in components
|
|
||||||
export function useTimeZoneSelectItems() {
|
|
||||||
return useMemo(
|
|
||||||
() =>
|
|
||||||
TIME_ZONE_LABELS.map((label) => (
|
|
||||||
<MenuItem key={label} value={label}>
|
|
||||||
{label}
|
|
||||||
</MenuItem>
|
|
||||||
)),
|
|
||||||
[]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Fallback export for backward compatibility - now memoized
|
|
||||||
const precomputedTimeZoneItems = TIME_ZONE_LABELS.map((label) => (
|
|
||||||
<MenuItem key={label} value={label}>
|
|
||||||
{label}
|
|
||||||
</MenuItem>
|
|
||||||
));
|
|
||||||
|
|
||||||
export function timeZoneSelectItems() {
|
export function timeZoneSelectItems() {
|
||||||
return precomputedTimeZoneItems;
|
return Object.keys(TIME_ZONES).map((label) => (
|
||||||
|
<MenuItem key={label} value={label}>
|
||||||
|
{label}
|
||||||
|
</MenuItem>
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useCallback, useMemo, useState } from 'react';
|
import { useCallback, useState } from 'react';
|
||||||
import {
|
import {
|
||||||
Navigate,
|
Navigate,
|
||||||
Route,
|
Route,
|
||||||
@@ -28,13 +28,14 @@ const Network = () => {
|
|||||||
[
|
[
|
||||||
{
|
{
|
||||||
path: '/settings/network/settings',
|
path: '/settings/network/settings',
|
||||||
element: <NetworkSettings />
|
element: <NetworkSettings />,
|
||||||
|
dog: 'woof'
|
||||||
},
|
},
|
||||||
{ path: '/settings/network/scan', element: <WiFiNetworkScanner /> }
|
{ path: '/settings/network/scan', element: <WiFiNetworkScanner /> }
|
||||||
],
|
],
|
||||||
useLocation()
|
useLocation()
|
||||||
);
|
);
|
||||||
const routerTab = matchedRoutes?.[0]?.route.path || false;
|
const routerTab = matchedRoutes?.[0].route.path || false;
|
||||||
|
|
||||||
const navigate = useNavigate();
|
const navigate = useNavigate();
|
||||||
|
|
||||||
@@ -52,17 +53,14 @@ const Network = () => {
|
|||||||
setSelectedNetwork(undefined);
|
setSelectedNetwork(undefined);
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const contextValue = useMemo(
|
|
||||||
() => ({
|
|
||||||
...(selectedNetwork && { selectedNetwork }),
|
|
||||||
selectNetwork,
|
|
||||||
deselectNetwork
|
|
||||||
}),
|
|
||||||
[selectedNetwork, selectNetwork, deselectNetwork]
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<WiFiConnectionContext.Provider value={contextValue}>
|
<WiFiConnectionContext.Provider
|
||||||
|
value={{
|
||||||
|
selectedNetwork,
|
||||||
|
selectNetwork,
|
||||||
|
deselectNetwork
|
||||||
|
}}
|
||||||
|
>
|
||||||
<RouterTabs value={routerTab}>
|
<RouterTabs value={routerTab}>
|
||||||
<Tab
|
<Tab
|
||||||
value="/settings/network/settings"
|
value="/settings/network/settings"
|
||||||
@@ -82,4 +80,4 @@ const Network = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default memo(Network);
|
export default Network;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useCallback, useContext, useEffect, useState } from 'react';
|
import { useContext, useEffect, useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -104,42 +104,43 @@ const NetworkSettings = () => {
|
|||||||
origData,
|
origData,
|
||||||
dirtyFlags,
|
dirtyFlags,
|
||||||
setDirtyFlags,
|
setDirtyFlags,
|
||||||
updateDataValue as (value: unknown) => void
|
updateDataValue
|
||||||
);
|
);
|
||||||
|
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
|
|
||||||
const validateAndSubmit = useCallback(async () => {
|
useEffect(() => deselectNetwork, [deselectNetwork]);
|
||||||
if (!data) return;
|
|
||||||
try {
|
|
||||||
setFieldErrors(undefined);
|
|
||||||
await validate(createNetworkSettingsValidator(data), data);
|
|
||||||
await saveData();
|
|
||||||
} catch (error) {
|
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
|
||||||
}
|
|
||||||
deselectNetwork();
|
|
||||||
}, [data, saveData, deselectNetwork]);
|
|
||||||
|
|
||||||
const setCancel = useCallback(async () => {
|
|
||||||
deselectNetwork();
|
|
||||||
await loadData();
|
|
||||||
}, [deselectNetwork, loadData]);
|
|
||||||
|
|
||||||
const doRestart = useCallback(async () => {
|
|
||||||
setRestarting(true);
|
|
||||||
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
|
||||||
(error: Error) => {
|
|
||||||
toast.error(error.message);
|
|
||||||
}
|
|
||||||
);
|
|
||||||
}, [sendAPI]);
|
|
||||||
|
|
||||||
const content = () => {
|
const content = () => {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const validateAndSubmit = async () => {
|
||||||
|
try {
|
||||||
|
setFieldErrors(undefined);
|
||||||
|
await validate(createNetworkSettingsValidator(data), data);
|
||||||
|
await saveData();
|
||||||
|
} catch (error) {
|
||||||
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
|
}
|
||||||
|
deselectNetwork();
|
||||||
|
};
|
||||||
|
|
||||||
|
const setCancel = async () => {
|
||||||
|
deselectNetwork();
|
||||||
|
await loadData();
|
||||||
|
};
|
||||||
|
|
||||||
|
const doRestart = async () => {
|
||||||
|
setRestarting(true);
|
||||||
|
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
||||||
|
(error: Error) => {
|
||||||
|
toast.error(error.message);
|
||||||
|
}
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Typography variant="h6" color="primary">
|
<Typography variant="h6" color="primary">
|
||||||
@@ -164,14 +165,14 @@ const NetworkSettings = () => {
|
|||||||
selectedNetwork.bssid
|
selectedNetwork.bssid
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
<IconButton onClick={setCancel} aria-label={LL.CANCEL()}>
|
<IconButton onClick={setCancel}>
|
||||||
<DeleteIcon />
|
<DeleteIcon />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
</List>
|
</List>
|
||||||
) : (
|
) : (
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="ssid"
|
name="ssid"
|
||||||
label={'SSID (' + LL.NETWORK_BLANK_SSID() + ')'}
|
label={'SSID (' + LL.NETWORK_BLANK_SSID() + ')'}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -182,7 +183,7 @@ const NetworkSettings = () => {
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="bssid"
|
name="bssid"
|
||||||
label={'BSSID (' + LL.NETWORK_BLANK_BSSID() + ')'}
|
label={'BSSID (' + LL.NETWORK_BLANK_BSSID() + ')'}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -193,7 +194,7 @@ const NetworkSettings = () => {
|
|||||||
/>
|
/>
|
||||||
{(!selectedNetwork || !isNetworkOpen(selectedNetwork)) && (
|
{(!selectedNetwork || !isNetworkOpen(selectedNetwork)) && (
|
||||||
<ValidatedPasswordField
|
<ValidatedPasswordField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="password"
|
name="password"
|
||||||
label={LL.PASSWORD()}
|
label={LL.PASSWORD()}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -250,7 +251,7 @@ const NetworkSettings = () => {
|
|||||||
{LL.GENERAL_OPTIONS()}
|
{LL.GENERAL_OPTIONS()}
|
||||||
</Typography>
|
</Typography>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="hostname"
|
name="hostname"
|
||||||
label={LL.HOSTNAME()}
|
label={LL.HOSTNAME()}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -303,7 +304,7 @@ const NetworkSettings = () => {
|
|||||||
{data.static_ip_config && (
|
{data.static_ip_config && (
|
||||||
<>
|
<>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="local_ip"
|
name="local_ip"
|
||||||
label={LL.AP_LOCAL_IP()}
|
label={LL.AP_LOCAL_IP()}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -313,7 +314,7 @@ const NetworkSettings = () => {
|
|||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="gateway_ip"
|
name="gateway_ip"
|
||||||
label={LL.NETWORK_GATEWAY()}
|
label={LL.NETWORK_GATEWAY()}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -323,7 +324,7 @@ const NetworkSettings = () => {
|
|||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="subnet_mask"
|
name="subnet_mask"
|
||||||
label={LL.NETWORK_SUBNET()}
|
label={LL.NETWORK_SUBNET()}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -333,7 +334,7 @@ const NetworkSettings = () => {
|
|||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="dns_ip_1"
|
name="dns_ip_1"
|
||||||
label="DNS #1"
|
label="DNS #1"
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -343,7 +344,7 @@ const NetworkSettings = () => {
|
|||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="dns_ip_2"
|
name="dns_ip_2"
|
||||||
label="DNS #2"
|
label="DNS #2"
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -355,9 +356,8 @@ const NetworkSettings = () => {
|
|||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
{restartNeeded && (
|
{restartNeeded && (
|
||||||
<MessageBox level="warning" message={LL.RESTART_TEXT(0)}>
|
<MessageBox my={2} level="warning" message={LL.RESTART_TEXT(0)}>
|
||||||
<Button
|
<Button
|
||||||
sx={{ ml: 2 }}
|
|
||||||
startIcon={<PowerSettingsNewIcon />}
|
startIcon={<PowerSettingsNewIcon />}
|
||||||
variant="contained"
|
variant="contained"
|
||||||
color="error"
|
color="error"
|
||||||
@@ -397,14 +397,12 @@ const NetworkSettings = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
return restarting ? (
|
return (
|
||||||
<SystemMonitor />
|
|
||||||
) : (
|
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
{blocker ? <BlockNavigation blocker={blocker} /> : null}
|
||||||
{content()}
|
{restarting ? <SystemMonitor /> : content()}
|
||||||
</SectionContent>
|
</SectionContent>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default memo(NetworkSettings);
|
export default NetworkSettings;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useCallback, useRef, useState } from 'react';
|
import { useRef, useState } from 'react';
|
||||||
|
|
||||||
import PermScanWifiIcon from '@mui/icons-material/PermScanWifi';
|
import PermScanWifiIcon from '@mui/icons-material/PermScanWifi';
|
||||||
import { Button } from '@mui/material';
|
import { Button } from '@mui/material';
|
||||||
@@ -48,12 +48,14 @@ const WiFiNetworkScanner = () => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
const renderNetworkScanner = useCallback(() => {
|
const renderNetworkScanner = () => {
|
||||||
if (!networkList) {
|
if (!networkList) {
|
||||||
return <FormLoader errorMessage={errorMessage || ''} />;
|
return (
|
||||||
|
<FormLoader message={LL.SCANNING() + '...'} errorMessage={errorMessage} />
|
||||||
|
);
|
||||||
}
|
}
|
||||||
return <WiFiNetworkSelector networkList={networkList} />;
|
return <WiFiNetworkSelector networkList={networkList} />;
|
||||||
}, [networkList, errorMessage]);
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
<SectionContent>
|
||||||
@@ -73,4 +75,4 @@ const WiFiNetworkScanner = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default memo(WiFiNetworkScanner);
|
export default WiFiNetworkScanner;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useCallback, useContext } from 'react';
|
import { useContext } from 'react';
|
||||||
|
|
||||||
import LockIcon from '@mui/icons-material/Lock';
|
import LockIcon from '@mui/icons-material/Lock';
|
||||||
import LockOpenIcon from '@mui/icons-material/LockOpen';
|
import LockOpenIcon from '@mui/icons-material/LockOpen';
|
||||||
@@ -63,41 +63,38 @@ const WiFiNetworkSelector = ({ networkList }: { networkList: WiFiNetworkList })
|
|||||||
|
|
||||||
const wifiConnectionContext = useContext(WiFiConnectionContext);
|
const wifiConnectionContext = useContext(WiFiConnectionContext);
|
||||||
|
|
||||||
const renderNetwork = useCallback(
|
const renderNetwork = (network: WiFiNetwork) => (
|
||||||
(network: WiFiNetwork) => (
|
<ListItem
|
||||||
<ListItem
|
key={network.bssid}
|
||||||
key={network.bssid}
|
onClick={() => wifiConnectionContext.selectNetwork(network)}
|
||||||
onClick={() => wifiConnectionContext.selectNetwork(network)}
|
>
|
||||||
>
|
<ListItemAvatar>
|
||||||
<ListItemAvatar>
|
<Avatar>{isNetworkOpen(network) ? <LockOpenIcon /> : <LockIcon />}</Avatar>
|
||||||
<Avatar>{isNetworkOpen(network) ? <LockOpenIcon /> : <LockIcon />}</Avatar>
|
</ListItemAvatar>
|
||||||
</ListItemAvatar>
|
<ListItemText
|
||||||
<ListItemText
|
primary={network.ssid}
|
||||||
primary={network.ssid}
|
secondary={
|
||||||
secondary={
|
'Security: ' +
|
||||||
'Security: ' +
|
networkSecurityMode(network) +
|
||||||
networkSecurityMode(network) +
|
', Ch: ' +
|
||||||
', Ch: ' +
|
network.channel +
|
||||||
network.channel +
|
', bssid: ' +
|
||||||
', bssid: ' +
|
network.bssid
|
||||||
network.bssid
|
}
|
||||||
}
|
/>
|
||||||
/>
|
<ListItemIcon>
|
||||||
<ListItemIcon>
|
<Badge badgeContent={network.rssi + 'dBm'}>
|
||||||
<Badge badgeContent={network.rssi + 'dBm'}>
|
<WifiIcon sx={{ color: networkQualityHighlight(network, theme) }} />
|
||||||
<WifiIcon sx={{ color: networkQualityHighlight(network, theme) }} />
|
</Badge>
|
||||||
</Badge>
|
</ListItemIcon>
|
||||||
</ListItemIcon>
|
</ListItem>
|
||||||
</ListItem>
|
|
||||||
),
|
|
||||||
[wifiConnectionContext, theme]
|
|
||||||
);
|
);
|
||||||
|
|
||||||
if (networkList.networks.length === 0) {
|
if (networkList.networks.length === 0) {
|
||||||
return <MessageBox message={LL.NETWORK_NO_WIFI()} level="info" />;
|
return <MessageBox mt={2} mb={1} message={LL.NETWORK_NO_WIFI()} level="info" />;
|
||||||
}
|
}
|
||||||
|
|
||||||
return <List>{networkList.networks.map(renderNetwork)}</List>;
|
return <List>{networkList.networks.map(renderNetwork)}</List>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default memo(WiFiNetworkSelector);
|
export default WiFiNetworkSelector;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useEffect } from 'react';
|
import { useEffect } from 'react';
|
||||||
|
|
||||||
import CloseIcon from '@mui/icons-material/Close';
|
import CloseIcon from '@mui/icons-material/Close';
|
||||||
import {
|
import {
|
||||||
@@ -40,7 +40,7 @@ const GenerateToken = ({ username, onClose }: GenerateTokenProps) => {
|
|||||||
if (open) {
|
if (open) {
|
||||||
void generateToken();
|
void generateToken();
|
||||||
}
|
}
|
||||||
}, [open, generateToken]);
|
}, [open]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog
|
<Dialog
|
||||||
@@ -86,4 +86,4 @@ const GenerateToken = ({ username, onClose }: GenerateTokenProps) => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default memo(GenerateToken);
|
export default GenerateToken;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useCallback, useContext, useMemo, useState } from 'react';
|
import { useContext, useState } from 'react';
|
||||||
import { useBlocker } from 'react-router';
|
import { useBlocker } from 'react-router';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -55,16 +55,14 @@ const ManageUsers = () => {
|
|||||||
const blocker = useBlocker(changed !== 0);
|
const blocker = useBlocker(changed !== 0);
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
const table_theme = useMemo(
|
const table_theme = useTheme({
|
||||||
() =>
|
Table: `
|
||||||
useTheme({
|
|
||||||
Table: `
|
|
||||||
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) minmax(120px, max-content) 120px;
|
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) minmax(120px, max-content) 120px;
|
||||||
`,
|
`,
|
||||||
BaseRow: `
|
BaseRow: `
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
`,
|
`,
|
||||||
HeaderRow: `
|
HeaderRow: `
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
background-color: black;
|
background-color: black;
|
||||||
color: #90CAF9;
|
color: #90CAF9;
|
||||||
@@ -74,7 +72,7 @@ const ManageUsers = () => {
|
|||||||
border-bottom: 1px solid #565656;
|
border-bottom: 1px solid #565656;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
Row: `
|
Row: `
|
||||||
.td {
|
.td {
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
border-top: 1px solid #565656;
|
border-top: 1px solid #565656;
|
||||||
@@ -87,7 +85,7 @@ const ManageUsers = () => {
|
|||||||
background-color: #1e1e1e;
|
background-color: #1e1e1e;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
BaseCell: `
|
BaseCell: `
|
||||||
&:nth-of-type(2) {
|
&:nth-of-type(2) {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
@@ -95,81 +93,72 @@ const ManageUsers = () => {
|
|||||||
text-align: right;
|
text-align: right;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
}),
|
});
|
||||||
[]
|
|
||||||
);
|
|
||||||
|
|
||||||
const noAdminConfigured = useCallback(
|
|
||||||
() => !data?.users.find((u) => u.admin),
|
|
||||||
[data]
|
|
||||||
);
|
|
||||||
|
|
||||||
const removeUser = useCallback(
|
|
||||||
(toRemove: UserType) => {
|
|
||||||
if (!data) return;
|
|
||||||
const users = data.users.filter((u) => u.username !== toRemove.username);
|
|
||||||
updateDataValue({ ...data, users });
|
|
||||||
setChanged(changed + 1);
|
|
||||||
},
|
|
||||||
[data, updateDataValue, changed]
|
|
||||||
);
|
|
||||||
|
|
||||||
const createUser = useCallback(() => {
|
|
||||||
setCreating(true);
|
|
||||||
setUser({
|
|
||||||
username: '',
|
|
||||||
password: '',
|
|
||||||
admin: true
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const editUser = useCallback((toEdit: UserType) => {
|
|
||||||
setCreating(false);
|
|
||||||
setUser({ ...toEdit });
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const cancelEditingUser = useCallback(() => {
|
|
||||||
setUser(undefined);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const doneEditingUser = useCallback(() => {
|
|
||||||
if (user && data) {
|
|
||||||
const users = [
|
|
||||||
...data.users.filter(
|
|
||||||
(u: { username: string }) => u.username !== user.username
|
|
||||||
),
|
|
||||||
user
|
|
||||||
];
|
|
||||||
updateDataValue({ ...data, users });
|
|
||||||
setUser(undefined);
|
|
||||||
setChanged(changed + 1);
|
|
||||||
}
|
|
||||||
}, [user, data, updateDataValue, changed]);
|
|
||||||
|
|
||||||
const closeGenerateToken = useCallback(() => {
|
|
||||||
setGeneratingToken(undefined);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const generateTokenForUser = useCallback((username: string) => {
|
|
||||||
setGeneratingToken(username);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const onSubmit = useCallback(async () => {
|
|
||||||
await saveData();
|
|
||||||
await authenticatedContext.refresh();
|
|
||||||
setChanged(0);
|
|
||||||
}, [saveData, authenticatedContext]);
|
|
||||||
|
|
||||||
const onCancelSubmit = useCallback(async () => {
|
|
||||||
await loadData();
|
|
||||||
setChanged(0);
|
|
||||||
}, [loadData]);
|
|
||||||
|
|
||||||
const content = () => {
|
const content = () => {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const noAdminConfigured = () => !data.users.find((u) => u.admin);
|
||||||
|
|
||||||
|
const removeUser = (toRemove: UserType) => {
|
||||||
|
const users = data.users.filter((u) => u.username !== toRemove.username);
|
||||||
|
updateDataValue({ ...data, users });
|
||||||
|
setChanged(changed + 1);
|
||||||
|
};
|
||||||
|
|
||||||
|
const createUser = () => {
|
||||||
|
setCreating(true);
|
||||||
|
setUser({
|
||||||
|
username: '',
|
||||||
|
password: '',
|
||||||
|
admin: true
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const editUser = (toEdit: UserType) => {
|
||||||
|
setCreating(false);
|
||||||
|
setUser({ ...toEdit });
|
||||||
|
};
|
||||||
|
|
||||||
|
const cancelEditingUser = () => {
|
||||||
|
setUser(undefined);
|
||||||
|
};
|
||||||
|
|
||||||
|
const doneEditingUser = () => {
|
||||||
|
if (user) {
|
||||||
|
const users = [
|
||||||
|
...data.users.filter(
|
||||||
|
(u: { username: string }) => u.username !== user.username
|
||||||
|
),
|
||||||
|
user
|
||||||
|
];
|
||||||
|
updateDataValue({ ...data, users });
|
||||||
|
setUser(undefined);
|
||||||
|
setChanged(changed + 1);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const closeGenerateToken = () => {
|
||||||
|
setGeneratingToken(undefined);
|
||||||
|
};
|
||||||
|
|
||||||
|
const generateToken = (username: string) => {
|
||||||
|
setGeneratingToken(username);
|
||||||
|
};
|
||||||
|
|
||||||
|
const onSubmit = async () => {
|
||||||
|
await saveData();
|
||||||
|
await authenticatedContext.refresh();
|
||||||
|
setChanged(0);
|
||||||
|
};
|
||||||
|
|
||||||
|
const onCancelSubmit = async () => {
|
||||||
|
await loadData();
|
||||||
|
setChanged(0);
|
||||||
|
};
|
||||||
|
|
||||||
interface UserType2 {
|
interface UserType2 {
|
||||||
id: string;
|
id: string;
|
||||||
username: string;
|
username: string;
|
||||||
@@ -178,14 +167,10 @@ const ManageUsers = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// add id to the type, needed for the table
|
// add id to the type, needed for the table
|
||||||
const user_table = useMemo(
|
const user_table = data.users.map((u) => ({
|
||||||
() =>
|
...u,
|
||||||
data.users.map((u) => ({
|
id: u.username
|
||||||
...u,
|
})) as UserType2[];
|
||||||
id: u.username
|
|
||||||
})) as UserType2[],
|
|
||||||
[data.users]
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -211,24 +196,15 @@ const ManageUsers = () => {
|
|||||||
<Cell stiff>
|
<Cell stiff>
|
||||||
<IconButton
|
<IconButton
|
||||||
size="small"
|
size="small"
|
||||||
aria-label={LL.GENERATING_TOKEN()}
|
|
||||||
disabled={!authenticatedContext.me.admin}
|
disabled={!authenticatedContext.me.admin}
|
||||||
onClick={() => generateTokenForUser(u.username)}
|
onClick={() => generateToken(u.username)}
|
||||||
>
|
>
|
||||||
<VpnKeyIcon />
|
<VpnKeyIcon />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
<IconButton
|
<IconButton size="small" onClick={() => removeUser(u)}>
|
||||||
size="small"
|
|
||||||
onClick={() => removeUser(u)}
|
|
||||||
aria-label={LL.REMOVE()}
|
|
||||||
>
|
|
||||||
<DeleteIcon />
|
<DeleteIcon />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
<IconButton
|
<IconButton size="small" onClick={() => editUser(u)}>
|
||||||
size="small"
|
|
||||||
onClick={() => editUser(u)}
|
|
||||||
aria-label={LL.EDIT()}
|
|
||||||
>
|
|
||||||
<EditIcon />
|
<EditIcon />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
</Cell>
|
</Cell>
|
||||||
@@ -284,20 +260,15 @@ const ManageUsers = () => {
|
|||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</Box>
|
||||||
|
|
||||||
<GenerateToken
|
<GenerateToken username={generatingToken} onClose={closeGenerateToken} />
|
||||||
username={generatingToken || ''}
|
<User
|
||||||
onClose={closeGenerateToken}
|
user={user}
|
||||||
|
setUser={setUser}
|
||||||
|
creating={creating}
|
||||||
|
onDoneEditing={doneEditingUser}
|
||||||
|
onCancelEditing={cancelEditingUser}
|
||||||
|
validator={createUserValidator(data.users, creating)}
|
||||||
/>
|
/>
|
||||||
{user && (
|
|
||||||
<User
|
|
||||||
user={user}
|
|
||||||
setUser={setUser}
|
|
||||||
creating={creating}
|
|
||||||
onDoneEditing={doneEditingUser}
|
|
||||||
onCancelEditing={cancelEditingUser}
|
|
||||||
validator={createUserValidator(data.users, creating)}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
@@ -310,4 +281,4 @@ const ManageUsers = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default memo(ManageUsers);
|
export default ManageUsers;
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
import { memo, useMemo } from 'react';
|
|
||||||
import { Navigate, Route, Routes, matchRoutes, useLocation } from 'react-router';
|
import { Navigate, Route, Routes, matchRoutes, useLocation } from 'react-router';
|
||||||
|
|
||||||
import { Tab } from '@mui/material';
|
import { Tab } from '@mui/material';
|
||||||
@@ -13,23 +12,14 @@ const Security = () => {
|
|||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
useLayoutTitle(LL.SECURITY(0));
|
useLayoutTitle(LL.SECURITY(0));
|
||||||
|
|
||||||
const location = useLocation();
|
const matchedRoutes = matchRoutes(
|
||||||
|
[
|
||||||
const matchedRoutes = useMemo(
|
{ path: '/settings/security/settings', element: <ManageUsers />, dog: 'woof' },
|
||||||
() =>
|
{ path: '/settings/security/users', element: <SecuritySettings /> }
|
||||||
matchRoutes(
|
],
|
||||||
[
|
useLocation()
|
||||||
{
|
|
||||||
path: '/settings/security/settings',
|
|
||||||
element: <ManageUsers />
|
|
||||||
},
|
|
||||||
{ path: '/settings/security/users', element: <SecuritySettings /> }
|
|
||||||
],
|
|
||||||
location
|
|
||||||
),
|
|
||||||
[location]
|
|
||||||
);
|
);
|
||||||
const routerTab = matchedRoutes?.[0]?.route.path || false;
|
const routerTab = matchedRoutes?.[0].route.path || false;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
@@ -52,4 +42,4 @@ const Security = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default memo(Security);
|
export default Security;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useCallback, useContext, useState } from 'react';
|
import { useContext, useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import WarningIcon from '@mui/icons-material/Warning';
|
import WarningIcon from '@mui/icons-material/Warning';
|
||||||
@@ -44,33 +44,32 @@ const SecuritySettings = () => {
|
|||||||
const authenticatedContext = useContext(AuthenticatedContext);
|
const authenticatedContext = useContext(AuthenticatedContext);
|
||||||
|
|
||||||
const updateFormValue = updateValueDirty(
|
const updateFormValue = updateValueDirty(
|
||||||
origData as unknown as Record<string, unknown>,
|
origData,
|
||||||
dirtyFlags,
|
dirtyFlags,
|
||||||
setDirtyFlags,
|
setDirtyFlags,
|
||||||
updateDataValue as (value: unknown) => void
|
updateDataValue
|
||||||
);
|
);
|
||||||
|
|
||||||
const validateAndSubmit = useCallback(async () => {
|
|
||||||
if (!data) return;
|
|
||||||
try {
|
|
||||||
setFieldErrors(undefined);
|
|
||||||
await validate(SECURITY_SETTINGS_VALIDATOR, data);
|
|
||||||
await saveData();
|
|
||||||
await authenticatedContext.refresh();
|
|
||||||
} catch (error) {
|
|
||||||
setFieldErrors(error as ValidateFieldsError);
|
|
||||||
}
|
|
||||||
}, [data, saveData, authenticatedContext]);
|
|
||||||
|
|
||||||
const content = () => {
|
const content = () => {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const validateAndSubmit = async () => {
|
||||||
|
try {
|
||||||
|
setFieldErrors(undefined);
|
||||||
|
await validate(SECURITY_SETTINGS_VALIDATOR, data);
|
||||||
|
await saveData();
|
||||||
|
await authenticatedContext.refresh();
|
||||||
|
} catch (error) {
|
||||||
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<ValidatedPasswordField
|
<ValidatedPasswordField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="jwt_secret"
|
name="jwt_secret"
|
||||||
label={LL.SU_PASSWORD()}
|
label={LL.SU_PASSWORD()}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -116,4 +115,4 @@ const SecuritySettings = () => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default memo(SecuritySettings);
|
export default SecuritySettings;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { memo, useCallback, useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
import type { FC } from 'react';
|
import type { FC } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
@@ -45,14 +45,7 @@ const User: FC<UserFormProps> = ({
|
|||||||
}) => {
|
}) => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
const updateFormValue = updateValue((updater) => {
|
const updateFormValue = updateValue(setUser);
|
||||||
setUser((prevState) => {
|
|
||||||
if (!prevState) return prevState;
|
|
||||||
return updater(
|
|
||||||
prevState as unknown as Record<string, unknown>
|
|
||||||
) as unknown as UserType;
|
|
||||||
});
|
|
||||||
});
|
|
||||||
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
const [fieldErrors, setFieldErrors] = useState<ValidateFieldsError>();
|
||||||
const open = !!user;
|
const open = !!user;
|
||||||
|
|
||||||
@@ -62,7 +55,7 @@ const User: FC<UserFormProps> = ({
|
|||||||
}
|
}
|
||||||
}, [open]);
|
}, [open]);
|
||||||
|
|
||||||
const validateAndDone = useCallback(async () => {
|
const validateAndDone = async () => {
|
||||||
if (user) {
|
if (user) {
|
||||||
try {
|
try {
|
||||||
setFieldErrors(undefined);
|
setFieldErrors(undefined);
|
||||||
@@ -72,7 +65,7 @@ const User: FC<UserFormProps> = ({
|
|||||||
setFieldErrors(error as ValidateFieldsError);
|
setFieldErrors(error as ValidateFieldsError);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}, [user, validator, onDoneEditing]);
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Dialog
|
<Dialog
|
||||||
@@ -89,7 +82,7 @@ const User: FC<UserFormProps> = ({
|
|||||||
</DialogTitle>
|
</DialogTitle>
|
||||||
<DialogContent dividers>
|
<DialogContent dividers>
|
||||||
<ValidatedTextField
|
<ValidatedTextField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="username"
|
name="username"
|
||||||
label={LL.USERNAME(1)}
|
label={LL.USERNAME(1)}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -100,7 +93,7 @@ const User: FC<UserFormProps> = ({
|
|||||||
margin="normal"
|
margin="normal"
|
||||||
/>
|
/>
|
||||||
<ValidatedPasswordField
|
<ValidatedPasswordField
|
||||||
fieldErrors={fieldErrors || {}}
|
fieldErrors={fieldErrors}
|
||||||
name="password"
|
name="password"
|
||||||
label={LL.PASSWORD()}
|
label={LL.PASSWORD()}
|
||||||
fullWidth
|
fullWidth
|
||||||
@@ -144,4 +137,4 @@ const User: FC<UserFormProps> = ({
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default memo(User);
|
export default User;
|
||||||
|
|||||||
@@ -34,43 +34,37 @@ export const apStatusHighlight = ({ status }: APStatusType, theme: Theme) => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const getApStatusText = (
|
|
||||||
status: APNetworkStatus,
|
|
||||||
LL: ReturnType<typeof useI18nContext>['LL']
|
|
||||||
) => {
|
|
||||||
switch (status) {
|
|
||||||
case APNetworkStatus.ACTIVE:
|
|
||||||
return LL.ACTIVE();
|
|
||||||
case APNetworkStatus.INACTIVE:
|
|
||||||
return LL.INACTIVE(0);
|
|
||||||
case APNetworkStatus.LINGERING:
|
|
||||||
return 'Lingering until idle';
|
|
||||||
default:
|
|
||||||
return LL.UNKNOWN();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const APStatus = () => {
|
const APStatus = () => {
|
||||||
const { data, send: loadData, error } = useRequest(APApi.readAPStatus);
|
const { data, send: loadData, error } = useRequest(APApi.readAPStatus);
|
||||||
const { LL } = useI18nContext();
|
|
||||||
const theme = useTheme();
|
|
||||||
|
|
||||||
useLayoutTitle(LL.ACCESS_POINT(0));
|
|
||||||
|
|
||||||
useInterval(() => {
|
useInterval(() => {
|
||||||
void loadData();
|
void loadData();
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!data) {
|
const { LL } = useI18nContext();
|
||||||
return (
|
useLayoutTitle(LL.ACCESS_POINT(0));
|
||||||
<SectionContent>
|
|
||||||
<FormLoader onRetry={loadData} errorMessage={error?.message || ''} />
|
|
||||||
</SectionContent>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
const theme = useTheme();
|
||||||
<SectionContent>
|
|
||||||
|
const apStatus = ({ status }: APStatusType) => {
|
||||||
|
switch (status) {
|
||||||
|
case APNetworkStatus.ACTIVE:
|
||||||
|
return LL.ACTIVE();
|
||||||
|
case APNetworkStatus.INACTIVE:
|
||||||
|
return LL.INACTIVE(0);
|
||||||
|
case APNetworkStatus.LINGERING:
|
||||||
|
return 'Lingering until idle';
|
||||||
|
default:
|
||||||
|
return LL.UNKNOWN();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const content = () => {
|
||||||
|
if (!data) {
|
||||||
|
return <FormLoader onRetry={loadData} errorMessage={error?.message} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
<List>
|
<List>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
@@ -78,26 +72,19 @@ const APStatus = () => {
|
|||||||
<SettingsInputAntennaIcon />
|
<SettingsInputAntennaIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText
|
<ListItemText primary={LL.STATUS_OF('')} secondary={apStatus(data)} />
|
||||||
primary={LL.STATUS_OF('')}
|
|
||||||
secondary={getApStatusText(data.status, LL)}
|
|
||||||
/>
|
|
||||||
</ListItem>
|
</ListItem>
|
||||||
|
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
|
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar sx={{ bgcolor: 'primary.main' }}>IP</Avatar>
|
<Avatar>IP</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText primary={LL.ADDRESS_OF('IP')} secondary={data.ip_address} />
|
<ListItemText primary={LL.ADDRESS_OF('IP')} secondary={data.ip_address} />
|
||||||
</ListItem>
|
</ListItem>
|
||||||
|
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
|
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar sx={{ bgcolor: 'primary.main' }}>
|
<Avatar>
|
||||||
<DeviceHubIcon />
|
<DeviceHubIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
@@ -106,22 +93,21 @@ const APStatus = () => {
|
|||||||
secondary={data.mac_address}
|
secondary={data.mac_address}
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
|
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
|
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar sx={{ bgcolor: 'primary.main' }}>
|
<Avatar>
|
||||||
<ComputerIcon />
|
<ComputerIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText primary={LL.AP_CLIENTS()} secondary={data.station_num} />
|
<ListItemText primary={LL.AP_CLIENTS()} secondary={data.station_num} />
|
||||||
</ListItem>
|
</ListItem>
|
||||||
|
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
</List>
|
</List>
|
||||||
</SectionContent>
|
);
|
||||||
);
|
};
|
||||||
|
|
||||||
|
return <SectionContent>{content()}</SectionContent>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default APStatus;
|
export default APStatus;
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
import { useCallback, useMemo } from 'react';
|
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Body,
|
Body,
|
||||||
Cell,
|
Cell,
|
||||||
@@ -19,12 +17,6 @@ import { useInterval } from 'utils';
|
|||||||
import { readActivity } from '../../api/app';
|
import { readActivity } from '../../api/app';
|
||||||
import type { Stat } from '../main/types';
|
import type { Stat } from '../main/types';
|
||||||
|
|
||||||
const QUALITY_COLORS = {
|
|
||||||
PERFECT: '#00FF7F',
|
|
||||||
WARNING: 'orange',
|
|
||||||
POOR: 'red'
|
|
||||||
} as const;
|
|
||||||
|
|
||||||
const SystemActivity = () => {
|
const SystemActivity = () => {
|
||||||
const { data, send: loadData, error } = useRequest(readActivity);
|
const { data, send: loadData, error } = useRequest(readActivity);
|
||||||
|
|
||||||
@@ -36,16 +28,14 @@ const SystemActivity = () => {
|
|||||||
|
|
||||||
useLayoutTitle(LL.DATA_TRAFFIC());
|
useLayoutTitle(LL.DATA_TRAFFIC());
|
||||||
|
|
||||||
const stats_theme = tableTheme(
|
const stats_theme = tableTheme({
|
||||||
useMemo(
|
Table: `
|
||||||
() => ({
|
|
||||||
Table: `
|
|
||||||
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) 90px 90px 80px;
|
--data-table-library_grid-template-columns: repeat(1, minmax(0, 1fr)) 90px 90px 80px;
|
||||||
`,
|
`,
|
||||||
BaseRow: `
|
BaseRow: `
|
||||||
font-size: 14px;
|
font-size: 14px;
|
||||||
`,
|
`,
|
||||||
HeaderRow: `
|
HeaderRow: `
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
background-color: black;
|
background-color: black;
|
||||||
color: #90CAF9;
|
color: #90CAF9;
|
||||||
@@ -55,7 +45,7 @@ const SystemActivity = () => {
|
|||||||
border-bottom: 1px solid #565656;
|
border-bottom: 1px solid #565656;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
Row: `
|
Row: `
|
||||||
.td {
|
.td {
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
border-top: 1px solid #565656;
|
border-top: 1px solid #565656;
|
||||||
@@ -69,42 +59,35 @@ const SystemActivity = () => {
|
|||||||
background-color: #1e1e1e;
|
background-color: #1e1e1e;
|
||||||
}
|
}
|
||||||
`,
|
`,
|
||||||
BaseCell: `
|
BaseCell: `
|
||||||
&:not(:first-of-type) {
|
&:not(:first-of-type) {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
`
|
`
|
||||||
}),
|
});
|
||||||
[]
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
const showName = useCallback(
|
const showName = (id: number) => {
|
||||||
(id: number) => {
|
const name: keyof Translation['STATUS_NAMES'] = id;
|
||||||
const name: keyof Translation['STATUS_NAMES'] =
|
return LL.STATUS_NAMES[name]();
|
||||||
id.toString() as keyof Translation['STATUS_NAMES'];
|
};
|
||||||
return LL.STATUS_NAMES[name]();
|
|
||||||
},
|
|
||||||
[LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const showQuality = useCallback((stat: Stat) => {
|
const showQuality = (stat: Stat) => {
|
||||||
if (stat.q === 0 || stat.s + stat.f === 0) {
|
if (stat.q === 0 || stat.s + stat.f === 0) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (stat.q === 100) {
|
if (stat.q === 100) {
|
||||||
return <div style={{ color: QUALITY_COLORS.PERFECT }}>{stat.q}%</div>;
|
return <div style={{ color: '#00FF7F' }}>{stat.q}%</div>;
|
||||||
}
|
}
|
||||||
if (stat.q >= 95) {
|
if (stat.q >= 95) {
|
||||||
return <div style={{ color: QUALITY_COLORS.WARNING }}>{stat.q}%</div>;
|
return <div style={{ color: 'orange' }}>{stat.q}%</div>;
|
||||||
} else {
|
} else {
|
||||||
return <div style={{ color: QUALITY_COLORS.POOR }}>{stat.q}%</div>;
|
return <div style={{ color: 'red' }}>{stat.q}%</div>;
|
||||||
}
|
}
|
||||||
}, []);
|
};
|
||||||
|
|
||||||
const content = useMemo(() => {
|
const content = () => {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
return <FormLoader onRetry={loadData} errorMessage={error?.message} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -137,9 +120,9 @@ const SystemActivity = () => {
|
|||||||
)}
|
)}
|
||||||
</Table>
|
</Table>
|
||||||
);
|
);
|
||||||
}, [data, loadData, error?.message, stats_theme, LL, showName, showQuality]);
|
};
|
||||||
|
|
||||||
return <SectionContent>{content}</SectionContent>;
|
return <SectionContent>{content()}</SectionContent>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default SystemActivity;
|
export default SystemActivity;
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
import { ReactElement } from 'react';
|
|
||||||
|
|
||||||
import AppsIcon from '@mui/icons-material/Apps';
|
import AppsIcon from '@mui/icons-material/Apps';
|
||||||
import DeveloperBoardIcon from '@mui/icons-material/DeveloperBoard';
|
import DeveloperBoardIcon from '@mui/icons-material/DeveloperBoard';
|
||||||
import DevicesIcon from '@mui/icons-material/Devices';
|
import DevicesIcon from '@mui/icons-material/Devices';
|
||||||
@@ -26,61 +24,10 @@ import { useInterval } from 'utils';
|
|||||||
|
|
||||||
import BBQKeesIcon from './bbqkees.svg';
|
import BBQKeesIcon from './bbqkees.svg';
|
||||||
|
|
||||||
// Constants
|
|
||||||
const AVATAR_COLORS = {
|
|
||||||
DEFAULT: '#5f9a5f',
|
|
||||||
BBQKEES: '#003289'
|
|
||||||
} as const;
|
|
||||||
|
|
||||||
const TEMP_THRESHOLD_CELSIUS = 90; // Temperature threshold to determine F vs C
|
|
||||||
|
|
||||||
function formatNumber(num: number) {
|
function formatNumber(num: number) {
|
||||||
return new Intl.NumberFormat().format(num);
|
return new Intl.NumberFormat().format(num);
|
||||||
}
|
}
|
||||||
|
|
||||||
function formatTemperature(temp?: number): string {
|
|
||||||
if (!temp) return '';
|
|
||||||
const unit = temp > TEMP_THRESHOLD_CELSIUS ? 'F' : 'C';
|
|
||||||
return `, T: ${temp} °${unit}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
function formatFlashSpeed(speed: number): string {
|
|
||||||
return (speed / 1000000).toFixed(0) + ' MHz';
|
|
||||||
}
|
|
||||||
|
|
||||||
function formatCPUCores(cores: number): string {
|
|
||||||
return cores === 1 ? 'single-core)' : 'dual-core)';
|
|
||||||
}
|
|
||||||
|
|
||||||
// Reusable component for hardware status list items
|
|
||||||
interface HardwareListItemProps {
|
|
||||||
icon: ReactElement;
|
|
||||||
primary: string;
|
|
||||||
secondary: string;
|
|
||||||
avatarColor?: string;
|
|
||||||
customIcon?: ReactElement | undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
const HardwareListItem = ({
|
|
||||||
icon,
|
|
||||||
primary,
|
|
||||||
secondary,
|
|
||||||
avatarColor = AVATAR_COLORS.DEFAULT,
|
|
||||||
customIcon
|
|
||||||
}: HardwareListItemProps) => (
|
|
||||||
<>
|
|
||||||
<ListItem>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar sx={{ bgcolor: avatarColor, color: 'white' }}>
|
|
||||||
{customIcon || icon}
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText primary={primary} secondary={secondary} />
|
|
||||||
</ListItem>
|
|
||||||
<Divider variant="inset" component="li" />
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
|
|
||||||
const HardwareStatus = () => {
|
const HardwareStatus = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
@@ -92,72 +39,175 @@ const HardwareStatus = () => {
|
|||||||
void loadData();
|
void loadData();
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!data) {
|
const content = () => {
|
||||||
return (
|
if (!data) {
|
||||||
<SectionContent>
|
return <FormLoader onRetry={loadData} errorMessage={error?.message} />;
|
||||||
<FormLoader onRetry={loadData} errorMessage={error?.message || ''} />
|
}
|
||||||
</SectionContent>
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
|
||||||
<List>
|
<List>
|
||||||
<HardwareListItem
|
<ListItem>
|
||||||
icon={<TapAndPlayIcon />}
|
<ListItemAvatar>
|
||||||
primary={`${LL.HARDWARE()} ${LL.DEVICE()}`}
|
{data.model ? (
|
||||||
secondary={data.model || data.cpu_type}
|
<Avatar sx={{ bgcolor: '#003289', color: 'white' }}>
|
||||||
avatarColor={data.model ? AVATAR_COLORS.BBQKEES : AVATAR_COLORS.DEFAULT}
|
<img
|
||||||
customIcon={
|
alt="BBQKees"
|
||||||
data.model ? (
|
src={BBQKeesIcon}
|
||||||
<img
|
style={{ width: 16, verticalAlign: 'middle' }}
|
||||||
alt="BBQKees"
|
/>
|
||||||
src={BBQKeesIcon}
|
</Avatar>
|
||||||
style={{ width: 16, verticalAlign: 'middle' }}
|
) : (
|
||||||
/>
|
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
||||||
) : undefined
|
<TapAndPlayIcon />
|
||||||
}
|
</Avatar>
|
||||||
/>
|
)}
|
||||||
<HardwareListItem
|
</ListItemAvatar>
|
||||||
icon={<DevicesIcon />}
|
<ListItemText
|
||||||
primary="SDK"
|
primary={LL.HARDWARE() + ' ' + LL.DEVICE()}
|
||||||
secondary={`${data.arduino_version} / ESP-IDF ${data.sdk_version}`}
|
secondary={data.model ? data.model : data.cpu_type}
|
||||||
/>
|
|
||||||
<HardwareListItem
|
|
||||||
icon={<DeveloperBoardIcon />}
|
|
||||||
primary="CPU"
|
|
||||||
secondary={`${data.esp_platform}/${data.cpu_type} (rev.${data.cpu_rev}, ${formatCPUCores(data.cpu_cores)} @ ${data.cpu_freq_mhz} Mhz${formatTemperature(data.temperature)}`}
|
|
||||||
/>
|
|
||||||
<HardwareListItem
|
|
||||||
icon={<MemoryIcon />}
|
|
||||||
primary={LL.FREE_MEMORY()}
|
|
||||||
secondary={`${formatNumber(data.free_heap)} KB (${formatNumber(data.max_alloc_heap)} KB max alloc, ${formatNumber(data.free_caps)} KB caps)`}
|
|
||||||
/>
|
|
||||||
{data.psram_size !== undefined && data.free_psram !== undefined && (
|
|
||||||
<HardwareListItem
|
|
||||||
icon={<AppsIcon />}
|
|
||||||
primary={LL.PSRAM()}
|
|
||||||
secondary={`${formatNumber(data.psram_size)} KB / ${formatNumber(data.free_psram)} KB`}
|
|
||||||
/>
|
/>
|
||||||
|
</ListItem>
|
||||||
|
<Divider variant="inset" component="li" />
|
||||||
|
<ListItem>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
||||||
|
<DevicesIcon />
|
||||||
|
</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText
|
||||||
|
primary="SDK"
|
||||||
|
secondary={data.arduino_version + ' / ESP-IDF ' + data.sdk_version}
|
||||||
|
/>
|
||||||
|
</ListItem>
|
||||||
|
<Divider variant="inset" component="li" />
|
||||||
|
<ListItem>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
||||||
|
<DeveloperBoardIcon />
|
||||||
|
</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText
|
||||||
|
primary="CPU"
|
||||||
|
secondary={
|
||||||
|
data.esp_platform +
|
||||||
|
'/' +
|
||||||
|
data.cpu_type +
|
||||||
|
' (rev.' +
|
||||||
|
data.cpu_rev +
|
||||||
|
', ' +
|
||||||
|
(data.cpu_cores === 1 ? 'single-core)' : 'dual-core)') +
|
||||||
|
' @ ' +
|
||||||
|
data.cpu_freq_mhz +
|
||||||
|
' Mhz' +
|
||||||
|
// bit of a hack : if the CPU temp is higher than 90 (=32 Fahrenheit if using Celsius), show F, otherwise C
|
||||||
|
(data.temperature
|
||||||
|
? ', T: ' +
|
||||||
|
data.temperature +
|
||||||
|
' °' +
|
||||||
|
(data.temperature > 90 ? 'F' : 'C')
|
||||||
|
: '')
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</ListItem>
|
||||||
|
<Divider variant="inset" component="li" />
|
||||||
|
<ListItem>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
||||||
|
<MemoryIcon />
|
||||||
|
</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText
|
||||||
|
primary={LL.FREE_MEMORY()}
|
||||||
|
secondary={
|
||||||
|
formatNumber(data.free_heap) +
|
||||||
|
' KB (' +
|
||||||
|
formatNumber(data.max_alloc_heap) +
|
||||||
|
' KB max alloc, ' +
|
||||||
|
formatNumber(data.free_caps) +
|
||||||
|
' KB caps)'
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</ListItem>
|
||||||
|
{data.psram_size !== undefined && data.free_psram !== undefined && (
|
||||||
|
<>
|
||||||
|
<Divider variant="inset" component="li" />
|
||||||
|
<ListItem>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
||||||
|
<AppsIcon />
|
||||||
|
</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText
|
||||||
|
primary={LL.PSRAM()}
|
||||||
|
secondary={
|
||||||
|
formatNumber(data.psram_size) +
|
||||||
|
' KB / ' +
|
||||||
|
formatNumber(data.free_psram) +
|
||||||
|
' KB'
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</ListItem>
|
||||||
|
</>
|
||||||
)}
|
)}
|
||||||
<HardwareListItem
|
<Divider variant="inset" component="li" />
|
||||||
icon={<SdStorageIcon />}
|
<ListItem>
|
||||||
primary={LL.FLASH()}
|
<ListItemAvatar>
|
||||||
secondary={`${formatNumber(data.flash_chip_size)} KB , ${formatFlashSpeed(data.flash_chip_speed)}`}
|
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
||||||
/>
|
<SdStorageIcon />
|
||||||
<HardwareListItem
|
</Avatar>
|
||||||
icon={<SdCardAlertIcon />}
|
</ListItemAvatar>
|
||||||
primary={LL.APPSIZE()}
|
<ListItemText
|
||||||
secondary={`${data.partition}: ${formatNumber(data.app_used)} KB / ${formatNumber(data.app_free)} KB`}
|
primary={LL.FLASH()}
|
||||||
/>
|
secondary={
|
||||||
<HardwareListItem
|
formatNumber(data.flash_chip_size) +
|
||||||
icon={<FolderIcon />}
|
' KB , ' +
|
||||||
primary={LL.FILESYSTEM()}
|
(data.flash_chip_speed / 1000000).toFixed(0) +
|
||||||
secondary={`${formatNumber(data.fs_used)} KB / ${formatNumber(data.fs_free)} KB`}
|
' MHz'
|
||||||
/>
|
}
|
||||||
|
/>
|
||||||
|
</ListItem>
|
||||||
|
<Divider variant="inset" component="li" />
|
||||||
|
<ListItem>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
||||||
|
<SdCardAlertIcon />
|
||||||
|
</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText
|
||||||
|
primary={LL.APPSIZE()}
|
||||||
|
secondary={
|
||||||
|
data.partition +
|
||||||
|
': ' +
|
||||||
|
formatNumber(data.app_used) +
|
||||||
|
' KB / ' +
|
||||||
|
formatNumber(data.app_free) +
|
||||||
|
' KB'
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</ListItem>
|
||||||
|
<Divider variant="inset" component="li" />
|
||||||
|
<ListItem>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar sx={{ bgcolor: '#5f9a5f', color: 'white' }}>
|
||||||
|
<FolderIcon />
|
||||||
|
</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText
|
||||||
|
primary={LL.FILESYSTEM()}
|
||||||
|
secondary={
|
||||||
|
formatNumber(data.fs_used) +
|
||||||
|
' KB / ' +
|
||||||
|
formatNumber(data.fs_free) +
|
||||||
|
' KB'
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</ListItem>
|
||||||
|
<Divider variant="inset" component="li" />
|
||||||
</List>
|
</List>
|
||||||
</SectionContent>
|
);
|
||||||
);
|
};
|
||||||
|
|
||||||
|
return <SectionContent>{content()}</SectionContent>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default HardwareStatus;
|
export default HardwareStatus;
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
import { type FC, memo, useMemo } from 'react';
|
|
||||||
|
|
||||||
import AutoAwesomeMotionIcon from '@mui/icons-material/AutoAwesomeMotion';
|
import AutoAwesomeMotionIcon from '@mui/icons-material/AutoAwesomeMotion';
|
||||||
import DeviceHubIcon from '@mui/icons-material/DeviceHub';
|
import DeviceHubIcon from '@mui/icons-material/DeviceHub';
|
||||||
import ReportIcon from '@mui/icons-material/Report';
|
import ReportIcon from '@mui/icons-material/Report';
|
||||||
@@ -24,28 +22,17 @@ import type { MqttStatusType } from 'types';
|
|||||||
import { MqttDisconnectReason } from 'types';
|
import { MqttDisconnectReason } from 'types';
|
||||||
import { useInterval } from 'utils';
|
import { useInterval } from 'utils';
|
||||||
|
|
||||||
// Disconnect reason lookup table - created once, reused across renders
|
|
||||||
const DISCONNECT_REASONS: Record<MqttDisconnectReason, string> = {
|
|
||||||
[MqttDisconnectReason.USER_OK]: 'User disconnected',
|
|
||||||
[MqttDisconnectReason.TCP_DISCONNECTED]: 'TCP disconnected',
|
|
||||||
[MqttDisconnectReason.MQTT_UNACCEPTABLE_PROTOCOL_VERSION]:
|
|
||||||
'Unacceptable protocol version',
|
|
||||||
[MqttDisconnectReason.MQTT_IDENTIFIER_REJECTED]: 'Client ID rejected',
|
|
||||||
[MqttDisconnectReason.MQTT_SERVER_UNAVAILABLE]: 'Server unavailable',
|
|
||||||
[MqttDisconnectReason.MQTT_MALFORMED_CREDENTIALS]: 'Malformed credentials',
|
|
||||||
[MqttDisconnectReason.MQTT_NOT_AUTHORIZED]: 'Not authorized',
|
|
||||||
[MqttDisconnectReason.TLS_BAD_FINGERPRINT]: 'TLS fingerprint invalid'
|
|
||||||
};
|
|
||||||
|
|
||||||
const getDisconnectReason = (disconnect_reason: MqttDisconnectReason): string =>
|
|
||||||
DISCONNECT_REASONS[disconnect_reason] ?? 'Unknown';
|
|
||||||
|
|
||||||
export const mqttStatusHighlight = (
|
export const mqttStatusHighlight = (
|
||||||
{ enabled, connected }: MqttStatusType,
|
{ enabled, connected }: MqttStatusType,
|
||||||
theme: Theme
|
theme: Theme
|
||||||
) => {
|
) => {
|
||||||
if (!enabled) return theme.palette.info.main;
|
if (!enabled) {
|
||||||
return connected ? theme.palette.success.main : theme.palette.error.main;
|
return theme.palette.info.main;
|
||||||
|
}
|
||||||
|
if (connected) {
|
||||||
|
return theme.palette.success.main;
|
||||||
|
}
|
||||||
|
return theme.palette.error.main;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const mqttPublishHighlight = (
|
export const mqttPublishHighlight = (
|
||||||
@@ -54,100 +41,114 @@ export const mqttPublishHighlight = (
|
|||||||
) => {
|
) => {
|
||||||
if (mqtt_fails === 0) return theme.palette.success.main;
|
if (mqtt_fails === 0) return theme.palette.success.main;
|
||||||
if (mqtt_fails < 10) return theme.palette.warning.main;
|
if (mqtt_fails < 10) return theme.palette.warning.main;
|
||||||
|
|
||||||
return theme.palette.error.main;
|
return theme.palette.error.main;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const mqttQueueHighlight = ({ mqtt_queued }: MqttStatusType, theme: Theme) =>
|
export const mqttQueueHighlight = (
|
||||||
mqtt_queued <= 1 ? theme.palette.success.main : theme.palette.warning.main;
|
{ mqtt_queued }: MqttStatusType,
|
||||||
|
theme: Theme
|
||||||
|
) => {
|
||||||
|
if (mqtt_queued <= 1) return theme.palette.success.main;
|
||||||
|
|
||||||
interface ConnectionStatusProps {
|
return theme.palette.warning.main;
|
||||||
data: MqttStatusType;
|
};
|
||||||
theme: Theme;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Memoized component to prevent unnecessary re-renders when parent updates
|
|
||||||
const ConnectionStatus: FC<ConnectionStatusProps> = memo(({ data, theme }) => {
|
|
||||||
const { LL } = useI18nContext();
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
{!data.connected && (
|
|
||||||
<>
|
|
||||||
<ListItem>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar>
|
|
||||||
<ReportIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText
|
|
||||||
primary={LL.DISCONNECT_REASON()}
|
|
||||||
secondary={getDisconnectReason(data.disconnect_reason)}
|
|
||||||
/>
|
|
||||||
</ListItem>
|
|
||||||
<Divider variant="inset" component="li" />
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
<ListItem>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar>#</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText primary={LL.ID_OF(LL.CLIENT())} secondary={data.client_id} />
|
|
||||||
</ListItem>
|
|
||||||
<Divider variant="inset" component="li" />
|
|
||||||
<ListItem>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar sx={{ bgcolor: mqttQueueHighlight(data, theme) }}>
|
|
||||||
<AutoAwesomeMotionIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText primary={LL.MQTT_QUEUE()} secondary={data.mqtt_queued} />
|
|
||||||
</ListItem>
|
|
||||||
<Divider variant="inset" component="li" />
|
|
||||||
<ListItem>
|
|
||||||
<ListItemAvatar>
|
|
||||||
<Avatar sx={{ bgcolor: mqttPublishHighlight(data, theme) }}>
|
|
||||||
<SpeakerNotesOffIcon />
|
|
||||||
</Avatar>
|
|
||||||
</ListItemAvatar>
|
|
||||||
<ListItemText primary={LL.ERRORS_OF('MQTT')} secondary={data.mqtt_fails} />
|
|
||||||
</ListItem>
|
|
||||||
<Divider variant="inset" component="li" />
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
const MqttStatus = () => {
|
const MqttStatus = () => {
|
||||||
const { data, send: loadData, error } = useRequest(MqttApi.readMqttStatus);
|
const { data, send: loadData, error } = useRequest(MqttApi.readMqttStatus);
|
||||||
const { LL } = useI18nContext();
|
|
||||||
const theme = useTheme();
|
|
||||||
|
|
||||||
useLayoutTitle('MQTT');
|
|
||||||
|
|
||||||
useInterval(() => {
|
useInterval(() => {
|
||||||
void loadData();
|
void loadData();
|
||||||
});
|
});
|
||||||
|
|
||||||
// Memoize error message separately to avoid re-renders on error object changes
|
const { LL } = useI18nContext();
|
||||||
const errorMessage = error?.message || '';
|
useLayoutTitle('MQTT');
|
||||||
|
|
||||||
const mqttStatusText = useMemo(() => {
|
const theme = useTheme();
|
||||||
if (!data) return '';
|
|
||||||
if (!data.enabled) return LL.NOT_ENABLED();
|
|
||||||
return data.connected
|
|
||||||
? `${LL.CONNECTED(0)} (${data.connect_count})`
|
|
||||||
: `${LL.DISCONNECTED()} (${data.connect_count})`;
|
|
||||||
}, [data, LL]);
|
|
||||||
|
|
||||||
if (!data) {
|
const mqttStatus = ({ enabled, connected, connect_count }: MqttStatusType) => {
|
||||||
return (
|
if (!enabled) {
|
||||||
<SectionContent>
|
return LL.NOT_ENABLED();
|
||||||
<FormLoader onRetry={loadData} errorMessage={errorMessage} />
|
}
|
||||||
</SectionContent>
|
if (connected) {
|
||||||
|
return LL.CONNECTED(0) + ' (' + connect_count + ')';
|
||||||
|
}
|
||||||
|
return LL.DISCONNECTED() + ' (' + connect_count + ')';
|
||||||
|
};
|
||||||
|
|
||||||
|
const disconnectReason = ({ disconnect_reason }: MqttStatusType) => {
|
||||||
|
switch (disconnect_reason) {
|
||||||
|
case MqttDisconnectReason.TCP_DISCONNECTED:
|
||||||
|
return 'TCP disconnected';
|
||||||
|
case MqttDisconnectReason.MQTT_UNACCEPTABLE_PROTOCOL_VERSION:
|
||||||
|
return 'Unacceptable protocol version';
|
||||||
|
case MqttDisconnectReason.MQTT_IDENTIFIER_REJECTED:
|
||||||
|
return 'Client ID rejected';
|
||||||
|
case MqttDisconnectReason.MQTT_SERVER_UNAVAILABLE:
|
||||||
|
return 'Server unavailable';
|
||||||
|
case MqttDisconnectReason.MQTT_MALFORMED_CREDENTIALS:
|
||||||
|
return 'Malformed credentials';
|
||||||
|
case MqttDisconnectReason.MQTT_NOT_AUTHORIZED:
|
||||||
|
return 'Not authorized';
|
||||||
|
case MqttDisconnectReason.TLS_BAD_FINGERPRINT:
|
||||||
|
return 'TLS fingerprint invalid';
|
||||||
|
default:
|
||||||
|
return 'Unknown';
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const content = () => {
|
||||||
|
if (!data) {
|
||||||
|
return <FormLoader onRetry={loadData} errorMessage={error?.message} />;
|
||||||
|
}
|
||||||
|
|
||||||
|
const renderConnectionStatus = () => (
|
||||||
|
<>
|
||||||
|
{!data.connected && (
|
||||||
|
<>
|
||||||
|
<ListItem>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar>
|
||||||
|
<ReportIcon />
|
||||||
|
</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText
|
||||||
|
primary={LL.DISCONNECT_REASON()}
|
||||||
|
secondary={disconnectReason(data)}
|
||||||
|
/>
|
||||||
|
</ListItem>
|
||||||
|
<Divider variant="inset" component="li" />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
<ListItem>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar>#</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText primary={LL.ID_OF(LL.CLIENT())} secondary={data.client_id} />
|
||||||
|
</ListItem>
|
||||||
|
<Divider variant="inset" component="li" />
|
||||||
|
<ListItem>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar sx={{ bgcolor: mqttQueueHighlight(data, theme) }}>
|
||||||
|
<AutoAwesomeMotionIcon />
|
||||||
|
</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText primary={LL.MQTT_QUEUE()} secondary={data.mqtt_queued} />
|
||||||
|
</ListItem>
|
||||||
|
<Divider variant="inset" component="li" />
|
||||||
|
<ListItem>
|
||||||
|
<ListItemAvatar>
|
||||||
|
<Avatar sx={{ bgcolor: mqttPublishHighlight(data, theme) }}>
|
||||||
|
<SpeakerNotesOffIcon />
|
||||||
|
</Avatar>
|
||||||
|
</ListItemAvatar>
|
||||||
|
<ListItemText primary={LL.ERRORS_OF('MQTT')} secondary={data.mqtt_fails} />
|
||||||
|
</ListItem>
|
||||||
|
<Divider variant="inset" component="li" />
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SectionContent>
|
|
||||||
<List>
|
<List>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
@@ -155,13 +156,15 @@ const MqttStatus = () => {
|
|||||||
<DeviceHubIcon />
|
<DeviceHubIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText primary={LL.STATUS_OF('')} secondary={mqttStatusText} />
|
<ListItemText primary={LL.STATUS_OF('')} secondary={mqttStatus(data)} />
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
{data.enabled && <ConnectionStatus data={data} theme={theme} />}
|
{data.enabled && renderConnectionStatus()}
|
||||||
</List>
|
</List>
|
||||||
</SectionContent>
|
);
|
||||||
);
|
};
|
||||||
|
|
||||||
|
return <SectionContent>{content()}</SectionContent>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default MqttStatus;
|
export default MqttStatus;
|
||||||
|
|||||||
@@ -1,46 +1,40 @@
|
|||||||
import { useMemo } from 'react';
|
import { useState } from 'react';
|
||||||
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
||||||
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import DnsIcon from '@mui/icons-material/Dns';
|
import DnsIcon from '@mui/icons-material/Dns';
|
||||||
import SwapVerticalCircleIcon from '@mui/icons-material/SwapVerticalCircle';
|
import SwapVerticalCircleIcon from '@mui/icons-material/SwapVerticalCircle';
|
||||||
import UpdateIcon from '@mui/icons-material/Update';
|
import UpdateIcon from '@mui/icons-material/Update';
|
||||||
import {
|
import {
|
||||||
Avatar,
|
Avatar,
|
||||||
|
Box,
|
||||||
|
Button,
|
||||||
|
Dialog,
|
||||||
|
DialogActions,
|
||||||
|
DialogContent,
|
||||||
|
DialogTitle,
|
||||||
Divider,
|
Divider,
|
||||||
List,
|
List,
|
||||||
ListItem,
|
ListItem,
|
||||||
ListItemAvatar,
|
ListItemAvatar,
|
||||||
ListItemText,
|
ListItemText,
|
||||||
|
TextField,
|
||||||
|
Typography,
|
||||||
useTheme
|
useTheme
|
||||||
} from '@mui/material';
|
} from '@mui/material';
|
||||||
import type { Theme } from '@mui/material';
|
import type { Theme } from '@mui/material';
|
||||||
|
|
||||||
import * as NTPApi from 'api/ntp';
|
import * as NTPApi from 'api/ntp';
|
||||||
|
|
||||||
|
import { dialogStyle } from 'CustomTheme';
|
||||||
import { useRequest } from 'alova/client';
|
import { useRequest } from 'alova/client';
|
||||||
import { FormLoader, SectionContent, useLayoutTitle } from 'components';
|
import { ButtonRow, FormLoader, SectionContent, useLayoutTitle } from 'components';
|
||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
import type { NTPStatusType } from 'types';
|
import type { NTPStatusType, Time } from 'types';
|
||||||
import { NTPSyncStatus } from 'types';
|
import { NTPSyncStatus } from 'types';
|
||||||
import { useInterval } from 'utils';
|
import { useInterval } from 'utils';
|
||||||
import { formatDateTime } from 'utils';
|
import { formatDateTime, formatLocalDateTime } from 'utils';
|
||||||
|
|
||||||
// Utility functions
|
|
||||||
const isNtpEnabled = ({ status }: NTPStatusType) =>
|
|
||||||
status !== NTPSyncStatus.NTP_DISABLED;
|
|
||||||
|
|
||||||
const ntpStatusHighlight = ({ status }: NTPStatusType, theme: Theme) => {
|
|
||||||
switch (status) {
|
|
||||||
case NTPSyncStatus.NTP_DISABLED:
|
|
||||||
return theme.palette.info.main;
|
|
||||||
case NTPSyncStatus.NTP_INACTIVE:
|
|
||||||
return theme.palette.error.main;
|
|
||||||
case NTPSyncStatus.NTP_ACTIVE:
|
|
||||||
return theme.palette.success.main;
|
|
||||||
default:
|
|
||||||
return theme.palette.error.main;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const NTPStatus = () => {
|
const NTPStatus = () => {
|
||||||
const { data, send: loadData, error } = useRequest(NTPApi.readNTPStatus);
|
const { data, send: loadData, error } = useRequest(NTPApi.readNTPStatus);
|
||||||
@@ -49,9 +43,48 @@ const NTPStatus = () => {
|
|||||||
void loadData();
|
void loadData();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const [localTime, setLocalTime] = useState<string>('');
|
||||||
|
const [settingTime, setSettingTime] = useState<boolean>(false);
|
||||||
|
const [processing, setProcessing] = useState<boolean>(false);
|
||||||
|
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
useLayoutTitle('NTP');
|
useLayoutTitle('NTP');
|
||||||
|
|
||||||
|
const { send: updateTime } = useRequest(
|
||||||
|
(local_time: Time) => NTPApi.updateTime(local_time),
|
||||||
|
{
|
||||||
|
immediate: false
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
NTPApi.updateTime;
|
||||||
|
|
||||||
|
const isNtpActive = ({ status }: NTPStatusType) =>
|
||||||
|
status === NTPSyncStatus.NTP_ACTIVE;
|
||||||
|
const isNtpEnabled = ({ status }: NTPStatusType) =>
|
||||||
|
status !== NTPSyncStatus.NTP_DISABLED;
|
||||||
|
|
||||||
|
const ntpStatusHighlight = ({ status }: NTPStatusType, theme: Theme) => {
|
||||||
|
switch (status) {
|
||||||
|
case NTPSyncStatus.NTP_DISABLED:
|
||||||
|
return theme.palette.info.main;
|
||||||
|
case NTPSyncStatus.NTP_INACTIVE:
|
||||||
|
return theme.palette.error.main;
|
||||||
|
case NTPSyncStatus.NTP_ACTIVE:
|
||||||
|
return theme.palette.success.main;
|
||||||
|
default:
|
||||||
|
return theme.palette.error.main;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const updateLocalTime = (event: React.ChangeEvent<HTMLInputElement>) =>
|
||||||
|
setLocalTime(event.target.value);
|
||||||
|
|
||||||
|
const openSetTime = () => {
|
||||||
|
setLocalTime(formatLocalDateTime(new Date()));
|
||||||
|
setSettingTime(true);
|
||||||
|
};
|
||||||
|
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
|
||||||
const ntpStatus = ({ status }: NTPStatusType) => {
|
const ntpStatus = ({ status }: NTPStatusType) => {
|
||||||
@@ -67,64 +100,147 @@ const NTPStatus = () => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const content = useMemo(() => {
|
const configureTime = async () => {
|
||||||
|
setProcessing(true);
|
||||||
|
|
||||||
|
await updateTime({ local_time: formatLocalDateTime(new Date(localTime)) })
|
||||||
|
.then(async () => {
|
||||||
|
toast.success(LL.TIME_SET());
|
||||||
|
setSettingTime(false);
|
||||||
|
await loadData();
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
toast.error(LL.PROBLEM_UPDATING());
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
setProcessing(false);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const renderSetTimeDialog = () => (
|
||||||
|
<Dialog
|
||||||
|
sx={dialogStyle}
|
||||||
|
open={settingTime}
|
||||||
|
onClose={() => setSettingTime(false)}
|
||||||
|
>
|
||||||
|
<DialogTitle>{LL.SET_TIME(1)}</DialogTitle>
|
||||||
|
<DialogContent dividers>
|
||||||
|
<Box color="warning.main" p={0} pl={0} pr={0} mt={0} mb={2}>
|
||||||
|
<Typography variant="body2">{LL.SET_TIME_TEXT()}</Typography>
|
||||||
|
</Box>
|
||||||
|
<TextField
|
||||||
|
label={LL.LOCAL_TIME(0)}
|
||||||
|
type="datetime-local"
|
||||||
|
value={localTime}
|
||||||
|
onChange={updateLocalTime}
|
||||||
|
disabled={processing}
|
||||||
|
fullWidth
|
||||||
|
slotProps={{
|
||||||
|
inputLabel: {
|
||||||
|
shrink: true
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</DialogContent>
|
||||||
|
<DialogActions>
|
||||||
|
<Button
|
||||||
|
startIcon={<CancelIcon />}
|
||||||
|
variant="outlined"
|
||||||
|
onClick={() => setSettingTime(false)}
|
||||||
|
color="secondary"
|
||||||
|
>
|
||||||
|
{LL.CANCEL()}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
startIcon={<AccessTimeIcon />}
|
||||||
|
variant="outlined"
|
||||||
|
onClick={configureTime}
|
||||||
|
disabled={processing}
|
||||||
|
color="primary"
|
||||||
|
>
|
||||||
|
{LL.UPDATE()}
|
||||||
|
</Button>
|
||||||
|
</DialogActions>
|
||||||
|
</Dialog>
|
||||||
|
);
|
||||||
|
|
||||||
|
const content = () => {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
return <FormLoader onRetry={loadData} errorMessage={error?.message} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<List>
|
<>
|
||||||
<ListItem>
|
<List>
|
||||||
<ListItemAvatar>
|
<ListItem>
|
||||||
<Avatar sx={{ bgcolor: ntpStatusHighlight(data, theme) }}>
|
<ListItemAvatar>
|
||||||
<UpdateIcon />
|
<Avatar sx={{ bgcolor: ntpStatusHighlight(data, theme) }}>
|
||||||
</Avatar>
|
<UpdateIcon />
|
||||||
</ListItemAvatar>
|
</Avatar>
|
||||||
<ListItemText primary={LL.STATUS_OF('')} secondary={ntpStatus(data)} />
|
</ListItemAvatar>
|
||||||
</ListItem>
|
<ListItemText primary={LL.STATUS_OF('')} secondary={ntpStatus(data)} />
|
||||||
<Divider variant="inset" component="li" />
|
</ListItem>
|
||||||
{isNtpEnabled(data) && (
|
<Divider variant="inset" component="li" />
|
||||||
<>
|
{isNtpEnabled(data) && (
|
||||||
<ListItem>
|
<>
|
||||||
<ListItemAvatar>
|
<ListItem>
|
||||||
<Avatar>
|
<ListItemAvatar>
|
||||||
<DnsIcon />
|
<Avatar>
|
||||||
</Avatar>
|
<DnsIcon />
|
||||||
</ListItemAvatar>
|
</Avatar>
|
||||||
<ListItemText primary={LL.NTP_SERVER()} secondary={data.server} />
|
</ListItemAvatar>
|
||||||
</ListItem>
|
<ListItemText primary={LL.NTP_SERVER()} secondary={data.server} />
|
||||||
<Divider variant="inset" component="li" />
|
</ListItem>
|
||||||
</>
|
<Divider variant="inset" component="li" />
|
||||||
)}
|
</>
|
||||||
<ListItem>
|
)}
|
||||||
<ListItemAvatar>
|
<ListItem>
|
||||||
<Avatar>
|
<ListItemAvatar>
|
||||||
<AccessTimeIcon />
|
<Avatar>
|
||||||
</Avatar>
|
<AccessTimeIcon />
|
||||||
</ListItemAvatar>
|
</Avatar>
|
||||||
<ListItemText
|
</ListItemAvatar>
|
||||||
primary={LL.LOCAL_TIME(0)}
|
<ListItemText
|
||||||
secondary={formatDateTime(data.local_time)}
|
primary={LL.LOCAL_TIME(0)}
|
||||||
/>
|
secondary={formatDateTime(data.local_time)}
|
||||||
</ListItem>
|
/>
|
||||||
<Divider variant="inset" component="li" />
|
</ListItem>
|
||||||
<ListItem>
|
<Divider variant="inset" component="li" />
|
||||||
<ListItemAvatar>
|
<ListItem>
|
||||||
<Avatar>
|
<ListItemAvatar>
|
||||||
<SwapVerticalCircleIcon />
|
<Avatar>
|
||||||
</Avatar>
|
<SwapVerticalCircleIcon />
|
||||||
</ListItemAvatar>
|
</Avatar>
|
||||||
<ListItemText
|
</ListItemAvatar>
|
||||||
primary={LL.UTC_TIME()}
|
<ListItemText
|
||||||
secondary={formatDateTime(data.utc_time)}
|
primary={LL.UTC_TIME()}
|
||||||
/>
|
secondary={formatDateTime(data.utc_time)}
|
||||||
</ListItem>
|
/>
|
||||||
<Divider variant="inset" component="li" />
|
</ListItem>
|
||||||
</List>
|
<Divider variant="inset" component="li" />
|
||||||
|
</List>
|
||||||
|
<Box display="flex" flexWrap="wrap">
|
||||||
|
{data && !isNtpActive(data) && (
|
||||||
|
<Box flexWrap="nowrap" whiteSpace="nowrap">
|
||||||
|
<ButtonRow>
|
||||||
|
<Button
|
||||||
|
onClick={openSetTime}
|
||||||
|
variant="outlined"
|
||||||
|
color="primary"
|
||||||
|
startIcon={<AccessTimeIcon />}
|
||||||
|
>
|
||||||
|
{LL.SET_TIME(0)}
|
||||||
|
</Button>
|
||||||
|
</ButtonRow>
|
||||||
|
</Box>
|
||||||
|
)}
|
||||||
|
</Box>
|
||||||
|
{renderSetTimeDialog()}
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
}, [data, error, loadData, LL, theme]);
|
};
|
||||||
|
|
||||||
return <SectionContent>{content}</SectionContent>;
|
return <SectionContent>{content()}</SectionContent>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default NTPStatus;
|
export default NTPStatus;
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
import { useMemo } from 'react';
|
|
||||||
|
|
||||||
import DeviceHubIcon from '@mui/icons-material/DeviceHub';
|
import DeviceHubIcon from '@mui/icons-material/DeviceHub';
|
||||||
import DnsIcon from '@mui/icons-material/Dns';
|
import DnsIcon from '@mui/icons-material/Dns';
|
||||||
import GiteIcon from '@mui/icons-material/Gite';
|
import GiteIcon from '@mui/icons-material/Gite';
|
||||||
@@ -27,17 +25,10 @@ import type { NetworkStatusType } from 'types';
|
|||||||
import { NetworkConnectionStatus } from 'types';
|
import { NetworkConnectionStatus } from 'types';
|
||||||
import { useInterval } from 'utils';
|
import { useInterval } from 'utils';
|
||||||
|
|
||||||
// Utility functions
|
|
||||||
const isConnected = ({ status }: NetworkStatusType) =>
|
const isConnected = ({ status }: NetworkStatusType) =>
|
||||||
status === NetworkConnectionStatus.WIFI_STATUS_CONNECTED ||
|
status === NetworkConnectionStatus.WIFI_STATUS_CONNECTED ||
|
||||||
status === NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED;
|
status === NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED;
|
||||||
|
|
||||||
export const isWiFi = ({ status }: NetworkStatusType) =>
|
|
||||||
status === NetworkConnectionStatus.WIFI_STATUS_CONNECTED;
|
|
||||||
|
|
||||||
export const isEthernet = ({ status }: NetworkStatusType) =>
|
|
||||||
status === NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED;
|
|
||||||
|
|
||||||
const networkStatusHighlight = ({ status }: NetworkStatusType, theme: Theme) => {
|
const networkStatusHighlight = ({ status }: NetworkStatusType, theme: Theme) => {
|
||||||
switch (status) {
|
switch (status) {
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_IDLE:
|
case NetworkConnectionStatus.WIFI_STATUS_IDLE:
|
||||||
@@ -64,6 +55,11 @@ const networkQualityHighlight = ({ rssi }: NetworkStatusType, theme: Theme) => {
|
|||||||
return theme.palette.success.main;
|
return theme.palette.success.main;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const isWiFi = ({ status }: NetworkStatusType) =>
|
||||||
|
status === NetworkConnectionStatus.WIFI_STATUS_CONNECTED;
|
||||||
|
export const isEthernet = ({ status }: NetworkStatusType) =>
|
||||||
|
status === NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED;
|
||||||
|
|
||||||
const dnsServers = ({ dns_ip_1, dns_ip_2 }: NetworkStatusType) => {
|
const dnsServers = ({ dns_ip_1, dns_ip_2 }: NetworkStatusType) => {
|
||||||
if (!dns_ip_1) {
|
if (!dns_ip_1) {
|
||||||
return 'none';
|
return 'none';
|
||||||
@@ -85,33 +81,6 @@ const IPs = (status: NetworkStatusType) => {
|
|||||||
return status.local_ip + ', ' + status.local_ipv6;
|
return status.local_ip + ', ' + status.local_ipv6;
|
||||||
};
|
};
|
||||||
|
|
||||||
const getNetworkStatusText = (
|
|
||||||
status: NetworkConnectionStatus,
|
|
||||||
reconnectCount: number,
|
|
||||||
LL: ReturnType<typeof useI18nContext>['LL']
|
|
||||||
) => {
|
|
||||||
switch (status) {
|
|
||||||
case NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED:
|
|
||||||
return LL.CONNECTED(0) + ' (Ethernet)';
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_NO_SHIELD:
|
|
||||||
return LL.INACTIVE(1);
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_IDLE:
|
|
||||||
return LL.IDLE();
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_NO_SSID_AVAIL:
|
|
||||||
return 'No SSID Available';
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_CONNECTED:
|
|
||||||
return LL.CONNECTED(0) + ' (WiFi) (' + reconnectCount + ')';
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_CONNECT_FAILED:
|
|
||||||
return LL.CONNECTED(1) + ' ' + LL.FAILED(0) + ' (' + reconnectCount + ')';
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_CONNECTION_LOST:
|
|
||||||
return LL.CONNECTED(1) + ' ' + LL.LOST() + ' (' + reconnectCount + ')';
|
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_DISCONNECTED:
|
|
||||||
return LL.DISCONNECTED();
|
|
||||||
default:
|
|
||||||
return LL.UNKNOWN();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const NetworkStatus = () => {
|
const NetworkStatus = () => {
|
||||||
const { data, send: loadData, error } = useRequest(NetworkApi.readNetworkStatus);
|
const { data, send: loadData, error } = useRequest(NetworkApi.readNetworkStatus);
|
||||||
|
|
||||||
@@ -124,30 +93,51 @@ const NetworkStatus = () => {
|
|||||||
|
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
|
||||||
const content = useMemo(() => {
|
const networkStatus = ({ status }: NetworkStatusType) => {
|
||||||
if (!data) {
|
switch (status) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
case NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED:
|
||||||
|
return LL.CONNECTED(0) + ' (Ethernet)';
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_NO_SHIELD:
|
||||||
|
return LL.INACTIVE(1);
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_IDLE:
|
||||||
|
return LL.IDLE();
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_NO_SSID_AVAIL:
|
||||||
|
return 'No SSID Available';
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_CONNECTED:
|
||||||
|
return LL.CONNECTED(0) + ' (WiFi) (' + data.reconnect_count + ')';
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_CONNECT_FAILED:
|
||||||
|
return (
|
||||||
|
LL.CONNECTED(1) + ' ' + LL.FAILED(0) + ' (' + data.reconnect_count + ')'
|
||||||
|
);
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_CONNECTION_LOST:
|
||||||
|
return LL.CONNECTED(1) + ' ' + LL.LOST() + ' (' + data.reconnect_count + ')';
|
||||||
|
case NetworkConnectionStatus.WIFI_STATUS_DISCONNECTED:
|
||||||
|
return LL.DISCONNECTED();
|
||||||
|
default:
|
||||||
|
return LL.UNKNOWN();
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
const statusText = getNetworkStatusText(data.status, data.reconnect_count, LL);
|
const content = () => {
|
||||||
const statusColor = networkStatusHighlight(data, theme);
|
if (!data) {
|
||||||
const qualityColor = networkQualityHighlight(data, theme);
|
return <FormLoader onRetry={loadData} errorMessage={error?.message} />;
|
||||||
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<List>
|
<List>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar sx={{ bgcolor: statusColor }}>
|
<Avatar sx={{ bgcolor: networkStatusHighlight(data, theme) }}>
|
||||||
{isWiFi(data) && <WifiIcon />}
|
{isWiFi(data) && <WifiIcon />}
|
||||||
{isEthernet(data) && <RouterIcon />}
|
{isEthernet(data) && <RouterIcon />}
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText primary="Status" secondary={statusText} />
|
<ListItemText primary="Status" secondary={networkStatus(data)} />
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar sx={{ bgcolor: statusColor }}>
|
<Avatar sx={{ bgcolor: networkStatusHighlight(data, theme) }}>
|
||||||
<GiteIcon />
|
<GiteIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
@@ -158,13 +148,13 @@ const NetworkStatus = () => {
|
|||||||
<>
|
<>
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar sx={{ bgcolor: qualityColor }}>
|
<Avatar sx={{ bgcolor: networkQualityHighlight(data, theme) }}>
|
||||||
<SettingsInputAntennaIcon />
|
<SettingsInputAntennaIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText
|
<ListItemText
|
||||||
primary="SSID (RSSI)"
|
primary="SSID (RSSI)"
|
||||||
secondary={`${data.ssid} (${data.rssi} dBm)`}
|
secondary={data.ssid + ' (' + data.rssi + ' dBm)'}
|
||||||
/>
|
/>
|
||||||
</ListItem>
|
</ListItem>
|
||||||
<Divider variant="inset" component="li" />
|
<Divider variant="inset" component="li" />
|
||||||
@@ -228,9 +218,9 @@ const NetworkStatus = () => {
|
|||||||
)}
|
)}
|
||||||
</List>
|
</List>
|
||||||
);
|
);
|
||||||
}, [data, error, loadData, LL, theme]);
|
};
|
||||||
|
|
||||||
return <SectionContent>{content}</SectionContent>;
|
return <SectionContent>{content()}</SectionContent>;
|
||||||
};
|
};
|
||||||
|
|
||||||
export default NetworkStatus;
|
export default NetworkStatus;
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useContext, useMemo, useState } from 'react';
|
import { useContext, useState } from 'react';
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
import AccessTimeIcon from '@mui/icons-material/AccessTime';
|
||||||
@@ -8,10 +8,10 @@ import DeviceHubIcon from '@mui/icons-material/DeviceHub';
|
|||||||
import DirectionsBusIcon from '@mui/icons-material/DirectionsBus';
|
import DirectionsBusIcon from '@mui/icons-material/DirectionsBus';
|
||||||
import LogoDevIcon from '@mui/icons-material/LogoDev';
|
import LogoDevIcon from '@mui/icons-material/LogoDev';
|
||||||
import MemoryIcon from '@mui/icons-material/Memory';
|
import MemoryIcon from '@mui/icons-material/Memory';
|
||||||
import MonitorHeartIcon from '@mui/icons-material/MonitorHeart';
|
|
||||||
import PowerSettingsNewIcon from '@mui/icons-material/PowerSettingsNew';
|
import PowerSettingsNewIcon from '@mui/icons-material/PowerSettingsNew';
|
||||||
import RouterIcon from '@mui/icons-material/Router';
|
import RouterIcon from '@mui/icons-material/Router';
|
||||||
import SettingsInputAntennaIcon from '@mui/icons-material/SettingsInputAntenna';
|
import SettingsInputAntennaIcon from '@mui/icons-material/SettingsInputAntenna';
|
||||||
|
import TimerIcon from '@mui/icons-material/Timer';
|
||||||
import WifiIcon from '@mui/icons-material/Wifi';
|
import WifiIcon from '@mui/icons-material/Wifi';
|
||||||
import {
|
import {
|
||||||
Avatar,
|
Avatar,
|
||||||
@@ -37,34 +37,12 @@ import { FormLoader, SectionContent, useLayoutTitle } from 'components';
|
|||||||
import ListMenuItem from 'components/layout/ListMenuItem';
|
import ListMenuItem from 'components/layout/ListMenuItem';
|
||||||
import { AuthenticatedContext } from 'contexts/authentication';
|
import { AuthenticatedContext } from 'contexts/authentication';
|
||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
import { NTPSyncStatus, NetworkConnectionStatus, SystemStatusCodes } from 'types';
|
import { NTPSyncStatus, NetworkConnectionStatus } from 'types';
|
||||||
import { useInterval } from 'utils';
|
import { useInterval } from 'utils';
|
||||||
import { formatDateTime } from 'utils/time';
|
import { formatDateTime } from 'utils/time';
|
||||||
|
|
||||||
import SystemMonitor from './SystemMonitor';
|
import SystemMonitor from './SystemMonitor';
|
||||||
|
|
||||||
// Pure functions moved outside component to avoid recreation on each render
|
|
||||||
const formatNumber = (num: number) => new Intl.NumberFormat().format(num);
|
|
||||||
|
|
||||||
const formatDurationSec = (
|
|
||||||
duration_sec: number,
|
|
||||||
LL: ReturnType<typeof useI18nContext>['LL']
|
|
||||||
) => {
|
|
||||||
const ms = duration_sec * 1000;
|
|
||||||
const days = Math.trunc(ms / 86400000);
|
|
||||||
const hours = Math.trunc(ms / 3600000) % 24;
|
|
||||||
const minutes = Math.trunc(ms / 60000) % 60;
|
|
||||||
const seconds = Math.trunc(ms / 1000) % 60;
|
|
||||||
|
|
||||||
const parts: string[] = [];
|
|
||||||
if (days) parts.push(LL.NUM_DAYS({ num: days }));
|
|
||||||
if (hours) parts.push(LL.NUM_HOURS({ num: hours }));
|
|
||||||
if (minutes) parts.push(LL.NUM_MINUTES({ num: minutes }));
|
|
||||||
parts.push(LL.NUM_SECONDS({ num: seconds }));
|
|
||||||
|
|
||||||
return parts.join(' ');
|
|
||||||
};
|
|
||||||
|
|
||||||
const SystemStatus = () => {
|
const SystemStatus = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
@@ -84,6 +62,7 @@ const SystemStatus = () => {
|
|||||||
send: loadData,
|
send: loadData,
|
||||||
error
|
error
|
||||||
} = useRequest(readSystemStatus, {
|
} = useRequest(readSystemStatus, {
|
||||||
|
initialData: [],
|
||||||
async middleware(_, next) {
|
async middleware(_, next) {
|
||||||
if (!restarting) {
|
if (!restarting) {
|
||||||
await next();
|
await next();
|
||||||
@@ -97,46 +76,51 @@ const SystemStatus = () => {
|
|||||||
|
|
||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
|
||||||
// Memoize derived status values to avoid recalculation on every render
|
const formatDurationSec = (duration_sec: number) => {
|
||||||
const busStatus = useMemo(() => {
|
const days = Math.trunc((duration_sec * 1000) / 86400000);
|
||||||
if (!data) return 'EMS state unknown';
|
const hours = Math.trunc((duration_sec * 1000) / 3600000) % 24;
|
||||||
|
const minutes = Math.trunc((duration_sec * 1000) / 60000) % 60;
|
||||||
|
const seconds = Math.trunc((duration_sec * 1000) / 1000) % 60;
|
||||||
|
|
||||||
switch (data.bus_status) {
|
let formatted = '';
|
||||||
case busConnectionStatus.BUS_STATUS_CONNECTED:
|
if (days) {
|
||||||
return `EMS ${LL.CONNECTED(0)} (${formatDurationSec(data.bus_uptime, LL)})`;
|
formatted += LL.NUM_DAYS({ num: days }) + ' ';
|
||||||
case busConnectionStatus.BUS_STATUS_TX_ERRORS:
|
|
||||||
return 'EMS ' + LL.TX_ISSUES();
|
|
||||||
case busConnectionStatus.BUS_STATUS_OFFLINE:
|
|
||||||
return 'EMS ' + LL.DISCONNECTED();
|
|
||||||
default:
|
|
||||||
return 'EMS state unknown';
|
|
||||||
}
|
}
|
||||||
}, [data?.bus_status, data?.bus_uptime, LL]);
|
if (hours) {
|
||||||
|
formatted += LL.NUM_HOURS({ num: hours }) + ' ';
|
||||||
// Memoize derived status values to avoid recalculation on every render
|
|
||||||
const systemStatus = useMemo(() => {
|
|
||||||
if (!data) return '??';
|
|
||||||
|
|
||||||
switch (data.status) {
|
|
||||||
case SystemStatusCodes.SYSTEM_STATUS_PENDING_UPLOAD:
|
|
||||||
case SystemStatusCodes.SYSTEM_STATUS_UPLOADING:
|
|
||||||
return LL.WAIT_FIRMWARE();
|
|
||||||
case SystemStatusCodes.SYSTEM_STATUS_ERROR_UPLOAD:
|
|
||||||
return LL.ERROR();
|
|
||||||
case SystemStatusCodes.SYSTEM_STATUS_PENDING_RESTART:
|
|
||||||
case SystemStatusCodes.SYSTEM_STATUS_RESTART_REQUESTED:
|
|
||||||
return LL.RESTARTING_PRE();
|
|
||||||
case SystemStatusCodes.SYSTEM_STATUS_INVALID_GPIO:
|
|
||||||
return LL.GPIO_OF(LL.FAILED(0));
|
|
||||||
default:
|
|
||||||
// SystemStatusCodes.SYSTEM_STATUS_NORMAL
|
|
||||||
return 'OK';
|
|
||||||
}
|
}
|
||||||
}, [data?.status, LL]);
|
if (minutes) {
|
||||||
|
formatted += LL.NUM_MINUTES({ num: minutes }) + ' ';
|
||||||
|
}
|
||||||
|
formatted += LL.NUM_SECONDS({ num: seconds });
|
||||||
|
return formatted;
|
||||||
|
};
|
||||||
|
|
||||||
const busStatusHighlight = useMemo(() => {
|
function formatNumber(num: number) {
|
||||||
if (!data) return theme.palette.warning.main;
|
return new Intl.NumberFormat().format(num);
|
||||||
|
}
|
||||||
|
|
||||||
|
const busStatus = () => {
|
||||||
|
if (data) {
|
||||||
|
switch (data.bus_status) {
|
||||||
|
case busConnectionStatus.BUS_STATUS_CONNECTED:
|
||||||
|
return (
|
||||||
|
'EMS ' +
|
||||||
|
LL.CONNECTED(0) +
|
||||||
|
' (' +
|
||||||
|
formatDurationSec(data.bus_uptime) +
|
||||||
|
')'
|
||||||
|
);
|
||||||
|
case busConnectionStatus.BUS_STATUS_TX_ERRORS:
|
||||||
|
return 'EMS ' + LL.TX_ISSUES();
|
||||||
|
case busConnectionStatus.BUS_STATUS_OFFLINE:
|
||||||
|
return 'EMS ' + LL.DISCONNECTED();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 'EMS state unknown';
|
||||||
|
};
|
||||||
|
|
||||||
|
const busStatusHighlight = () => {
|
||||||
switch (data.bus_status) {
|
switch (data.bus_status) {
|
||||||
case busConnectionStatus.BUS_STATUS_TX_ERRORS:
|
case busConnectionStatus.BUS_STATUS_TX_ERRORS:
|
||||||
return theme.palette.warning.main;
|
return theme.palette.warning.main;
|
||||||
@@ -147,28 +131,27 @@ const SystemStatus = () => {
|
|||||||
default:
|
default:
|
||||||
return theme.palette.warning.main;
|
return theme.palette.warning.main;
|
||||||
}
|
}
|
||||||
}, [data?.bus_status, theme.palette]);
|
};
|
||||||
|
|
||||||
const ntpStatus = useMemo(() => {
|
|
||||||
if (!data) return LL.UNKNOWN();
|
|
||||||
|
|
||||||
|
const ntpStatus = () => {
|
||||||
switch (data.ntp_status) {
|
switch (data.ntp_status) {
|
||||||
case NTPSyncStatus.NTP_DISABLED:
|
case NTPSyncStatus.NTP_DISABLED:
|
||||||
return LL.NOT_ENABLED();
|
return LL.NOT_ENABLED();
|
||||||
case NTPSyncStatus.NTP_INACTIVE:
|
case NTPSyncStatus.NTP_INACTIVE:
|
||||||
return LL.INACTIVE(0);
|
return LL.INACTIVE(0);
|
||||||
case NTPSyncStatus.NTP_ACTIVE:
|
case NTPSyncStatus.NTP_ACTIVE:
|
||||||
return data.ntp_time
|
return (
|
||||||
? `${LL.ACTIVE()} (${formatDateTime(data.ntp_time)})`
|
LL.ACTIVE() +
|
||||||
: LL.ACTIVE();
|
(data.ntp_time !== undefined
|
||||||
|
? ' (' + formatDateTime(data.ntp_time) + ')'
|
||||||
|
: '')
|
||||||
|
);
|
||||||
default:
|
default:
|
||||||
return LL.UNKNOWN();
|
return LL.UNKNOWN();
|
||||||
}
|
}
|
||||||
}, [data?.ntp_status, data?.ntp_time, LL]);
|
};
|
||||||
|
|
||||||
const ntpStatusHighlight = useMemo(() => {
|
|
||||||
if (!data) return theme.palette.error.main;
|
|
||||||
|
|
||||||
|
const ntpStatusHighlight = () => {
|
||||||
switch (data.ntp_status) {
|
switch (data.ntp_status) {
|
||||||
case NTPSyncStatus.NTP_DISABLED:
|
case NTPSyncStatus.NTP_DISABLED:
|
||||||
return theme.palette.info.main;
|
return theme.palette.info.main;
|
||||||
@@ -179,11 +162,9 @@ const SystemStatus = () => {
|
|||||||
default:
|
default:
|
||||||
return theme.palette.error.main;
|
return theme.palette.error.main;
|
||||||
}
|
}
|
||||||
}, [data?.ntp_status, theme.palette]);
|
};
|
||||||
|
|
||||||
const networkStatusHighlight = useMemo(() => {
|
|
||||||
if (!data) return theme.palette.warning.main;
|
|
||||||
|
|
||||||
|
const networkStatusHighlight = () => {
|
||||||
switch (data.network_status) {
|
switch (data.network_status) {
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_IDLE:
|
case NetworkConnectionStatus.WIFI_STATUS_IDLE:
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_DISCONNECTED:
|
case NetworkConnectionStatus.WIFI_STATUS_DISCONNECTED:
|
||||||
@@ -198,11 +179,9 @@ const SystemStatus = () => {
|
|||||||
default:
|
default:
|
||||||
return theme.palette.warning.main;
|
return theme.palette.warning.main;
|
||||||
}
|
}
|
||||||
}, [data?.network_status, theme.palette]);
|
};
|
||||||
|
|
||||||
const networkStatus = useMemo(() => {
|
|
||||||
if (!data) return LL.UNKNOWN();
|
|
||||||
|
|
||||||
|
const networkStatus = () => {
|
||||||
switch (data.network_status) {
|
switch (data.network_status) {
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_NO_SHIELD:
|
case NetworkConnectionStatus.WIFI_STATUS_NO_SHIELD:
|
||||||
return LL.INACTIVE(1);
|
return LL.INACTIVE(1);
|
||||||
@@ -211,27 +190,24 @@ const SystemStatus = () => {
|
|||||||
case NetworkConnectionStatus.WIFI_STATUS_NO_SSID_AVAIL:
|
case NetworkConnectionStatus.WIFI_STATUS_NO_SSID_AVAIL:
|
||||||
return 'No SSID Available';
|
return 'No SSID Available';
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_CONNECTED:
|
case NetworkConnectionStatus.WIFI_STATUS_CONNECTED:
|
||||||
return `${LL.CONNECTED(0)} (WiFi, ${data.wifi_rssi} dBm)`;
|
return LL.CONNECTED(0) + ' (WiFi, ' + data.wifi_rssi + ' dBm)';
|
||||||
case NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED:
|
case NetworkConnectionStatus.ETHERNET_STATUS_CONNECTED:
|
||||||
return `${LL.CONNECTED(0)} (Ethernet)`;
|
return LL.CONNECTED(0) + ' (Ethernet)';
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_CONNECT_FAILED:
|
case NetworkConnectionStatus.WIFI_STATUS_CONNECT_FAILED:
|
||||||
return `${LL.CONNECTED(1)} ${LL.FAILED(0)}`;
|
return LL.CONNECTED(1) + ' ' + LL.FAILED(0);
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_CONNECTION_LOST:
|
case NetworkConnectionStatus.WIFI_STATUS_CONNECTION_LOST:
|
||||||
return `${LL.CONNECTED(1)} ${LL.LOST()}`;
|
return LL.CONNECTED(1) + ' ' + LL.LOST();
|
||||||
case NetworkConnectionStatus.WIFI_STATUS_DISCONNECTED:
|
case NetworkConnectionStatus.WIFI_STATUS_DISCONNECTED:
|
||||||
return LL.DISCONNECTED();
|
return LL.DISCONNECTED();
|
||||||
default:
|
default:
|
||||||
return LL.UNKNOWN();
|
return LL.UNKNOWN();
|
||||||
}
|
}
|
||||||
}, [data?.network_status, data?.wifi_rssi, LL]);
|
};
|
||||||
|
|
||||||
const activeHighlight = useCallback(
|
const activeHighlight = (value: boolean) =>
|
||||||
(value: boolean) =>
|
value ? theme.palette.success.main : theme.palette.info.main;
|
||||||
value ? theme.palette.success.main : theme.palette.info.main,
|
|
||||||
[theme.palette]
|
|
||||||
);
|
|
||||||
|
|
||||||
const doRestart = useCallback(async () => {
|
const doRestart = async () => {
|
||||||
setConfirmRestart(false);
|
setConfirmRestart(false);
|
||||||
setRestarting(true);
|
setRestarting(true);
|
||||||
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
await sendAPI({ device: 'system', cmd: 'restart', id: 0 }).catch(
|
||||||
@@ -239,114 +215,69 @@ const SystemStatus = () => {
|
|||||||
toast.error(error.message);
|
toast.error(error.message);
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
}, [sendAPI]);
|
};
|
||||||
|
|
||||||
const handleCloseRestartDialog = useCallback(() => {
|
const renderRestartDialog = () => (
|
||||||
setConfirmRestart(false);
|
<Dialog
|
||||||
}, []);
|
sx={dialogStyle}
|
||||||
|
open={confirmRestart}
|
||||||
const renderRestartDialog = useMemo(
|
onClose={() => setConfirmRestart(false)}
|
||||||
() => (
|
>
|
||||||
<Dialog
|
<DialogTitle>{LL.RESTART()}</DialogTitle>
|
||||||
sx={dialogStyle}
|
<DialogContent dividers>{LL.RESTART_CONFIRM()}</DialogContent>
|
||||||
open={confirmRestart}
|
<DialogActions>
|
||||||
onClose={handleCloseRestartDialog}
|
<Button
|
||||||
>
|
startIcon={<CancelIcon />}
|
||||||
<DialogTitle>{LL.RESTART()}</DialogTitle>
|
variant="outlined"
|
||||||
<DialogContent dividers>{LL.RESTART_CONFIRM()}</DialogContent>
|
onClick={() => setConfirmRestart(false)}
|
||||||
<DialogActions>
|
color="secondary"
|
||||||
<Button
|
>
|
||||||
startIcon={<CancelIcon />}
|
{LL.CANCEL()}
|
||||||
variant="outlined"
|
</Button>
|
||||||
onClick={handleCloseRestartDialog}
|
<Button
|
||||||
color="secondary"
|
startIcon={<PowerSettingsNewIcon />}
|
||||||
>
|
variant="outlined"
|
||||||
{LL.CANCEL()}
|
onClick={doRestart}
|
||||||
</Button>
|
color="error"
|
||||||
<Button
|
>
|
||||||
startIcon={<PowerSettingsNewIcon />}
|
{LL.RESTART()}
|
||||||
variant="outlined"
|
</Button>
|
||||||
onClick={doRestart}
|
</DialogActions>
|
||||||
color="error"
|
</Dialog>
|
||||||
>
|
|
||||||
{LL.RESTART()}
|
|
||||||
</Button>
|
|
||||||
</DialogActions>
|
|
||||||
</Dialog>
|
|
||||||
),
|
|
||||||
[confirmRestart, handleCloseRestartDialog, doRestart, LL]
|
|
||||||
);
|
);
|
||||||
|
|
||||||
// Memoize formatted values
|
const content = () => {
|
||||||
const firmwareVersion = useMemo(
|
|
||||||
() => `v${data?.emsesp_version || ''}`,
|
|
||||||
[data?.emsesp_version]
|
|
||||||
);
|
|
||||||
|
|
||||||
const uptimeText = useMemo(
|
|
||||||
() => (data ? formatDurationSec(data.uptime, LL) : ''),
|
|
||||||
[data?.uptime, LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const freeMemoryText = useMemo(
|
|
||||||
() => (data ? `${formatNumber(data.free_heap)} KB ${LL.FREE_MEMORY()}` : ''),
|
|
||||||
[data?.free_heap, LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const networkIcon = useMemo(
|
|
||||||
() =>
|
|
||||||
data?.network_status === NetworkConnectionStatus.WIFI_STATUS_CONNECTED
|
|
||||||
? WifiIcon
|
|
||||||
: RouterIcon,
|
|
||||||
[data?.network_status]
|
|
||||||
);
|
|
||||||
|
|
||||||
const mqttStatusText = useMemo(
|
|
||||||
() => (data?.mqtt_status ? LL.CONNECTED(0) : LL.INACTIVE(0)),
|
|
||||||
[data?.mqtt_status, LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const apStatusText = useMemo(
|
|
||||||
() => (data?.ap_status ? LL.ACTIVE() : LL.INACTIVE(0)),
|
|
||||||
[data?.ap_status, LL]
|
|
||||||
);
|
|
||||||
|
|
||||||
const handleRestartClick = useCallback(() => {
|
|
||||||
setConfirmRestart(true);
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
const content = useMemo(() => {
|
|
||||||
if (!data || !LL) {
|
if (!data || !LL) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={error?.message || ''} />;
|
return <FormLoader onRetry={loadData} errorMessage={error?.message} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<List>
|
<List sx={{ borderRadius: 3, border: '2px solid grey' }}>
|
||||||
<ListMenuItem
|
<ListMenuItem
|
||||||
icon={BuildIcon}
|
icon={BuildIcon}
|
||||||
bgcolor="#72caf9"
|
bgcolor="#72caf9"
|
||||||
label="EMS-ESP Firmware"
|
label="EMS-ESP Firmware"
|
||||||
text={firmwareVersion}
|
text={'v' + data.emsesp_version}
|
||||||
to="version"
|
to="version"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<ListItem>
|
<ListItem>
|
||||||
<ListItemAvatar>
|
<ListItemAvatar>
|
||||||
<Avatar sx={{ bgcolor: '#c5572c', color: 'white' }}>
|
<Avatar sx={{ bgcolor: '#c5572c', color: 'white' }}>
|
||||||
<MonitorHeartIcon />
|
<TimerIcon />
|
||||||
</Avatar>
|
</Avatar>
|
||||||
</ListItemAvatar>
|
</ListItemAvatar>
|
||||||
<ListItemText
|
<ListItemText
|
||||||
primary={LL.STATUS_OF(LL.SYSTEM(0))}
|
primary={LL.UPTIME()}
|
||||||
secondary={`${systemStatus} (${LL.UPTIME()}: ${uptimeText})`}
|
secondary={formatDurationSec(data.uptime)}
|
||||||
/>
|
/>
|
||||||
{me.admin && (
|
{me.admin && (
|
||||||
<Button
|
<Button
|
||||||
startIcon={<PowerSettingsNewIcon />}
|
startIcon={<PowerSettingsNewIcon />}
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
color="error"
|
color="error"
|
||||||
onClick={handleRestartClick}
|
onClick={() => setConfirmRestart(true)}
|
||||||
>
|
>
|
||||||
{LL.RESTART()}
|
{LL.RESTART()}
|
||||||
</Button>
|
</Button>
|
||||||
@@ -358,25 +289,29 @@ const SystemStatus = () => {
|
|||||||
icon={MemoryIcon}
|
icon={MemoryIcon}
|
||||||
bgcolor="#68374d"
|
bgcolor="#68374d"
|
||||||
label={LL.HARDWARE()}
|
label={LL.HARDWARE()}
|
||||||
text={freeMemoryText}
|
text={formatNumber(data.free_heap) + ' KB' + ' ' + LL.FREE_MEMORY()}
|
||||||
to="/status/hardwarestatus"
|
to="/status/hardwarestatus"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<ListMenuItem
|
<ListMenuItem
|
||||||
disabled={!me.admin}
|
disabled={!me.admin}
|
||||||
icon={DirectionsBusIcon}
|
icon={DirectionsBusIcon}
|
||||||
bgcolor={busStatusHighlight}
|
bgcolor={busStatusHighlight()}
|
||||||
label={LL.DATA_TRAFFIC()}
|
label={LL.DATA_TRAFFIC()}
|
||||||
text={busStatus}
|
text={busStatus()}
|
||||||
to="/status/activity"
|
to="/status/activity"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<ListMenuItem
|
<ListMenuItem
|
||||||
disabled={!me.admin}
|
disabled={!me.admin}
|
||||||
icon={networkIcon}
|
icon={
|
||||||
bgcolor={networkStatusHighlight}
|
data.network_status === NetworkConnectionStatus.WIFI_STATUS_CONNECTED
|
||||||
|
? WifiIcon
|
||||||
|
: RouterIcon
|
||||||
|
}
|
||||||
|
bgcolor={networkStatusHighlight()}
|
||||||
label={LL.NETWORK(1)}
|
label={LL.NETWORK(1)}
|
||||||
text={networkStatus}
|
text={networkStatus()}
|
||||||
to="/status/network"
|
to="/status/network"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
@@ -385,16 +320,16 @@ const SystemStatus = () => {
|
|||||||
icon={DeviceHubIcon}
|
icon={DeviceHubIcon}
|
||||||
bgcolor={activeHighlight(data.mqtt_status)}
|
bgcolor={activeHighlight(data.mqtt_status)}
|
||||||
label="MQTT"
|
label="MQTT"
|
||||||
text={mqttStatusText}
|
text={data.mqtt_status ? LL.CONNECTED(0) : LL.INACTIVE(0)}
|
||||||
to="/status/mqtt"
|
to="/status/mqtt"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
<ListMenuItem
|
<ListMenuItem
|
||||||
disabled={!me.admin}
|
disabled={!me.admin}
|
||||||
icon={AccessTimeIcon}
|
icon={AccessTimeIcon}
|
||||||
bgcolor={ntpStatusHighlight}
|
bgcolor={ntpStatusHighlight()}
|
||||||
label="NTP"
|
label="NTP"
|
||||||
text={ntpStatus}
|
text={ntpStatus()}
|
||||||
to="/status/ntp"
|
to="/status/ntp"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
@@ -403,7 +338,7 @@ const SystemStatus = () => {
|
|||||||
icon={SettingsInputAntennaIcon}
|
icon={SettingsInputAntennaIcon}
|
||||||
bgcolor={activeHighlight(data.ap_status)}
|
bgcolor={activeHighlight(data.ap_status)}
|
||||||
label={LL.ACCESS_POINT(0)}
|
label={LL.ACCESS_POINT(0)}
|
||||||
text={apStatusText}
|
text={data.ap_status ? LL.ACTIVE() : LL.INACTIVE(0)}
|
||||||
to="/status/ap"
|
to="/status/ap"
|
||||||
/>
|
/>
|
||||||
|
|
||||||
@@ -417,33 +352,14 @@ const SystemStatus = () => {
|
|||||||
/>
|
/>
|
||||||
</List>
|
</List>
|
||||||
|
|
||||||
{renderRestartDialog}
|
{renderRestartDialog()}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}, [
|
};
|
||||||
data,
|
|
||||||
LL,
|
|
||||||
firmwareVersion,
|
|
||||||
uptimeText,
|
|
||||||
freeMemoryText,
|
|
||||||
networkIcon,
|
|
||||||
mqttStatusText,
|
|
||||||
apStatusText,
|
|
||||||
busStatus,
|
|
||||||
busStatusHighlight,
|
|
||||||
networkStatusHighlight,
|
|
||||||
networkStatus,
|
|
||||||
ntpStatusHighlight,
|
|
||||||
ntpStatus,
|
|
||||||
activeHighlight,
|
|
||||||
me.admin,
|
|
||||||
handleRestartClick,
|
|
||||||
error,
|
|
||||||
loadData,
|
|
||||||
renderRestartDialog
|
|
||||||
]);
|
|
||||||
|
|
||||||
return restarting ? <SystemMonitor /> : <SectionContent>{content}</SectionContent>;
|
return (
|
||||||
|
<SectionContent>{restarting ? <SystemMonitor /> : content()}</SectionContent>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default SystemStatus;
|
export default SystemStatus;
|
||||||
|
|||||||
@@ -1,11 +1,4 @@
|
|||||||
import {
|
import { useEffect, useRef, useState } from 'react';
|
||||||
memo,
|
|
||||||
useCallback,
|
|
||||||
useEffect,
|
|
||||||
useLayoutEffect,
|
|
||||||
useRef,
|
|
||||||
useState
|
|
||||||
} from 'react';
|
|
||||||
import { toast } from 'react-toastify';
|
import { toast } from 'react-toastify';
|
||||||
|
|
||||||
import DownloadIcon from '@mui/icons-material/GetApp';
|
import DownloadIcon from '@mui/icons-material/GetApp';
|
||||||
@@ -15,7 +8,7 @@ import {
|
|||||||
Box,
|
Box,
|
||||||
Button,
|
Button,
|
||||||
Checkbox,
|
Checkbox,
|
||||||
Grid,
|
Grid2 as Grid,
|
||||||
IconButton,
|
IconButton,
|
||||||
MenuItem,
|
MenuItem,
|
||||||
TextField,
|
TextField,
|
||||||
@@ -38,14 +31,13 @@ import type { LogEntry, LogSettings } from 'types';
|
|||||||
import { LogLevel } from 'types';
|
import { LogLevel } from 'types';
|
||||||
import { updateValueDirty, useRest } from 'utils';
|
import { updateValueDirty, useRest } from 'utils';
|
||||||
|
|
||||||
const TextColors: Record<LogLevel, string> = {
|
const TextColors = {
|
||||||
[LogLevel.ERROR]: '#ff0000', // red
|
[LogLevel.ERROR]: '#ff0000', // red
|
||||||
[LogLevel.WARNING]: '#ff0000', // red
|
[LogLevel.WARNING]: '#ff0000', // red
|
||||||
[LogLevel.NOTICE]: '#ffffff', // white
|
[LogLevel.NOTICE]: '#ffffff', // white
|
||||||
[LogLevel.INFO]: '#ffcc00', // yellow
|
[LogLevel.INFO]: '#ffcc00', // yellow
|
||||||
[LogLevel.DEBUG]: '#00ffff', // cyan
|
[LogLevel.DEBUG]: '#00ffff', // cyan
|
||||||
[LogLevel.TRACE]: '#00ffff', // cyan
|
[LogLevel.TRACE]: '#00ffff' // cyan
|
||||||
[LogLevel.ALL]: '#ffffff' // white
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const LogEntryLine = styled('span')(
|
const LogEntryLine = styled('span')(
|
||||||
@@ -54,6 +46,11 @@ const LogEntryLine = styled('span')(
|
|||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const topOffset = () =>
|
||||||
|
document.getElementById('log-window')?.getBoundingClientRect().bottom || 0;
|
||||||
|
const leftOffset = () =>
|
||||||
|
document.getElementById('log-window')?.getBoundingClientRect().left || 0;
|
||||||
|
|
||||||
const levelLabel = (level: LogLevel) => {
|
const levelLabel = (level: LogLevel) => {
|
||||||
switch (level) {
|
switch (level) {
|
||||||
case LogLevel.ERROR:
|
case LogLevel.ERROR:
|
||||||
@@ -73,39 +70,6 @@ const levelLabel = (level: LogLevel) => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const paddedLevelLabel = (level: LogLevel, compact: boolean) => {
|
|
||||||
const label = levelLabel(level);
|
|
||||||
return compact ? ' ' + label[0] : label.padStart(8, '\xa0');
|
|
||||||
};
|
|
||||||
|
|
||||||
const paddedNameLabel = (name: string, compact: boolean) => {
|
|
||||||
const label = '[' + name + ']';
|
|
||||||
return compact ? label : label.padEnd(12, '\xa0');
|
|
||||||
};
|
|
||||||
|
|
||||||
const paddedIDLabel = (id: number, compact: boolean) => {
|
|
||||||
const label = id + ':';
|
|
||||||
return compact ? label : label.padEnd(7, '\xa0');
|
|
||||||
};
|
|
||||||
|
|
||||||
// Memoized log entry component to prevent unnecessary re-renders
|
|
||||||
const LogEntryItem = memo(
|
|
||||||
({ entry, compact }: { entry: LogEntry; compact: boolean }) => {
|
|
||||||
return (
|
|
||||||
<div style={{ font: '13px monospace', whiteSpace: 'nowrap' }}>
|
|
||||||
<span>{entry.t}</span>
|
|
||||||
<span>{paddedLevelLabel(entry.l, compact)} </span>
|
|
||||||
<span>{paddedIDLabel(entry.i, compact)} </span>
|
|
||||||
<span>{paddedNameLabel(entry.n, compact)} </span>
|
|
||||||
<LogEntryLine details={{ level: entry.l }}>{entry.m}</LogEntryLine>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
(prevProps, nextProps) =>
|
|
||||||
prevProps.entry.i === nextProps.entry.i &&
|
|
||||||
prevProps.compact === nextProps.compact
|
|
||||||
);
|
|
||||||
|
|
||||||
const SystemLog = () => {
|
const SystemLog = () => {
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
@@ -137,85 +101,54 @@ const SystemLog = () => {
|
|||||||
const [readOpen, setReadOpen] = useState(false);
|
const [readOpen, setReadOpen] = useState(false);
|
||||||
const [logEntries, setLogEntries] = useState<LogEntry[]>([]);
|
const [logEntries, setLogEntries] = useState<LogEntry[]>([]);
|
||||||
const [autoscroll, setAutoscroll] = useState(true);
|
const [autoscroll, setAutoscroll] = useState(true);
|
||||||
const [boxPosition, setBoxPosition] = useState({ top: 0, left: 0 });
|
const [lastId, setLastId] = useState<number>(-1);
|
||||||
|
|
||||||
const ALPHA_NUMERIC_DASH_REGEX = /^[a-fA-F0-9 ]+$/;
|
const ALPHA_NUMERIC_DASH_REGEX = /^[a-fA-F0-9 ]+$/;
|
||||||
|
|
||||||
const updateFormValue = updateValueDirty(
|
const updateFormValue = updateValueDirty(
|
||||||
origData as unknown as Record<string, unknown>,
|
origData,
|
||||||
dirtyFlags,
|
dirtyFlags,
|
||||||
setDirtyFlags,
|
setDirtyFlags,
|
||||||
updateDataValue as (value: unknown) => void
|
updateDataValue
|
||||||
);
|
);
|
||||||
|
|
||||||
// Calculate box position after layout
|
|
||||||
useLayoutEffect(() => {
|
|
||||||
const logWindow = document.getElementById('log-window');
|
|
||||||
if (!logWindow) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const updatePosition = () => {
|
|
||||||
const windowElement = document.getElementById('log-window');
|
|
||||||
if (!windowElement) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const rect = windowElement.getBoundingClientRect();
|
|
||||||
setBoxPosition({ top: rect.bottom, left: rect.left });
|
|
||||||
};
|
|
||||||
|
|
||||||
updatePosition();
|
|
||||||
|
|
||||||
// Debounce resize events with requestAnimationFrame
|
|
||||||
let rafId: number;
|
|
||||||
const handleResize = () => {
|
|
||||||
cancelAnimationFrame(rafId);
|
|
||||||
rafId = requestAnimationFrame(updatePosition);
|
|
||||||
};
|
|
||||||
|
|
||||||
// Update position on window resize
|
|
||||||
window.addEventListener('resize', handleResize);
|
|
||||||
const resizeObserver = new ResizeObserver(handleResize);
|
|
||||||
resizeObserver.observe(logWindow);
|
|
||||||
|
|
||||||
return () => {
|
|
||||||
window.removeEventListener('resize', handleResize);
|
|
||||||
resizeObserver.disconnect();
|
|
||||||
cancelAnimationFrame(rafId);
|
|
||||||
};
|
|
||||||
}, [data]); // Recalculate when data changes (in case layout shifts)
|
|
||||||
|
|
||||||
// Memoize message handler to avoid recreating on every render
|
|
||||||
const handleLogMessage = useCallback((message: { data: string }) => {
|
|
||||||
const rawData = message.data;
|
|
||||||
const logentry = JSON.parse(rawData) as LogEntry;
|
|
||||||
setLogEntries((log) => {
|
|
||||||
// Skip if this is a duplicate entry (check last entry id)
|
|
||||||
if (log.length > 0) {
|
|
||||||
const lastEntry = log[log.length - 1];
|
|
||||||
if (lastEntry && logentry.i <= lastEntry.i) {
|
|
||||||
return log;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
const newLog = [...log, logentry];
|
|
||||||
return newLog;
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
useSSE(fetchLogES, {
|
useSSE(fetchLogES, {
|
||||||
immediate: true,
|
immediate: true,
|
||||||
interceptByGlobalResponded: false
|
interceptByGlobalResponded: false
|
||||||
})
|
})
|
||||||
.onMessage(handleLogMessage)
|
.onMessage((message: { data: string }) => {
|
||||||
|
const rawData = message.data;
|
||||||
|
const logentry = JSON.parse(rawData) as LogEntry;
|
||||||
|
if (lastId < logentry.i) {
|
||||||
|
setLogEntries((log) => [...log, logentry]);
|
||||||
|
setLastId(logentry.i);
|
||||||
|
}
|
||||||
|
})
|
||||||
.onError(() => {
|
.onError(() => {
|
||||||
toast.error('No connection to Log service');
|
toast.error('No connection to Log service');
|
||||||
});
|
});
|
||||||
|
|
||||||
const onDownload = useCallback(() => {
|
const paddedLevelLabel = (level: LogLevel) => {
|
||||||
const result = logEntries
|
const label = levelLabel(level);
|
||||||
.map((i) => `${i.t} ${levelLabel(i.l)} ${i.i}: [${i.n}] ${i.m}`)
|
return data?.compact ? ' ' + label[0] : label.padStart(8, '\xa0');
|
||||||
.join('\n');
|
};
|
||||||
|
|
||||||
|
const paddedNameLabel = (name: string) => {
|
||||||
|
const label = '[' + name + ']';
|
||||||
|
return data?.compact ? label : label.padEnd(12, '\xa0');
|
||||||
|
};
|
||||||
|
|
||||||
|
const paddedIDLabel = (id: number) => {
|
||||||
|
const label = id + ':';
|
||||||
|
return data?.compact ? label : label.padEnd(7, '\xa0');
|
||||||
|
};
|
||||||
|
|
||||||
|
const onDownload = () => {
|
||||||
|
let result = '';
|
||||||
|
for (const i of logEntries) {
|
||||||
|
result +=
|
||||||
|
i.t + ' ' + levelLabel(i.l) + ' ' + i.i + ': [' + i.n + '] ' + i.m + '\n';
|
||||||
|
}
|
||||||
const a = document.createElement('a');
|
const a = document.createElement('a');
|
||||||
a.setAttribute(
|
a.setAttribute(
|
||||||
'href',
|
'href',
|
||||||
@@ -225,28 +158,24 @@ const SystemLog = () => {
|
|||||||
document.body.appendChild(a);
|
document.body.appendChild(a);
|
||||||
a.click();
|
a.click();
|
||||||
document.body.removeChild(a);
|
document.body.removeChild(a);
|
||||||
}, [logEntries]);
|
};
|
||||||
|
|
||||||
const saveSettings = useCallback(async () => {
|
const saveSettings = async () => {
|
||||||
await saveData();
|
await saveData();
|
||||||
}, [saveData]);
|
};
|
||||||
|
|
||||||
// handle scrolling - optimized to only scroll when needed
|
// handle scrolling
|
||||||
const ref = useRef<HTMLDivElement>(null);
|
const ref = useRef<HTMLDivElement>(null);
|
||||||
const logWindowRef = useRef<HTMLDivElement>(null);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (logEntries.length && autoscroll) {
|
if (logEntries.length && autoscroll) {
|
||||||
const container = logWindowRef.current;
|
ref.current?.scrollIntoView({
|
||||||
if (container) {
|
behavior: 'smooth',
|
||||||
requestAnimationFrame(() => {
|
block: 'end'
|
||||||
container.scrollTop = container.scrollHeight;
|
});
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}, [logEntries.length, autoscroll]);
|
}, [logEntries.length]);
|
||||||
|
|
||||||
const sendReadCommand = useCallback(() => {
|
const sendReadCommand = () => {
|
||||||
if (readValue === '') {
|
if (readValue === '') {
|
||||||
setReadOpen(!readOpen);
|
setReadOpen(!readOpen);
|
||||||
return;
|
return;
|
||||||
@@ -257,11 +186,11 @@ const SystemLog = () => {
|
|||||||
setReadOpen(false);
|
setReadOpen(false);
|
||||||
setReadValue('');
|
setReadValue('');
|
||||||
}
|
}
|
||||||
}, [readValue, readOpen, send]);
|
};
|
||||||
|
|
||||||
const content = () => {
|
const content = () => {
|
||||||
if (!data) {
|
if (!data) {
|
||||||
return <FormLoader onRetry={loadData} errorMessage={errorMessage || ''} />;
|
return <FormLoader onRetry={loadData} errorMessage={errorMessage} />;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
@@ -302,8 +231,6 @@ const SystemLog = () => {
|
|||||||
<MenuItem value={50}>50</MenuItem>
|
<MenuItem value={50}>50</MenuItem>
|
||||||
<MenuItem value={75}>75</MenuItem>
|
<MenuItem value={75}>75</MenuItem>
|
||||||
<MenuItem value={100}>100</MenuItem>
|
<MenuItem value={100}>100</MenuItem>
|
||||||
<MenuItem value={500}>500</MenuItem>
|
|
||||||
<MenuItem value={1000}>1000</MenuItem>
|
|
||||||
</TextField>
|
</TextField>
|
||||||
</Grid>
|
</Grid>
|
||||||
)}
|
)}
|
||||||
@@ -351,7 +278,6 @@ const SystemLog = () => {
|
|||||||
>
|
>
|
||||||
<IconButton
|
<IconButton
|
||||||
disableRipple
|
disableRipple
|
||||||
aria-label={LL.CANCEL()}
|
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
setReadOpen(false);
|
setReadOpen(false);
|
||||||
setReadValue('');
|
setReadValue('');
|
||||||
@@ -377,7 +303,7 @@ const SystemLog = () => {
|
|||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
{data.developer_mode && (
|
{data.developer_mode && (
|
||||||
<IconButton onClick={sendReadCommand} aria-label={LL.EXECUTE()}>
|
<IconButton onClick={sendReadCommand}>
|
||||||
<PlayArrowIcon color="primary" />
|
<PlayArrowIcon color="primary" />
|
||||||
</IconButton>
|
</IconButton>
|
||||||
)}
|
)}
|
||||||
@@ -399,20 +325,27 @@ const SystemLog = () => {
|
|||||||
</Grid>
|
</Grid>
|
||||||
|
|
||||||
<Box
|
<Box
|
||||||
ref={logWindowRef}
|
|
||||||
sx={{
|
sx={{
|
||||||
backgroundColor: 'black',
|
backgroundColor: 'black',
|
||||||
overflowY: 'scroll',
|
overflowY: 'scroll',
|
||||||
position: 'absolute',
|
position: 'absolute',
|
||||||
right: 18,
|
right: 18,
|
||||||
bottom: 18,
|
bottom: 18,
|
||||||
left: boxPosition.left,
|
left: () => leftOffset(),
|
||||||
top: boxPosition.top,
|
top: () => topOffset(),
|
||||||
p: 1
|
p: 1
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{logEntries.map((e) => (
|
{logEntries.map((e) => (
|
||||||
<LogEntryItem key={e.i} entry={e} compact={data.compact} />
|
<div key={e.i} style={{ font: '14px monospace', whiteSpace: 'nowrap' }}>
|
||||||
|
<span>{e.t}</span>
|
||||||
|
<span>{paddedLevelLabel(e.l)} </span>
|
||||||
|
<span>{paddedIDLabel(e.i)} </span>
|
||||||
|
<span>{paddedNameLabel(e.n)} </span>
|
||||||
|
<LogEntryLine details={{ level: e.l }} key={e.i}>
|
||||||
|
{e.m}
|
||||||
|
</LogEntryLine>
|
||||||
|
</div>
|
||||||
))}
|
))}
|
||||||
|
|
||||||
<div ref={ref} />
|
<div ref={ref} />
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
import { useCallback, useMemo, useRef, useState } from 'react';
|
import { useState } from 'react';
|
||||||
|
|
||||||
import CancelIcon from '@mui/icons-material/Cancel';
|
import CancelIcon from '@mui/icons-material/Cancel';
|
||||||
import { Box, Button, Typography } from '@mui/material';
|
import { Box, Button, Dialog, DialogContent, Typography } from '@mui/material';
|
||||||
|
|
||||||
import { callAction } from 'api/app';
|
import { callAction } from 'api/app';
|
||||||
import { readSystemStatus } from 'api/system';
|
import { readSystemStatus } from 'api/system';
|
||||||
|
|
||||||
|
import { dialogStyle } from 'CustomTheme';
|
||||||
import { useRequest } from 'alova/client';
|
import { useRequest } from 'alova/client';
|
||||||
import MessageBox from 'components/MessageBox';
|
import MessageBox from 'components/MessageBox';
|
||||||
import { useI18nContext } from 'i18n/i18n-react';
|
import { useI18nContext } from 'i18n/i18n-react';
|
||||||
@@ -16,9 +17,11 @@ import { LinearProgressWithLabel } from '../../components/upload/LinearProgressW
|
|||||||
|
|
||||||
const SystemMonitor = () => {
|
const SystemMonitor = () => {
|
||||||
const [errorMessage, setErrorMessage] = useState<string>();
|
const [errorMessage, setErrorMessage] = useState<string>();
|
||||||
const hasInitialized = useRef(false);
|
|
||||||
const { LL } = useI18nContext();
|
const { LL } = useI18nContext();
|
||||||
|
|
||||||
|
let count = 0;
|
||||||
|
|
||||||
const { send: setSystemStatus } = useRequest(
|
const { send: setSystemStatus } = useRequest(
|
||||||
(status: string) => callAction({ action: 'systemStatus', param: status }),
|
(status: string) => callAction({ action: 'systemStatus', param: status }),
|
||||||
{
|
{
|
||||||
@@ -29,12 +32,10 @@ const SystemMonitor = () => {
|
|||||||
const { data, send } = useRequest(readSystemStatus, {
|
const { data, send } = useRequest(readSystemStatus, {
|
||||||
force: true,
|
force: true,
|
||||||
async middleware(_, next) {
|
async middleware(_, next) {
|
||||||
// Skip first request to allow AsyncWS to send its response
|
if (count++ >= 1) {
|
||||||
if (!hasInitialized.current) {
|
// skip first request (1 second) to allow AsyncWS to send its response
|
||||||
hasInitialized.current = true;
|
await next();
|
||||||
return; // Don't await next() on first call
|
|
||||||
}
|
}
|
||||||
await next();
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.onSuccess((event) => {
|
.onSuccess((event) => {
|
||||||
@@ -50,100 +51,53 @@ const SystemMonitor = () => {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
.onError((error) => {
|
.onError((error) => {
|
||||||
setErrorMessage(String(error.error?.message || 'An error occurred'));
|
setErrorMessage(error.message);
|
||||||
});
|
});
|
||||||
|
|
||||||
useInterval(() => {
|
useInterval(() => {
|
||||||
void send();
|
void send();
|
||||||
}, 1000); // check every 1 second
|
}, 1000); // check every 1 second
|
||||||
|
|
||||||
const { statusMessage, isUploading, progressValue } = useMemo(() => {
|
const onCancel = async () => {
|
||||||
const status = data?.status;
|
|
||||||
|
|
||||||
const message =
|
|
||||||
status && status >= SystemStatusCodes.SYSTEM_STATUS_UPLOADING
|
|
||||||
? LL.WAIT_FIRMWARE()
|
|
||||||
: status === SystemStatusCodes.SYSTEM_STATUS_PENDING_RESTART
|
|
||||||
? LL.APPLICATION_RESTARTING()
|
|
||||||
: status === SystemStatusCodes.SYSTEM_STATUS_NORMAL
|
|
||||||
? LL.RESTARTING_PRE()
|
|
||||||
: status === SystemStatusCodes.SYSTEM_STATUS_ERROR_UPLOAD
|
|
||||||
? 'Upload Failed'
|
|
||||||
: LL.RESTARTING_POST();
|
|
||||||
|
|
||||||
const uploading =
|
|
||||||
status !== undefined && status >= SystemStatusCodes.SYSTEM_STATUS_UPLOADING;
|
|
||||||
const progress =
|
|
||||||
uploading && status
|
|
||||||
? Math.round(status - SystemStatusCodes.SYSTEM_STATUS_UPLOADING)
|
|
||||||
: 0;
|
|
||||||
|
|
||||||
return {
|
|
||||||
statusMessage: message,
|
|
||||||
isUploading: uploading,
|
|
||||||
progressValue: progress
|
|
||||||
};
|
|
||||||
}, [data?.status, LL]);
|
|
||||||
|
|
||||||
const onCancel = useCallback(async () => {
|
|
||||||
setErrorMessage(undefined);
|
setErrorMessage(undefined);
|
||||||
await setSystemStatus(String(SystemStatusCodes.SYSTEM_STATUS_NORMAL));
|
await setSystemStatus(
|
||||||
|
SystemStatusCodes.SYSTEM_STATUS_NORMAL as unknown as string
|
||||||
|
);
|
||||||
document.location.href = '/';
|
document.location.href = '/';
|
||||||
}, [setSystemStatus]);
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Box
|
<Dialog fullWidth={true} sx={dialogStyle} open={true}>
|
||||||
sx={{
|
<DialogContent dividers>
|
||||||
position: 'fixed',
|
<Box m={0} py={0} display="flex" alignItems="center" flexDirection="column">
|
||||||
top: 0,
|
|
||||||
left: 0,
|
|
||||||
width: '100vw',
|
|
||||||
height: '100vh',
|
|
||||||
display: 'flex',
|
|
||||||
alignItems: 'center',
|
|
||||||
justifyContent: 'center'
|
|
||||||
// backdropFilter: 'blur(8px)'
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Box
|
|
||||||
sx={{
|
|
||||||
width: '30%',
|
|
||||||
minWidth: '300px',
|
|
||||||
maxWidth: '500px',
|
|
||||||
backgroundColor: '#393939',
|
|
||||||
border: 2,
|
|
||||||
borderColor: '#565656',
|
|
||||||
borderRadius: '8px',
|
|
||||||
boxShadow: '0 8px 32px rgba(0, 0, 0, 0.3)',
|
|
||||||
p: 3
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Box display="flex" alignItems="center" flexDirection="column">
|
|
||||||
<img
|
|
||||||
src="/app/icon.png"
|
|
||||||
alt="EMS-ESP"
|
|
||||||
style={{ width: '40px', height: '40px', marginBottom: '16px' }}
|
|
||||||
/>
|
|
||||||
<Typography
|
<Typography
|
||||||
color="secondary"
|
color="secondary"
|
||||||
variant="h6"
|
variant="h6"
|
||||||
fontWeight={400}
|
fontWeight={400}
|
||||||
textAlign="center"
|
textAlign="center"
|
||||||
>
|
>
|
||||||
{statusMessage}
|
{data?.status >= SystemStatusCodes.SYSTEM_STATUS_UPLOADING
|
||||||
|
? LL.WAIT_FIRMWARE()
|
||||||
|
: data?.status === SystemStatusCodes.SYSTEM_STATUS_PENDING_RESTART
|
||||||
|
? LL.APPLICATION_RESTARTING()
|
||||||
|
: data?.status === SystemStatusCodes.SYSTEM_STATUS_NORMAL
|
||||||
|
? LL.RESTARTING_PRE()
|
||||||
|
: data?.status === SystemStatusCodes.SYSTEM_STATUS_ERROR_UPLOAD
|
||||||
|
? 'Upload Failed'
|
||||||
|
: LL.RESTARTING_POST()}
|
||||||
</Typography>
|
</Typography>
|
||||||
|
|
||||||
{errorMessage ? (
|
{errorMessage ? (
|
||||||
<MessageBox level="error" message={errorMessage}>
|
<MessageBox my={2} level="error" message={errorMessage}>
|
||||||
<Button
|
<Button
|
||||||
sx={{ ml: 2 }}
|
|
||||||
size="small"
|
size="small"
|
||||||
|
sx={{ ml: 2 }}
|
||||||
startIcon={<CancelIcon />}
|
startIcon={<CancelIcon />}
|
||||||
variant="contained"
|
variant="contained"
|
||||||
color="error"
|
color="error"
|
||||||
onClick={onCancel}
|
onClick={onCancel}
|
||||||
>
|
>
|
||||||
{LL.RESTART()}
|
{LL.RESET(0)}
|
||||||
</Button>
|
</Button>
|
||||||
</MessageBox>
|
</MessageBox>
|
||||||
) : (
|
) : (
|
||||||
@@ -151,16 +105,20 @@ const SystemMonitor = () => {
|
|||||||
<Typography mt={2} variant="h6" fontWeight={400} textAlign="center">
|
<Typography mt={2} variant="h6" fontWeight={400} textAlign="center">
|
||||||
{LL.PLEASE_WAIT()}…
|
{LL.PLEASE_WAIT()}…
|
||||||
</Typography>
|
</Typography>
|
||||||
{isUploading && (
|
{data && data.status >= SystemStatusCodes.SYSTEM_STATUS_UPLOADING && (
|
||||||
<Box width="100%" pl={2} pr={2} py={2}>
|
<Box width="100%" pl={2} pr={2} py={2}>
|
||||||
<LinearProgressWithLabel value={progressValue} />
|
<LinearProgressWithLabel
|
||||||
|
value={Math.round(
|
||||||
|
data?.status - SystemStatusCodes.SYSTEM_STATUS_UPLOADING
|
||||||
|
)}
|
||||||
|
/>
|
||||||
</Box>
|
</Box>
|
||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
</Box>
|
</Box>
|
||||||
</Box>
|
</DialogContent>
|
||||||
</Box>
|
</Dialog>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,22 +1,26 @@
|
|||||||
import { type FC, type PropsWithChildren, memo } from 'react';
|
import type { FC } from 'react';
|
||||||
|
|
||||||
import { Box } from '@mui/material';
|
import { Box } from '@mui/material';
|
||||||
import type { BoxProps } from '@mui/material';
|
import type { BoxProps } from '@mui/material';
|
||||||
|
|
||||||
const ButtonRow: FC<PropsWithChildren<BoxProps>> = memo(({ children, ...rest }) => (
|
const ButtonRow: FC<BoxProps> = ({ children, ...rest }) => (
|
||||||
<Box
|
<Box
|
||||||
sx={{
|
sx={{
|
||||||
'& button, & a, & .MuiCard-root': {
|
'& button, & a, & .MuiCard-root': {
|
||||||
mt: 2,
|
mt: 2,
|
||||||
mx: 0.6,
|
mx: 0.6,
|
||||||
'&:last-child': { mr: 0 },
|
'&:last-child': {
|
||||||
'&:first-of-type': { ml: 0 }
|
mr: 0
|
||||||
|
},
|
||||||
|
'&:first-of-type': {
|
||||||
|
ml: 0
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
{...rest}
|
{...rest}
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
</Box>
|
</Box>
|
||||||
));
|
);
|
||||||
|
|
||||||
export default ButtonRow;
|
export default ButtonRow;
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user