mirror of
https://github.com/home-assistant/frontend.git
synced 2025-08-25 09:09:26 +00:00
Compare commits
14 Commits
fix-menu-o
...
responsive
Author | SHA1 | Date | |
---|---|---|---|
![]() |
ea9ef9a165 | ||
![]() |
5270ca4db6 | ||
![]() |
12cb94cdc0 | ||
![]() |
d473a30263 | ||
![]() |
e767b807b0 | ||
![]() |
f593fb6043 | ||
![]() |
bbf8ee7c64 | ||
![]() |
3d9993563f | ||
![]() |
fc3e353e94 | ||
![]() |
e9738bd5ae | ||
![]() |
87d9298b20 | ||
![]() |
a29b92a92a | ||
![]() |
5b4504688a | ||
![]() |
0a4401b417 |
@@ -1,5 +1,5 @@
|
|||||||
# See here for image contents: https://github.com/microsoft/vscode-dev-containers/tree/v0.148.1/containers/python-3/.devcontainer/base.Dockerfile
|
# See here for image contents: https://github.com/microsoft/vscode-dev-containers/tree/v0.148.1/containers/python-3/.devcontainer/base.Dockerfile
|
||||||
FROM mcr.microsoft.com/devcontainers/python:3.12
|
FROM mcr.microsoft.com/vscode/devcontainers/python:0-3.11
|
||||||
|
|
||||||
ENV \
|
ENV \
|
||||||
DEBIAN_FRONTEND=noninteractive \
|
DEBIAN_FRONTEND=noninteractive \
|
||||||
|
@@ -5,7 +5,6 @@
|
|||||||
"context": ".."
|
"context": ".."
|
||||||
},
|
},
|
||||||
"appPort": "8124:8123",
|
"appPort": "8124:8123",
|
||||||
"postCreateCommand": "sudo apt update && sudo apt upgrade -y && sudo apt install -y libpcap-dev",
|
|
||||||
"postStartCommand": "script/bootstrap",
|
"postStartCommand": "script/bootstrap",
|
||||||
"containerEnv": {
|
"containerEnv": {
|
||||||
"WORKSPACE_DIRECTORY": "${containerWorkspaceFolder}"
|
"WORKSPACE_DIRECTORY": "${containerWorkspaceFolder}"
|
||||||
|
3
.github/ISSUE_TEMPLATE/bug_report.yml
vendored
3
.github/ISSUE_TEMPLATE/bug_report.yml
vendored
@@ -9,7 +9,7 @@ body:
|
|||||||
|
|
||||||
If you have a feature or enhancement request for the frontend, please [start an discussion][fr] instead of creating an issue.
|
If you have a feature or enhancement request for the frontend, please [start an discussion][fr] instead of creating an issue.
|
||||||
|
|
||||||
**Please do not report issues for custom cards.**
|
**Please not not report issues for custom cards.**
|
||||||
|
|
||||||
[fr]: https://github.com/home-assistant/frontend/discussions
|
[fr]: https://github.com/home-assistant/frontend/discussions
|
||||||
[releases]: https://github.com/home-assistant/home-assistant/releases
|
[releases]: https://github.com/home-assistant/home-assistant/releases
|
||||||
@@ -24,7 +24,6 @@ body:
|
|||||||
required: true
|
required: true
|
||||||
- label: I have tried a different browser to see if it is related to my browser.
|
- label: I have tried a different browser to see if it is related to my browser.
|
||||||
required: true
|
required: true
|
||||||
- label: I have tried reproducing the issue in [safe mode](https://www.home-assistant.io/blog/2023/11/01/release-202311/#restarting-into-safe-mode) to rule out problems with unsupported custom resources.
|
|
||||||
- type: markdown
|
- type: markdown
|
||||||
attributes:
|
attributes:
|
||||||
value: |
|
value: |
|
||||||
|
56
.github/labeler.yml
vendored
56
.github/labeler.yml
vendored
@@ -1,51 +1,31 @@
|
|||||||
Build:
|
Build:
|
||||||
- changed-files:
|
- build-scripts/**
|
||||||
- any-glob-to-any-file:
|
- .browserslistrc
|
||||||
- build-scripts/**
|
- gulpfile.js
|
||||||
- .browserslistrc
|
|
||||||
- gulpfile.js
|
|
||||||
|
|
||||||
Cast:
|
Cast:
|
||||||
- changed-files:
|
- cast/src/**
|
||||||
- any-glob-to-any-file:
|
- src/cast/**
|
||||||
- cast/src/**
|
|
||||||
- src/cast/**
|
|
||||||
|
|
||||||
Demo:
|
Demo:
|
||||||
- changed-files:
|
- demo/src/**
|
||||||
- any-glob-to-any-file:
|
- src/fake_data/**
|
||||||
- demo/src/**
|
|
||||||
- src/fake_data/**
|
|
||||||
|
|
||||||
Design:
|
Design:
|
||||||
- changed-files:
|
- gallery/src/**
|
||||||
- any-glob-to-any-file:
|
- src/fake_data/**
|
||||||
- gallery/src/**
|
|
||||||
- src/fake_data/**
|
|
||||||
|
|
||||||
Dependencies:
|
Dependencies:
|
||||||
- any:
|
- package.json
|
||||||
- changed-files:
|
- renovate.json
|
||||||
# Match when only these files are changed (i.e. don't match PRs that happen to add or remove packages)
|
- yarn.lock
|
||||||
- any-glob-to-all-files:
|
- .yarn/**
|
||||||
- package.json
|
- .yarnrc.yml
|
||||||
- renovate.json
|
- .nvmrc
|
||||||
- yarn.lock
|
|
||||||
- .yarn/**
|
|
||||||
- .yarnrc.yml
|
|
||||||
- .nvmrc
|
|
||||||
# Dependabot and Renovate branches always match (i.e. compatibility tweaks by members considered minor)
|
|
||||||
- head-branch:
|
|
||||||
- "^renovate/"
|
|
||||||
- "^dependabot/"
|
|
||||||
|
|
||||||
GitHub Actions:
|
GitHub Actions:
|
||||||
- changed-files:
|
- .github/workflows/**
|
||||||
- any-glob-to-any-file:
|
- .github/*.yml
|
||||||
- .github/workflows/**
|
|
||||||
- .github/*.yml
|
|
||||||
|
|
||||||
Supervisor:
|
Supervisor:
|
||||||
- changed-files:
|
- hassio/src/**
|
||||||
- any-glob-to-any-file:
|
|
||||||
- hassio/src/**
|
|
||||||
|
8
.github/workflows/cast_deployment.yaml
vendored
8
.github/workflows/cast_deployment.yaml
vendored
@@ -21,12 +21,12 @@ jobs:
|
|||||||
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
with:
|
with:
|
||||||
ref: dev
|
ref: dev
|
||||||
|
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v3.8.1
|
||||||
with:
|
with:
|
||||||
node-version-file: ".nvmrc"
|
node-version-file: ".nvmrc"
|
||||||
cache: yarn
|
cache: yarn
|
||||||
@@ -57,12 +57,12 @@ jobs:
|
|||||||
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
with:
|
with:
|
||||||
ref: master
|
ref: master
|
||||||
|
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v3.8.1
|
||||||
with:
|
with:
|
||||||
node-version-file: ".nvmrc"
|
node-version-file: ".nvmrc"
|
||||||
cache: yarn
|
cache: yarn
|
||||||
|
35
.github/workflows/ci.yaml
vendored
35
.github/workflows/ci.yaml
vendored
@@ -24,9 +24,9 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v3.8.1
|
||||||
with:
|
with:
|
||||||
node-version-file: ".nvmrc"
|
node-version-file: ".nvmrc"
|
||||||
cache: yarn
|
cache: yarn
|
||||||
@@ -37,20 +37,17 @@ jobs:
|
|||||||
- name: Build resources
|
- name: Build resources
|
||||||
run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data gather-gallery-pages
|
run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data gather-gallery-pages
|
||||||
- name: Setup lint cache
|
- name: Setup lint cache
|
||||||
uses: actions/cache@v4.0.2
|
uses: actions/cache@v3.3.2
|
||||||
with:
|
with:
|
||||||
path: |
|
path: |
|
||||||
node_modules/.cache/prettier
|
node_modules/.cache/prettier
|
||||||
node_modules/.cache/eslint
|
node_modules/.cache/eslint
|
||||||
node_modules/.cache/typescript
|
|
||||||
key: lint-${{ github.sha }}
|
key: lint-${{ github.sha }}
|
||||||
restore-keys: lint-
|
restore-keys: lint-
|
||||||
- name: Run eslint
|
- name: Run eslint
|
||||||
run: yarn run lint:eslint --quiet
|
run: yarn run lint:eslint --quiet
|
||||||
- name: Run tsc
|
- name: Run tsc
|
||||||
run: yarn run lint:types
|
run: yarn run lint:types
|
||||||
- name: Run lit-analyzer
|
|
||||||
run: yarn run lint:lit --quiet
|
|
||||||
- name: Run prettier
|
- name: Run prettier
|
||||||
run: yarn run lint:prettier
|
run: yarn run lint:prettier
|
||||||
test:
|
test:
|
||||||
@@ -58,16 +55,16 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v3.8.1
|
||||||
with:
|
with:
|
||||||
node-version-file: ".nvmrc"
|
node-version-file: ".nvmrc"
|
||||||
cache: yarn
|
cache: yarn
|
||||||
- name: Install dependencies
|
- name: Install dependencies
|
||||||
run: yarn install --immutable
|
run: yarn install --immutable
|
||||||
- name: Build resources
|
- name: Build resources
|
||||||
run: ./node_modules/.bin/gulp gen-icons-json build-translations build-locale-data
|
run: ./node_modules/.bin/gulp build-translations build-locale-data
|
||||||
- name: Run Tests
|
- name: Run Tests
|
||||||
run: yarn run test
|
run: yarn run test
|
||||||
build:
|
build:
|
||||||
@@ -76,9 +73,9 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v3.8.1
|
||||||
with:
|
with:
|
||||||
node-version-file: ".nvmrc"
|
node-version-file: ".nvmrc"
|
||||||
cache: yarn
|
cache: yarn
|
||||||
@@ -88,21 +85,15 @@ jobs:
|
|||||||
run: ./node_modules/.bin/gulp build-app
|
run: ./node_modules/.bin/gulp build-app
|
||||||
env:
|
env:
|
||||||
IS_TEST: "true"
|
IS_TEST: "true"
|
||||||
- name: Upload bundle stats
|
|
||||||
uses: actions/upload-artifact@v4.3.1
|
|
||||||
with:
|
|
||||||
name: frontend-bundle-stats
|
|
||||||
path: build/stats/*.json
|
|
||||||
if-no-files-found: error
|
|
||||||
supervisor:
|
supervisor:
|
||||||
name: Build supervisor
|
name: Build supervisor
|
||||||
needs: [lint, test]
|
needs: [lint, test]
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v3.8.1
|
||||||
with:
|
with:
|
||||||
node-version-file: ".nvmrc"
|
node-version-file: ".nvmrc"
|
||||||
cache: yarn
|
cache: yarn
|
||||||
@@ -112,9 +103,3 @@ jobs:
|
|||||||
run: ./node_modules/.bin/gulp build-hassio
|
run: ./node_modules/.bin/gulp build-hassio
|
||||||
env:
|
env:
|
||||||
IS_TEST: "true"
|
IS_TEST: "true"
|
||||||
- name: Upload bundle stats
|
|
||||||
uses: actions/upload-artifact@v4.3.1
|
|
||||||
with:
|
|
||||||
name: supervisor-bundle-stats
|
|
||||||
path: build/stats/*.json
|
|
||||||
if-no-files-found: error
|
|
||||||
|
8
.github/workflows/codeql-analysis.yml
vendored
8
.github/workflows/codeql-analysis.yml
vendored
@@ -23,7 +23,7 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
with:
|
with:
|
||||||
# We must fetch at least the immediate parents so that if this is
|
# We must fetch at least the immediate parents so that if this is
|
||||||
# a pull request then we can checkout the head.
|
# a pull request then we can checkout the head.
|
||||||
@@ -36,14 +36,14 @@ jobs:
|
|||||||
|
|
||||||
# Initializes the CodeQL tools for scanning.
|
# Initializes the CodeQL tools for scanning.
|
||||||
- name: Initialize CodeQL
|
- name: Initialize CodeQL
|
||||||
uses: github/codeql-action/init@v3
|
uses: github/codeql-action/init@v2
|
||||||
with:
|
with:
|
||||||
languages: ${{ matrix.language }}
|
languages: ${{ matrix.language }}
|
||||||
|
|
||||||
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
|
# Autobuild attempts to build any compiled languages (C/C++, C#, or Java).
|
||||||
# If this step fails, then you should remove it and run the build manually (see below)
|
# If this step fails, then you should remove it and run the build manually (see below)
|
||||||
- name: Autobuild
|
- name: Autobuild
|
||||||
uses: github/codeql-action/autobuild@v3
|
uses: github/codeql-action/autobuild@v2
|
||||||
|
|
||||||
# ℹ️ Command-line programs to run using the OS shell.
|
# ℹ️ Command-line programs to run using the OS shell.
|
||||||
# 📚 https://git.io/JvXDl
|
# 📚 https://git.io/JvXDl
|
||||||
@@ -57,4 +57,4 @@ jobs:
|
|||||||
# make release
|
# make release
|
||||||
|
|
||||||
- name: Perform CodeQL Analysis
|
- name: Perform CodeQL Analysis
|
||||||
uses: github/codeql-action/analyze@v3
|
uses: github/codeql-action/analyze@v2
|
||||||
|
8
.github/workflows/demo_deployment.yaml
vendored
8
.github/workflows/demo_deployment.yaml
vendored
@@ -22,12 +22,12 @@ jobs:
|
|||||||
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
with:
|
with:
|
||||||
ref: dev
|
ref: dev
|
||||||
|
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v3.8.1
|
||||||
with:
|
with:
|
||||||
node-version-file: ".nvmrc"
|
node-version-file: ".nvmrc"
|
||||||
cache: yarn
|
cache: yarn
|
||||||
@@ -58,12 +58,12 @@ jobs:
|
|||||||
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
with:
|
with:
|
||||||
ref: master
|
ref: master
|
||||||
|
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v3.8.1
|
||||||
with:
|
with:
|
||||||
node-version-file: ".nvmrc"
|
node-version-file: ".nvmrc"
|
||||||
cache: yarn
|
cache: yarn
|
||||||
|
4
.github/workflows/design_deployment.yaml
vendored
4
.github/workflows/design_deployment.yaml
vendored
@@ -16,10 +16,10 @@ jobs:
|
|||||||
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
url: ${{ steps.deploy.outputs.NETLIFY_LIVE_URL || steps.deploy.outputs.NETLIFY_URL }}
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
|
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v3.8.1
|
||||||
with:
|
with:
|
||||||
node-version-file: ".nvmrc"
|
node-version-file: ".nvmrc"
|
||||||
cache: yarn
|
cache: yarn
|
||||||
|
4
.github/workflows/design_preview.yaml
vendored
4
.github/workflows/design_preview.yaml
vendored
@@ -21,10 +21,10 @@ jobs:
|
|||||||
if: github.repository == 'home-assistant/frontend' && contains(github.event.pull_request.labels.*.name, 'needs design preview')
|
if: github.repository == 'home-assistant/frontend' && contains(github.event.pull_request.labels.*.name, 'needs design preview')
|
||||||
steps:
|
steps:
|
||||||
- name: Check out files from GitHub
|
- name: Check out files from GitHub
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
|
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v3.8.1
|
||||||
with:
|
with:
|
||||||
node-version-file: ".nvmrc"
|
node-version-file: ".nvmrc"
|
||||||
cache: yarn
|
cache: yarn
|
||||||
|
2
.github/workflows/labeler.yaml
vendored
2
.github/workflows/labeler.yaml
vendored
@@ -10,6 +10,6 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Apply labels
|
- name: Apply labels
|
||||||
uses: actions/labeler@v5.0.0
|
uses: actions/labeler@v4.3.0
|
||||||
with:
|
with:
|
||||||
sync-labels: true
|
sync-labels: true
|
||||||
|
3
.github/workflows/lock.yml
vendored
3
.github/workflows/lock.yml
vendored
@@ -9,10 +9,9 @@ jobs:
|
|||||||
lock:
|
lock:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: dessant/lock-threads@v5.0.1
|
- uses: dessant/lock-threads@v4.0.1
|
||||||
with:
|
with:
|
||||||
github-token: ${{ github.token }}
|
github-token: ${{ github.token }}
|
||||||
process-only: "issues, prs"
|
|
||||||
issue-lock-inactive-days: "30"
|
issue-lock-inactive-days: "30"
|
||||||
issue-exclude-created-before: "2020-10-01T00:00:00Z"
|
issue-exclude-created-before: "2020-10-01T00:00:00Z"
|
||||||
issue-lock-reason: ""
|
issue-lock-reason: ""
|
||||||
|
14
.github/workflows/nightly.yaml
vendored
14
.github/workflows/nightly.yaml
vendored
@@ -6,7 +6,7 @@ on:
|
|||||||
- cron: "0 1 * * *"
|
- cron: "0 1 * * *"
|
||||||
|
|
||||||
env:
|
env:
|
||||||
PYTHON_VERSION: "3.12"
|
PYTHON_VERSION: "3.11"
|
||||||
NODE_OPTIONS: --max_old_space_size=6144
|
NODE_OPTIONS: --max_old_space_size=6144
|
||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
@@ -20,15 +20,15 @@ jobs:
|
|||||||
contents: write
|
contents: write
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
|
|
||||||
- name: Set up Python ${{ env.PYTHON_VERSION }}
|
- name: Set up Python ${{ env.PYTHON_VERSION }}
|
||||||
uses: actions/setup-python@v5
|
uses: actions/setup-python@v4
|
||||||
with:
|
with:
|
||||||
python-version: ${{ env.PYTHON_VERSION }}
|
python-version: ${{ env.PYTHON_VERSION }}
|
||||||
|
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v3.8.1
|
||||||
with:
|
with:
|
||||||
node-version-file: ".nvmrc"
|
node-version-file: ".nvmrc"
|
||||||
cache: yarn
|
cache: yarn
|
||||||
@@ -42,7 +42,7 @@ jobs:
|
|||||||
LOKALISE_TOKEN: ${{ secrets.LOKALISE_TOKEN }}
|
LOKALISE_TOKEN: ${{ secrets.LOKALISE_TOKEN }}
|
||||||
|
|
||||||
- name: Bump version
|
- name: Bump version
|
||||||
run: script/version_bump.js nightly
|
run: script/version_bump.cjs nightly
|
||||||
|
|
||||||
- name: Build nightly Python wheels
|
- name: Build nightly Python wheels
|
||||||
run: |
|
run: |
|
||||||
@@ -57,14 +57,14 @@ jobs:
|
|||||||
run: tar -czvf translations.tar.gz translations
|
run: tar -czvf translations.tar.gz translations
|
||||||
|
|
||||||
- name: Upload build artifacts
|
- name: Upload build artifacts
|
||||||
uses: actions/upload-artifact@v4.3.1
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: wheels
|
name: wheels
|
||||||
path: dist/home_assistant_frontend*.whl
|
path: dist/home_assistant_frontend*.whl
|
||||||
if-no-files-found: error
|
if-no-files-found: error
|
||||||
|
|
||||||
- name: Upload translations
|
- name: Upload translations
|
||||||
uses: actions/upload-artifact@v4.3.1
|
uses: actions/upload-artifact@v3
|
||||||
with:
|
with:
|
||||||
name: translations
|
name: translations
|
||||||
path: translations.tar.gz
|
path: translations.tar.gz
|
||||||
|
25
.github/workflows/relative-ci.yaml
vendored
25
.github/workflows/relative-ci.yaml
vendored
@@ -1,25 +0,0 @@
|
|||||||
name: RelativeCI
|
|
||||||
|
|
||||||
on:
|
|
||||||
workflow_run:
|
|
||||||
workflows: [CI]
|
|
||||||
types:
|
|
||||||
- completed
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
upload:
|
|
||||||
name: Upload stats
|
|
||||||
if: ${{ github.event.workflow_run.conclusion == 'success' }}
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
bundle: [frontend, supervisor]
|
|
||||||
build: [modern, legacy]
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Send bundle stats and build information to RelativeCI
|
|
||||||
uses: relative-ci/agent-action@v2.1.10
|
|
||||||
with:
|
|
||||||
key: ${{ secrets[format('RELATIVE_CI_KEY_{0}_{1}', matrix.bundle, matrix.build)] }}
|
|
||||||
token: ${{ github.token }}
|
|
||||||
artifactName: ${{ format('{0}-bundle-stats', matrix.bundle) }}
|
|
||||||
webpackStatsFile: ${{ format('{0}-{1}.json', matrix.bundle, matrix.build) }}
|
|
2
.github/workflows/release-drafter.yaml
vendored
2
.github/workflows/release-drafter.yaml
vendored
@@ -18,6 +18,6 @@ jobs:
|
|||||||
pull-requests: read
|
pull-requests: read
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- uses: release-drafter/release-drafter@v6.0.0
|
- uses: release-drafter/release-drafter@v5
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
12
.github/workflows/release.yaml
vendored
12
.github/workflows/release.yaml
vendored
@@ -6,7 +6,7 @@ on:
|
|||||||
- published
|
- published
|
||||||
|
|
||||||
env:
|
env:
|
||||||
PYTHON_VERSION: "3.12"
|
PYTHON_VERSION: "3.11"
|
||||||
NODE_OPTIONS: --max_old_space_size=6144
|
NODE_OPTIONS: --max_old_space_size=6144
|
||||||
|
|
||||||
# Set default workflow permissions
|
# Set default workflow permissions
|
||||||
@@ -23,18 +23,18 @@ jobs:
|
|||||||
contents: write # Required to upload release assets
|
contents: write # Required to upload release assets
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
|
|
||||||
- name: Verify version
|
- name: Verify version
|
||||||
uses: home-assistant/actions/helpers/verify-version@master
|
uses: home-assistant/actions/helpers/verify-version@master
|
||||||
|
|
||||||
- name: Set up Python ${{ env.PYTHON_VERSION }}
|
- name: Set up Python ${{ env.PYTHON_VERSION }}
|
||||||
uses: actions/setup-python@v5
|
uses: actions/setup-python@v4
|
||||||
with:
|
with:
|
||||||
python-version: ${{ env.PYTHON_VERSION }}
|
python-version: ${{ env.PYTHON_VERSION }}
|
||||||
|
|
||||||
- name: Setup Node
|
- name: Setup Node
|
||||||
uses: actions/setup-node@v4.0.2
|
uses: actions/setup-node@v3.8.1
|
||||||
with:
|
with:
|
||||||
node-version-file: ".nvmrc"
|
node-version-file: ".nvmrc"
|
||||||
cache: yarn
|
cache: yarn
|
||||||
@@ -55,7 +55,7 @@ jobs:
|
|||||||
script/release
|
script/release
|
||||||
|
|
||||||
- name: Upload release assets
|
- name: Upload release assets
|
||||||
uses: softprops/action-gh-release@v2.0.4
|
uses: softprops/action-gh-release@v0.1.15
|
||||||
with:
|
with:
|
||||||
files: |
|
files: |
|
||||||
dist/*.whl
|
dist/*.whl
|
||||||
@@ -74,7 +74,7 @@ jobs:
|
|||||||
echo "home-assistant-frontend==$version" > ./requirements.txt
|
echo "home-assistant-frontend==$version" > ./requirements.txt
|
||||||
|
|
||||||
- name: Build wheels
|
- name: Build wheels
|
||||||
uses: home-assistant/wheels@2024.01.0
|
uses: home-assistant/wheels@2023.10.1
|
||||||
with:
|
with:
|
||||||
abi: cp311
|
abi: cp311
|
||||||
tag: musllinux_1_2
|
tag: musllinux_1_2
|
||||||
|
2
.github/workflows/stale.yml
vendored
2
.github/workflows/stale.yml
vendored
@@ -10,7 +10,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: 90 days stale policy
|
- name: 90 days stale policy
|
||||||
uses: actions/stale@v9.0.0
|
uses: actions/stale@v8.0.0
|
||||||
with:
|
with:
|
||||||
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
repo-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
days-before-stale: 90
|
days-before-stale: 90
|
||||||
|
2
.github/workflows/translations.yaml
vendored
2
.github/workflows/translations.yaml
vendored
@@ -13,7 +13,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@v4.1.2
|
uses: actions/checkout@v4.1.0
|
||||||
|
|
||||||
- name: Upload Translations
|
- name: Upload Translations
|
||||||
run: |
|
run: |
|
||||||
|
3
.gitignore
vendored
3
.gitignore
vendored
@@ -47,6 +47,3 @@ src/cast/dev_const.ts
|
|||||||
|
|
||||||
# Home Assistant config
|
# Home Assistant config
|
||||||
/config/
|
/config/
|
||||||
|
|
||||||
# Jetbrains
|
|
||||||
/.idea/
|
|
||||||
|
@@ -1,18 +0,0 @@
|
|||||||
diff --git a/dist/hls.light.mjs b/dist/hls.light.mjs
|
|
||||||
index eed9d788fafdb159975e1a2eb08ac88ba9c9ac33..ace881935e6665946f1c8110ebd2f739cde4427e 100644
|
|
||||||
--- a/dist/hls.light.mjs
|
|
||||||
+++ b/dist/hls.light.mjs
|
|
||||||
@@ -20523,9 +20523,9 @@ class Hls {
|
|
||||||
}
|
|
||||||
Hls.defaultConfig = void 0;
|
|
||||||
|
|
||||||
-var KeySystemFormats = empty.KeySystemFormats;
|
|
||||||
-var KeySystems = empty.KeySystems;
|
|
||||||
-var SubtitleStreamController = empty.SubtitleStreamController;
|
|
||||||
-var TimelineController = empty.TimelineController;
|
|
||||||
+var KeySystemFormats = empty;
|
|
||||||
+var KeySystems = empty;
|
|
||||||
+var SubtitleStreamController = empty;
|
|
||||||
+var TimelineController = empty;
|
|
||||||
export { AbrController, AttrList, Cues as AudioStreamController, Cues as AudioTrackController, BasePlaylistController, BaseSegment, BaseStreamController, BufferController, Cues as CMCDController, CapLevelController, ChunkMetadata, ContentSteeringController, DateRange, Cues as EMEController, ErrorActionFlags, ErrorController, ErrorDetails, ErrorTypes, Events, FPSController, Fragment, Hls, HlsSkip, HlsUrlParameters, KeySystemFormats, KeySystems, Level, LevelDetails, LevelKey, LoadStats, MetadataSchema, NetworkErrorAction, Part, PlaylistLevelType, SubtitleStreamController, Cues as SubtitleTrackController, TimelineController, Hls as default, getMediaSource, isMSESupported, isSupported };
|
|
||||||
//# sourceMappingURL=hls.light.mjs.map
|
|
39
.yarn/patches/sortablejs-npm-1.15.0-f3a393abcc.patch
Normal file
39
.yarn/patches/sortablejs-npm-1.15.0-f3a393abcc.patch
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
diff --git a/modular/sortable.complete.esm.js b/modular/sortable.complete.esm.js
|
||||||
|
index 02e9f2d6bebeb430fe6e7c1cc3f9c3c9df051f14..bb8268b0844a1faa4108cc92c0be2a3dbaf23f83 100644
|
||||||
|
--- a/modular/sortable.complete.esm.js
|
||||||
|
+++ b/modular/sortable.complete.esm.js
|
||||||
|
@@ -1657,7 +1657,7 @@ Sortable.prototype =
|
||||||
|
target = parent; // store last element
|
||||||
|
}
|
||||||
|
/* jshint boss:true */
|
||||||
|
- while (parent = parent.parentNode);
|
||||||
|
+ while (parent = parent.parentNode || parent.getRootNode().host);
|
||||||
|
}
|
||||||
|
|
||||||
|
_unhideGhostForTarget();
|
||||||
|
diff --git a/modular/sortable.core.esm.js b/modular/sortable.core.esm.js
|
||||||
|
index b04c8b4634f7c6b4ef1aadbb48afe6564306dea9..39a107163c8c336ebd669b5ea8a936af87e1c1e7 100644
|
||||||
|
--- a/modular/sortable.core.esm.js
|
||||||
|
+++ b/modular/sortable.core.esm.js
|
||||||
|
@@ -1657,7 +1657,7 @@ Sortable.prototype =
|
||||||
|
target = parent; // store last element
|
||||||
|
}
|
||||||
|
/* jshint boss:true */
|
||||||
|
- while (parent = parent.parentNode);
|
||||||
|
+ while (parent = parent.parentNode || parent.getRootNode().host);
|
||||||
|
}
|
||||||
|
|
||||||
|
_unhideGhostForTarget();
|
||||||
|
diff --git a/modular/sortable.esm.js b/modular/sortable.esm.js
|
||||||
|
index 6ec7ed1bb557e21c2578200161e989c65d23150b..0a05475a22904472fac6c13f524c674da76584b0 100644
|
||||||
|
--- a/modular/sortable.esm.js
|
||||||
|
+++ b/modular/sortable.esm.js
|
||||||
|
@@ -1657,7 +1657,7 @@ Sortable.prototype =
|
||||||
|
target = parent; // store last element
|
||||||
|
}
|
||||||
|
/* jshint boss:true */
|
||||||
|
- while (parent = parent.parentNode);
|
||||||
|
+ while (parent = parent.parentNode || parent.getRootNode().host);
|
||||||
|
}
|
||||||
|
|
||||||
|
_unhideGhostForTarget();
|
@@ -1,73 +0,0 @@
|
|||||||
diff --git a/modular/sortable.core.esm.js b/modular/sortable.core.esm.js
|
|
||||||
index 93ba17509e2e8583ab241fea6845fbe714c584a2..de0651ddb5dced30d36f7d764da0dd0b441f523f 100644
|
|
||||||
--- a/modular/sortable.core.esm.js
|
|
||||||
+++ b/modular/sortable.core.esm.js
|
|
||||||
@@ -1461,7 +1461,7 @@ Sortable.prototype = /** @lends Sortable.prototype */{
|
|
||||||
}
|
|
||||||
target = parent; // store last element
|
|
||||||
}
|
|
||||||
- /* jshint boss:true */ while (parent = parent.parentNode);
|
|
||||||
+ /* jshint boss:true */ while (parent = parent.parentNode || parent.getRootNode().host);
|
|
||||||
}
|
|
||||||
_unhideGhostForTarget();
|
|
||||||
}
|
|
||||||
@@ -1781,11 +1781,16 @@ Sortable.prototype = /** @lends Sortable.prototype */{
|
|
||||||
}
|
|
||||||
if (_onMove(rootEl, el, dragEl, dragRect, target, targetRect, evt, !!target) !== false) {
|
|
||||||
capture();
|
|
||||||
- if (elLastChild && elLastChild.nextSibling) {
|
|
||||||
- // the last draggable element is not the last node
|
|
||||||
- el.insertBefore(dragEl, elLastChild.nextSibling);
|
|
||||||
- } else {
|
|
||||||
- el.appendChild(dragEl);
|
|
||||||
+ try {
|
|
||||||
+ if (elLastChild && elLastChild.nextSibling) {
|
|
||||||
+ // the last draggable element is not the last node
|
|
||||||
+ el.insertBefore(dragEl, elLastChild.nextSibling);
|
|
||||||
+ } else {
|
|
||||||
+ el.appendChild(dragEl);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ catch(err) {
|
|
||||||
+ return completed(false);
|
|
||||||
}
|
|
||||||
parentEl = el; // actualization
|
|
||||||
|
|
||||||
@@ -1802,7 +1807,13 @@ Sortable.prototype = /** @lends Sortable.prototype */{
|
|
||||||
targetRect = getRect(target);
|
|
||||||
if (_onMove(rootEl, el, dragEl, dragRect, target, targetRect, evt, false) !== false) {
|
|
||||||
capture();
|
|
||||||
- el.insertBefore(dragEl, firstChild);
|
|
||||||
+ try {
|
|
||||||
+ el.insertBefore(dragEl, firstChild);
|
|
||||||
+ }
|
|
||||||
+ catch(err) {
|
|
||||||
+ return completed(false);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
parentEl = el; // actualization
|
|
||||||
|
|
||||||
changed();
|
|
||||||
@@ -1849,12 +1860,17 @@ Sortable.prototype = /** @lends Sortable.prototype */{
|
|
||||||
_silent = true;
|
|
||||||
setTimeout(_unsilent, 30);
|
|
||||||
capture();
|
|
||||||
- if (after && !nextSibling) {
|
|
||||||
- el.appendChild(dragEl);
|
|
||||||
- } else {
|
|
||||||
- target.parentNode.insertBefore(dragEl, after ? nextSibling : target);
|
|
||||||
- }
|
|
||||||
|
|
||||||
+ try {
|
|
||||||
+ if (after && !nextSibling) {
|
|
||||||
+ el.appendChild(dragEl);
|
|
||||||
+ } else {
|
|
||||||
+ target.parentNode.insertBefore(dragEl, after ? nextSibling : target);
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ catch(err) {
|
|
||||||
+ return completed(false);
|
|
||||||
+ }
|
|
||||||
// Undo chrome's scroll adjustment (has no effect on other browsers)
|
|
||||||
if (scrolledPastTop) {
|
|
||||||
scrollBy(scrolledPastTop, 0, scrollBefore - scrolledPastTop.scrollTop);
|
|
541
.yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
vendored
Normal file
541
.yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
vendored
Normal file
File diff suppressed because one or more lines are too long
9
.yarn/plugins/@yarnpkg/plugin-typescript.cjs
vendored
Normal file
9
.yarn/plugins/@yarnpkg/plugin-typescript.cjs
vendored
Normal file
File diff suppressed because one or more lines are too long
874
.yarn/releases/yarn-3.6.3.cjs
vendored
Executable file
874
.yarn/releases/yarn-3.6.3.cjs
vendored
Executable file
File diff suppressed because one or more lines are too long
893
.yarn/releases/yarn-4.1.1.cjs
vendored
893
.yarn/releases/yarn-4.1.1.cjs
vendored
File diff suppressed because one or more lines are too long
12
.yarnrc.yml
12
.yarnrc.yml
@@ -1,9 +1,11 @@
|
|||||||
compressionLevel: mixed
|
|
||||||
|
|
||||||
defaultSemverRangePrefix: ""
|
defaultSemverRangePrefix: ""
|
||||||
|
|
||||||
enableGlobalCache: false
|
|
||||||
|
|
||||||
nodeLinker: node-modules
|
nodeLinker: node-modules
|
||||||
|
|
||||||
yarnPath: .yarn/releases/yarn-4.1.1.cjs
|
plugins:
|
||||||
|
- path: .yarn/plugins/@yarnpkg/plugin-typescript.cjs
|
||||||
|
spec: "@yarnpkg/plugin-typescript"
|
||||||
|
- path: .yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
|
||||||
|
spec: "@yarnpkg/plugin-interactive-tools"
|
||||||
|
|
||||||
|
yarnPath: .yarn/releases/yarn-3.6.3.cjs
|
||||||
|
@@ -1,56 +0,0 @@
|
|||||||
import defineProvider from "@babel/helper-define-polyfill-provider";
|
|
||||||
|
|
||||||
// List of polyfill keys with supported browser targets for the functionality
|
|
||||||
const PolyfillSupport = {
|
|
||||||
fetch: {
|
|
||||||
android: 42,
|
|
||||||
chrome: 42,
|
|
||||||
edge: 14,
|
|
||||||
firefox: 39,
|
|
||||||
ios: 10.3,
|
|
||||||
opera: 29,
|
|
||||||
opera_mobile: 29,
|
|
||||||
safari: 10.1,
|
|
||||||
samsung: 4.0,
|
|
||||||
},
|
|
||||||
proxy: {
|
|
||||||
android: 49,
|
|
||||||
chrome: 49,
|
|
||||||
edge: 12,
|
|
||||||
firefox: 18,
|
|
||||||
ios: 10.0,
|
|
||||||
opera: 36,
|
|
||||||
opera_mobile: 36,
|
|
||||||
safari: 10.0,
|
|
||||||
samsung: 5.0,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
// Map of global variables and/or instance and static properties to the
|
|
||||||
// corresponding polyfill key and actual module to import
|
|
||||||
const polyfillMap = {
|
|
||||||
global: {
|
|
||||||
Proxy: { key: "proxy", module: "proxy-polyfill" },
|
|
||||||
fetch: { key: "fetch", module: "unfetch/polyfill" },
|
|
||||||
},
|
|
||||||
instance: {},
|
|
||||||
static: {},
|
|
||||||
};
|
|
||||||
|
|
||||||
// Create plugin using the same factory as for CoreJS
|
|
||||||
export default defineProvider(
|
|
||||||
({ createMetaResolver, debug, shouldInjectPolyfill }) => {
|
|
||||||
const resolvePolyfill = createMetaResolver(polyfillMap);
|
|
||||||
return {
|
|
||||||
name: "HA Custom",
|
|
||||||
polyfills: PolyfillSupport,
|
|
||||||
usageGlobal(meta, utils) {
|
|
||||||
const polyfill = resolvePolyfill(meta);
|
|
||||||
if (polyfill && shouldInjectPolyfill(polyfill.desc.key)) {
|
|
||||||
debug(polyfill.desc.key);
|
|
||||||
utils.injectGlobalImport(polyfill.desc.module);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
);
|
|
@@ -1,7 +1,6 @@
|
|||||||
const path = require("path");
|
const path = require("path");
|
||||||
const env = require("./env.cjs");
|
const env = require("./env.cjs");
|
||||||
const paths = require("./paths.cjs");
|
const paths = require("./paths.cjs");
|
||||||
const { dependencies } = require("../package.json");
|
|
||||||
|
|
||||||
// GitHub base URL to use for production source maps
|
// GitHub base URL to use for production source maps
|
||||||
// Nightly builds use the commit SHA, otherwise assumes there is a tag that matches the version
|
// Nightly builds use the commit SHA, otherwise assumes there is a tag that matches the version
|
||||||
@@ -13,7 +12,11 @@ module.exports.sourceMapURL = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
// Files from NPM Packages that should not be imported
|
// Files from NPM Packages that should not be imported
|
||||||
module.exports.ignorePackages = () => [];
|
// eslint-disable-next-line unused-imports/no-unused-vars
|
||||||
|
module.exports.ignorePackages = ({ latestBuild }) => [
|
||||||
|
// Part of yaml.js and only used for !!js functions that we don't use
|
||||||
|
require.resolve("esprima"),
|
||||||
|
];
|
||||||
|
|
||||||
// Files from NPM packages that we should replace with empty file
|
// Files from NPM packages that we should replace with empty file
|
||||||
module.exports.emptyPackages = ({ latestBuild, isHassioBuild }) =>
|
module.exports.emptyPackages = ({ latestBuild, isHassioBuild }) =>
|
||||||
@@ -32,6 +35,8 @@ module.exports.emptyPackages = ({ latestBuild, isHassioBuild }) =>
|
|||||||
require.resolve(
|
require.resolve(
|
||||||
path.resolve(paths.polymer_dir, "src/resources/compatibility.ts")
|
path.resolve(paths.polymer_dir, "src/resources/compatibility.ts")
|
||||||
),
|
),
|
||||||
|
// This polyfill is loaded in workers to support ES5, filter it out.
|
||||||
|
latestBuild && require.resolve("proxy-polyfill/src/index.js"),
|
||||||
// Icons in supervisor conflict with icons in HA so we don't load.
|
// Icons in supervisor conflict with icons in HA so we don't load.
|
||||||
isHassioBuild &&
|
isHassioBuild &&
|
||||||
require.resolve(
|
require.resolve(
|
||||||
@@ -86,12 +91,14 @@ module.exports.babelOptions = ({ latestBuild, isProdBuild, isTestBuild }) => ({
|
|||||||
setSpreadProperties: true,
|
setSpreadProperties: true,
|
||||||
},
|
},
|
||||||
browserslistEnv: latestBuild ? "modern" : "legacy",
|
browserslistEnv: latestBuild ? "modern" : "legacy",
|
||||||
|
// Must be unambiguous because some dependencies are CommonJS only
|
||||||
|
sourceType: "unambiguous",
|
||||||
presets: [
|
presets: [
|
||||||
[
|
[
|
||||||
"@babel/preset-env",
|
"@babel/preset-env",
|
||||||
{
|
{
|
||||||
useBuiltIns: latestBuild ? false : "usage",
|
useBuiltIns: latestBuild ? false : "entry",
|
||||||
corejs: latestBuild ? false : dependencies["core-js"],
|
corejs: latestBuild ? false : { version: "3.32", proposals: true },
|
||||||
bugfixes: true,
|
bugfixes: true,
|
||||||
shippedProposals: true,
|
shippedProposals: true,
|
||||||
},
|
},
|
||||||
@@ -109,39 +116,27 @@ module.exports.babelOptions = ({ latestBuild, isProdBuild, isTestBuild }) => ({
|
|||||||
ignoreModuleNotFound: true,
|
ignoreModuleNotFound: true,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
[
|
|
||||||
path.resolve(
|
|
||||||
paths.polymer_dir,
|
|
||||||
"build-scripts/babel-plugins/custom-polyfill-plugin.js"
|
|
||||||
),
|
|
||||||
{ method: "usage-global" },
|
|
||||||
],
|
|
||||||
// Minify template literals for production
|
// Minify template literals for production
|
||||||
isProdBuild && [
|
isProdBuild && [
|
||||||
"template-html-minifier",
|
"template-html-minifier",
|
||||||
{
|
{
|
||||||
modules: {
|
modules: {
|
||||||
...Object.fromEntries(
|
lit: [
|
||||||
["lit", "lit-element", "lit-html"].map((m) => [
|
"html",
|
||||||
m,
|
{ name: "svg", encapsulation: "svg" },
|
||||||
[
|
{ name: "css", encapsulation: "style" },
|
||||||
"html",
|
],
|
||||||
{ name: "svg", encapsulation: "svg" },
|
"@polymer/polymer/lib/utils/html-tag": ["html"],
|
||||||
{ name: "css", encapsulation: "style" },
|
|
||||||
],
|
|
||||||
])
|
|
||||||
),
|
|
||||||
"@polymer/polymer/lib/utils/html-tag.js": ["html"],
|
|
||||||
},
|
},
|
||||||
strictCSS: true,
|
strictCSS: true,
|
||||||
htmlMinifier: module.exports.htmlMinifierOptions,
|
htmlMinifier: module.exports.htmlMinifierOptions,
|
||||||
failOnError: false, // we can turn this off in case of false positives
|
failOnError: true, // we can turn this off in case of false positives
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
// Import helpers and regenerator from runtime package
|
// Import helpers and regenerator from runtime package
|
||||||
[
|
[
|
||||||
"@babel/plugin-transform-runtime",
|
"@babel/plugin-transform-runtime",
|
||||||
{ version: dependencies["@babel/runtime"] },
|
{ version: require("../package.json").dependencies["@babel/runtime"] },
|
||||||
],
|
],
|
||||||
// Support some proposals still in TC39 process
|
// Support some proposals still in TC39 process
|
||||||
["@babel/plugin-proposal-decorators", { decoratorsBeforeExport: true }],
|
["@babel/plugin-proposal-decorators", { decoratorsBeforeExport: true }],
|
||||||
@@ -152,21 +147,9 @@ module.exports.babelOptions = ({ latestBuild, isProdBuild, isTestBuild }) => ({
|
|||||||
/node_modules[\\/]webpack[\\/]buildin/,
|
/node_modules[\\/]webpack[\\/]buildin/,
|
||||||
],
|
],
|
||||||
sourceMaps: !isTestBuild,
|
sourceMaps: !isTestBuild,
|
||||||
overrides: [
|
|
||||||
{
|
|
||||||
// Use unambiguous for dependencies so that require() is correctly injected into CommonJS files
|
|
||||||
// Exclusions are needed in some cases where ES modules have no static imports or exports, such as polyfills
|
|
||||||
sourceType: "unambiguous",
|
|
||||||
include: /\/node_modules\//,
|
|
||||||
exclude: [
|
|
||||||
"element-internals-polyfill",
|
|
||||||
"@?lit(?:-labs|-element|-html)?",
|
|
||||||
].map((p) => new RegExp(`/node_modules/${p}/`)),
|
|
||||||
},
|
|
||||||
],
|
|
||||||
});
|
});
|
||||||
|
|
||||||
const nameSuffix = (latestBuild) => (latestBuild ? "-modern" : "-legacy");
|
const nameSuffix = (latestBuild) => (latestBuild ? "-latest" : "-es5");
|
||||||
|
|
||||||
const outputPath = (outputRoot, latestBuild) =>
|
const outputPath = (outputRoot, latestBuild) =>
|
||||||
path.resolve(outputRoot, latestBuild ? "frontend_latest" : "frontend_es5");
|
path.resolve(outputRoot, latestBuild ? "frontend_latest" : "frontend_es5");
|
||||||
@@ -200,7 +183,7 @@ const publicPath = (latestBuild, root = "") =>
|
|||||||
module.exports.config = {
|
module.exports.config = {
|
||||||
app({ isProdBuild, latestBuild, isStatsBuild, isTestBuild, isWDS }) {
|
app({ isProdBuild, latestBuild, isStatsBuild, isTestBuild, isWDS }) {
|
||||||
return {
|
return {
|
||||||
name: "frontend" + nameSuffix(latestBuild),
|
name: "app" + nameSuffix(latestBuild),
|
||||||
entry: {
|
entry: {
|
||||||
service_worker: "./src/entrypoints/service_worker.ts",
|
service_worker: "./src/entrypoints/service_worker.ts",
|
||||||
app: "./src/entrypoints/app.ts",
|
app: "./src/entrypoints/app.ts",
|
||||||
|
@@ -30,8 +30,8 @@ gulp.task(
|
|||||||
env.useWDS()
|
env.useWDS()
|
||||||
? "wds-watch-app"
|
? "wds-watch-app"
|
||||||
: env.useRollup()
|
: env.useRollup()
|
||||||
? "rollup-watch-app"
|
? "rollup-watch-app"
|
||||||
: "webpack-watch-app"
|
: "webpack-watch-app"
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -45,8 +45,8 @@ gulp.task(
|
|||||||
gulp.parallel("gen-icons-json", "build-translations", "build-locale-data"),
|
gulp.parallel("gen-icons-json", "build-translations", "build-locale-data"),
|
||||||
"copy-static-app",
|
"copy-static-app",
|
||||||
env.useRollup() ? "rollup-prod-app" : "webpack-prod-app",
|
env.useRollup() ? "rollup-prod-app" : "webpack-prod-app",
|
||||||
gulp.parallel("gen-pages-app-prod", "gen-service-worker-app-prod"),
|
|
||||||
// Don't compress running tests
|
// Don't compress running tests
|
||||||
...(env.isTestBuild() ? [] : ["compress-app"])
|
...(env.isTestBuild() ? [] : ["compress-app"]),
|
||||||
|
gulp.parallel("gen-pages-app-prod", "gen-service-worker-app-prod")
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
@@ -8,10 +8,7 @@ const zopfliOptions = { threshold: 150 };
|
|||||||
|
|
||||||
const compressDist = (rootDir) =>
|
const compressDist = (rootDir) =>
|
||||||
gulp
|
gulp
|
||||||
.src([
|
.src([`${rootDir}/**/*.{js,json,css,svg}`])
|
||||||
`${rootDir}/**/*.{js,json,css,svg,xml}`,
|
|
||||||
`${rootDir}/{authorize,onboarding}.html`,
|
|
||||||
])
|
|
||||||
.pipe(zopfli(zopfliOptions))
|
.pipe(zopfli(zopfliOptions))
|
||||||
.pipe(gulp.dest(rootDir));
|
.pipe(gulp.dest(rootDir));
|
||||||
|
|
||||||
|
@@ -161,10 +161,6 @@ gulp.task("fetch-lokalise", async function () {
|
|||||||
})
|
})
|
||||||
);
|
);
|
||||||
})
|
})
|
||||||
.catch((err) => {
|
|
||||||
console.error(err);
|
|
||||||
throw err;
|
|
||||||
})
|
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
@@ -4,7 +4,6 @@ import fs from "fs-extra";
|
|||||||
import gulp from "gulp";
|
import gulp from "gulp";
|
||||||
import path from "path";
|
import path from "path";
|
||||||
import paths from "../paths.cjs";
|
import paths from "../paths.cjs";
|
||||||
import env from "../env.cjs";
|
|
||||||
|
|
||||||
const npmPath = (...parts) =>
|
const npmPath = (...parts) =>
|
||||||
path.resolve(paths.polymer_dir, "node_modules", ...parts);
|
path.resolve(paths.polymer_dir, "node_modules", ...parts);
|
||||||
@@ -63,9 +62,6 @@ function copyPolyfills(staticDir) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function copyLoaderJS(staticDir) {
|
function copyLoaderJS(staticDir) {
|
||||||
if (!env.useRollup()) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const staticPath = genStaticPath(staticDir);
|
const staticPath = genStaticPath(staticDir);
|
||||||
copyFileDir(npmPath("systemjs/dist/s.min.js"), staticPath("js"));
|
copyFileDir(npmPath("systemjs/dist/s.min.js"), staticPath("js"));
|
||||||
copyFileDir(npmPath("systemjs/dist/s.min.js.map"), staticPath("js"));
|
copyFileDir(npmPath("systemjs/dist/s.min.js.map"), staticPath("js"));
|
||||||
|
@@ -1,54 +1,51 @@
|
|||||||
import { deleteSync } from "del";
|
import { deleteSync } from "del";
|
||||||
import { mkdir, readFile, writeFile } from "fs/promises";
|
import { mkdir, readFile, writeFile } from "fs/promises";
|
||||||
import gulp from "gulp";
|
import gulp from "gulp";
|
||||||
import { join, resolve } from "node:path";
|
import path from "path";
|
||||||
import paths from "../paths.cjs";
|
import paths from "../paths.cjs";
|
||||||
|
|
||||||
const formatjsDir = join(paths.polymer_dir, "node_modules", "@formatjs");
|
const outDir = path.join(paths.build_dir, "locale-data");
|
||||||
const outDir = join(paths.build_dir, "locale-data");
|
|
||||||
|
|
||||||
const INTL_POLYFILLS = {
|
const INTL_PACKAGES = {
|
||||||
|
"intl-relativetimeformat": "RelativeTimeFormat",
|
||||||
"intl-datetimeformat": "DateTimeFormat",
|
"intl-datetimeformat": "DateTimeFormat",
|
||||||
|
"intl-numberformat": "NumberFormat",
|
||||||
"intl-displaynames": "DisplayNames",
|
"intl-displaynames": "DisplayNames",
|
||||||
"intl-listformat": "ListFormat",
|
"intl-listformat": "ListFormat",
|
||||||
"intl-numberformat": "NumberFormat",
|
|
||||||
"intl-relativetimeformat": "RelativeTimeFormat",
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const convertToJSON = async (
|
const convertToJSON = async (pkg, lang) => {
|
||||||
pkg,
|
|
||||||
lang,
|
|
||||||
subDir = "locale-data",
|
|
||||||
addFunc = "__addLocaleData",
|
|
||||||
skipMissing = true
|
|
||||||
) => {
|
|
||||||
let localeData;
|
let localeData;
|
||||||
try {
|
try {
|
||||||
localeData = await readFile(
|
localeData = await readFile(
|
||||||
join(formatjsDir, pkg, subDir, `${lang}.js`),
|
path.resolve(
|
||||||
|
paths.polymer_dir,
|
||||||
|
`node_modules/@formatjs/${pkg}/locale-data/${lang}.js`
|
||||||
|
),
|
||||||
"utf-8"
|
"utf-8"
|
||||||
);
|
);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
// Ignore if language is missing (i.e. not supported by @formatjs)
|
// Ignore if language is missing (i.e. not supported by @formatjs)
|
||||||
if (e.code === "ENOENT" && skipMissing) {
|
if (e.code === "ENOENT") {
|
||||||
console.warn(`Skipped missing data for language ${lang} from ${pkg}`);
|
|
||||||
return;
|
return;
|
||||||
|
} else {
|
||||||
|
throw e;
|
||||||
}
|
}
|
||||||
throw e;
|
|
||||||
}
|
}
|
||||||
// Convert to JSON
|
// Convert to JSON
|
||||||
const obj = INTL_POLYFILLS[pkg];
|
const className = INTL_PACKAGES[pkg];
|
||||||
const dataRegex = new RegExp(
|
localeData = localeData
|
||||||
`Intl\\.${obj}\\.${addFunc}\\((?<data>.*)\\)`,
|
.replace(
|
||||||
"s"
|
new RegExp(
|
||||||
);
|
`\\/\\*\\s*@generated\\s*\\*\\/\\s*\\/\\/\\s*prettier-ignore\\s*if\\s*\\(Intl\\.${className}\\s*&&\\s*typeof\\s*Intl\\.${className}\\.__addLocaleData\\s*===\\s*'function'\\)\\s*{\\s*Intl\\.${className}\\.__addLocaleData\\(`,
|
||||||
localeData = localeData.match(dataRegex)?.groups?.data;
|
"im"
|
||||||
if (!localeData) {
|
),
|
||||||
throw Error(`Failed to extract data for language ${lang} from ${pkg}`);
|
""
|
||||||
}
|
)
|
||||||
|
.replace(/\)\s*}/im, "");
|
||||||
// Parse to validate JSON, then stringify to minify
|
// Parse to validate JSON, then stringify to minify
|
||||||
localeData = JSON.stringify(JSON.parse(localeData));
|
localeData = JSON.stringify(JSON.parse(localeData));
|
||||||
await writeFile(join(outDir, `${pkg}/${lang}.json`), localeData);
|
await writeFile(path.join(outDir, `${pkg}/${lang}.json`), localeData);
|
||||||
};
|
};
|
||||||
|
|
||||||
gulp.task("clean-locale-data", async () => deleteSync([outDir]));
|
gulp.task("clean-locale-data", async () => deleteSync([outDir]));
|
||||||
@@ -56,27 +53,17 @@ gulp.task("clean-locale-data", async () => deleteSync([outDir]));
|
|||||||
gulp.task("create-locale-data", async () => {
|
gulp.task("create-locale-data", async () => {
|
||||||
const translationMeta = JSON.parse(
|
const translationMeta = JSON.parse(
|
||||||
await readFile(
|
await readFile(
|
||||||
resolve(paths.translations_src, "translationMetadata.json"),
|
path.resolve(paths.translations_src, "translationMetadata.json"),
|
||||||
"utf-8"
|
"utf-8"
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
const conversions = [];
|
const conversions = [];
|
||||||
for (const pkg of Object.keys(INTL_POLYFILLS)) {
|
for (const pkg of Object.keys(INTL_PACKAGES)) {
|
||||||
// eslint-disable-next-line no-await-in-loop
|
await mkdir(path.join(outDir, pkg), { recursive: true });
|
||||||
await mkdir(join(outDir, pkg), { recursive: true });
|
|
||||||
for (const lang of Object.keys(translationMeta)) {
|
for (const lang of Object.keys(translationMeta)) {
|
||||||
conversions.push(convertToJSON(pkg, lang));
|
conversions.push(convertToJSON(pkg, lang));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
conversions.push(
|
|
||||||
convertToJSON(
|
|
||||||
"intl-datetimeformat",
|
|
||||||
"add-all-tz",
|
|
||||||
".",
|
|
||||||
"__addTZData",
|
|
||||||
false
|
|
||||||
)
|
|
||||||
);
|
|
||||||
await Promise.all(conversions);
|
await Promise.all(conversions);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@@ -1,7 +1,12 @@
|
|||||||
import { createHash } from "crypto";
|
import { createHash } from "crypto";
|
||||||
import { deleteSync } from "del";
|
import { deleteSync } from "del";
|
||||||
import { mkdirSync, readdirSync, readFileSync, renameSync } from "fs";
|
import {
|
||||||
import { writeFile } from "node:fs/promises";
|
mkdirSync,
|
||||||
|
readdirSync,
|
||||||
|
readFileSync,
|
||||||
|
renameSync,
|
||||||
|
writeFile,
|
||||||
|
} from "fs";
|
||||||
import gulp from "gulp";
|
import gulp from "gulp";
|
||||||
import flatmap from "gulp-flatmap";
|
import flatmap from "gulp-flatmap";
|
||||||
import transform from "gulp-json-transform";
|
import transform from "gulp-json-transform";
|
||||||
@@ -131,23 +136,27 @@ gulp.task("ensure-translations-build-dir", async () => {
|
|||||||
mkdirSync(workDir, { recursive: true });
|
mkdirSync(workDir, { recursive: true });
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task("create-test-metadata", () =>
|
gulp.task("create-test-metadata", (cb) => {
|
||||||
env.isProdBuild()
|
writeFile(
|
||||||
? Promise.resolve()
|
workDir + "/testMetadata.json",
|
||||||
: writeFile(
|
JSON.stringify({
|
||||||
workDir + "/testMetadata.json",
|
test: {
|
||||||
JSON.stringify({ test: { nativeName: "Test" } })
|
nativeName: "Test",
|
||||||
)
|
},
|
||||||
);
|
}),
|
||||||
|
cb
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
gulp.task("create-test-translation", () =>
|
gulp.task(
|
||||||
env.isProdBuild()
|
"create-test-translation",
|
||||||
? Promise.resolve()
|
gulp.series("create-test-metadata", () =>
|
||||||
: gulp
|
gulp
|
||||||
.src(path.join(paths.translations_src, "en.json"))
|
.src(path.join(paths.translations_src, "en.json"))
|
||||||
.pipe(transform((data, _file) => recursiveEmpty(data)))
|
.pipe(transform((data, _file) => recursiveEmpty(data)))
|
||||||
.pipe(rename("test.json"))
|
.pipe(rename("test.json"))
|
||||||
.pipe(gulp.dest(workDir))
|
.pipe(gulp.dest(workDir))
|
||||||
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -179,11 +188,16 @@ gulp.task("build-master-translation", () => {
|
|||||||
|
|
||||||
gulp.task("build-merged-translations", () =>
|
gulp.task("build-merged-translations", () =>
|
||||||
gulp
|
gulp
|
||||||
.src([
|
.src(
|
||||||
inFrontendDir + "/*.json",
|
[
|
||||||
"!" + inFrontendDir + "/en.json",
|
inFrontendDir + "/*.json",
|
||||||
...(env.isProdBuild() ? [] : [workDir + "/test.json"]),
|
"!" + inFrontendDir + "/en.json",
|
||||||
])
|
workDir + "/test.json",
|
||||||
|
],
|
||||||
|
{
|
||||||
|
allowEmpty: true,
|
||||||
|
}
|
||||||
|
)
|
||||||
.pipe(transform((data, file) => lokaliseTransform(data, data, file)))
|
.pipe(transform((data, file) => lokaliseTransform(data, data, file)))
|
||||||
.pipe(
|
.pipe(
|
||||||
flatmap((stream, file) => {
|
flatmap((stream, file) => {
|
||||||
@@ -363,11 +377,14 @@ gulp.task("build-translation-flatten-supervisor", () =>
|
|||||||
|
|
||||||
gulp.task("build-translation-write-metadata", () =>
|
gulp.task("build-translation-write-metadata", () =>
|
||||||
gulp
|
gulp
|
||||||
.src([
|
.src(
|
||||||
path.join(paths.translations_src, "translationMetadata.json"),
|
[
|
||||||
...(env.isProdBuild() ? [] : [workDir + "/testMetadata.json"]),
|
path.join(paths.translations_src, "translationMetadata.json"),
|
||||||
workDir + "/translationFingerprints.json",
|
workDir + "/testMetadata.json",
|
||||||
])
|
workDir + "/translationFingerprints.json",
|
||||||
|
],
|
||||||
|
{ allowEmpty: true }
|
||||||
|
)
|
||||||
.pipe(merge({}))
|
.pipe(merge({}))
|
||||||
.pipe(
|
.pipe(
|
||||||
transform((data) => {
|
transform((data) => {
|
||||||
@@ -398,7 +415,7 @@ gulp.task("build-translation-write-metadata", () =>
|
|||||||
gulp.task(
|
gulp.task(
|
||||||
"create-translations",
|
"create-translations",
|
||||||
gulp.series(
|
gulp.series(
|
||||||
gulp.parallel("create-test-metadata", "create-test-translation"),
|
...(env.isProdBuild() ? [] : ["create-test-translation"]),
|
||||||
"build-master-translation",
|
"build-master-translation",
|
||||||
"build-merged-translations",
|
"build-merged-translations",
|
||||||
gulp.parallel(...splitTasks),
|
gulp.parallel(...splitTasks),
|
||||||
@@ -426,7 +443,6 @@ gulp.task(
|
|||||||
"fetch-nightly-translations",
|
"fetch-nightly-translations",
|
||||||
gulp.series("clean-translations", "ensure-translations-build-dir")
|
gulp.series("clean-translations", "ensure-translations-build-dir")
|
||||||
),
|
),
|
||||||
gulp.parallel("create-test-metadata", "create-test-translation"),
|
|
||||||
"build-master-translation",
|
"build-master-translation",
|
||||||
"build-merged-translations",
|
"build-merged-translations",
|
||||||
"build-translation-fragment-supervisor",
|
"build-translation-fragment-supervisor",
|
||||||
|
@@ -115,9 +115,7 @@ gulp.task("webpack-prod-app", () =>
|
|||||||
|
|
||||||
gulp.task("webpack-dev-server-demo", () =>
|
gulp.task("webpack-dev-server-demo", () =>
|
||||||
runDevServer({
|
runDevServer({
|
||||||
compiler: webpack(
|
compiler: webpack(bothBuilds(createDemoConfig, { isProdBuild: false })),
|
||||||
createDemoConfig({ isProdBuild: false, latestBuild: true })
|
|
||||||
),
|
|
||||||
contentBase: paths.demo_output_root,
|
contentBase: paths.demo_output_root,
|
||||||
port: 8090,
|
port: 8090,
|
||||||
})
|
})
|
||||||
@@ -133,9 +131,7 @@ gulp.task("webpack-prod-demo", () =>
|
|||||||
|
|
||||||
gulp.task("webpack-dev-server-cast", () =>
|
gulp.task("webpack-dev-server-cast", () =>
|
||||||
runDevServer({
|
runDevServer({
|
||||||
compiler: webpack(
|
compiler: webpack(bothBuilds(createCastConfig, { isProdBuild: false })),
|
||||||
createCastConfig({ isProdBuild: false, latestBuild: true })
|
|
||||||
),
|
|
||||||
contentBase: paths.cast_output_root,
|
contentBase: paths.cast_output_root,
|
||||||
port: 8080,
|
port: 8080,
|
||||||
// Accessible from the network, because that's how Cast hits it.
|
// Accessible from the network, because that's how Cast hits it.
|
||||||
@@ -178,9 +174,8 @@ gulp.task("webpack-prod-hassio", () =>
|
|||||||
|
|
||||||
gulp.task("webpack-dev-server-gallery", () =>
|
gulp.task("webpack-dev-server-gallery", () =>
|
||||||
runDevServer({
|
runDevServer({
|
||||||
compiler: webpack(
|
// We don't use the es5 build, but the dev server will fuck up the publicPath if we don't
|
||||||
createGalleryConfig({ isProdBuild: false, latestBuild: true })
|
compiler: webpack(bothBuilds(createGalleryConfig, { isProdBuild: false })),
|
||||||
),
|
|
||||||
contentBase: paths.gallery_output_root,
|
contentBase: paths.gallery_output_root,
|
||||||
port: 8100,
|
port: 8100,
|
||||||
listenHost: "0.0.0.0",
|
listenHost: "0.0.0.0",
|
||||||
|
@@ -1,15 +1,10 @@
|
|||||||
const { existsSync } = require("fs");
|
const { existsSync } = require("fs");
|
||||||
const path = require("path");
|
const path = require("path");
|
||||||
const webpack = require("webpack");
|
const webpack = require("webpack");
|
||||||
const { StatsWriterPlugin } = require("webpack-stats-plugin");
|
|
||||||
const filterStats = require("@bundle-stats/plugin-webpack-filter").default;
|
|
||||||
const TerserPlugin = require("terser-webpack-plugin");
|
const TerserPlugin = require("terser-webpack-plugin");
|
||||||
const { WebpackManifestPlugin } = require("webpack-manifest-plugin");
|
const { WebpackManifestPlugin } = require("webpack-manifest-plugin");
|
||||||
const log = require("fancy-log");
|
const log = require("fancy-log");
|
||||||
const WebpackBar = require("webpackbar");
|
const WebpackBar = require("webpackbar");
|
||||||
const {
|
|
||||||
TransformAsyncModulesPlugin,
|
|
||||||
} = require("transform-async-modules-webpack-plugin");
|
|
||||||
const paths = require("./paths.cjs");
|
const paths = require("./paths.cjs");
|
||||||
const bundle = require("./bundle.cjs");
|
const bundle = require("./bundle.cjs");
|
||||||
|
|
||||||
@@ -54,8 +49,8 @@ const createWebpackConfig = ({
|
|||||||
devtool: isTestBuild
|
devtool: isTestBuild
|
||||||
? false
|
? false
|
||||||
: isProdBuild
|
: isProdBuild
|
||||||
? "nosources-source-map"
|
? "nosources-source-map"
|
||||||
: "eval-cheap-module-source-map",
|
: "eval-cheap-module-source-map",
|
||||||
entry,
|
entry,
|
||||||
node: false,
|
node: false,
|
||||||
module: {
|
module: {
|
||||||
@@ -145,18 +140,18 @@ const createWebpackConfig = ({
|
|||||||
),
|
),
|
||||||
path.resolve(paths.polymer_dir, "src/util/empty.js")
|
path.resolve(paths.polymer_dir, "src/util/empty.js")
|
||||||
),
|
),
|
||||||
!isProdBuild && new LogStartCompilePlugin(),
|
// See `src/resources/intl-polyfill-legacy.ts` for explanation
|
||||||
isProdBuild &&
|
|
||||||
new StatsWriterPlugin({
|
|
||||||
filename: path.relative(
|
|
||||||
outputPath,
|
|
||||||
path.join(paths.build_dir, "stats", `${name}.json`)
|
|
||||||
),
|
|
||||||
stats: { assets: true, chunks: true, modules: true },
|
|
||||||
transform: (stats) => JSON.stringify(filterStats(stats)),
|
|
||||||
}),
|
|
||||||
!latestBuild &&
|
!latestBuild &&
|
||||||
new TransformAsyncModulesPlugin({ browserslistEnv: "legacy" }),
|
new webpack.NormalModuleReplacementPlugin(
|
||||||
|
new RegExp(
|
||||||
|
path.resolve(paths.polymer_dir, "src/resources/intl-polyfill.ts")
|
||||||
|
),
|
||||||
|
path.resolve(
|
||||||
|
paths.polymer_dir,
|
||||||
|
"src/resources/intl-polyfill-legacy.ts"
|
||||||
|
)
|
||||||
|
),
|
||||||
|
!isProdBuild && new LogStartCompilePlugin(),
|
||||||
].filter(Boolean),
|
].filter(Boolean),
|
||||||
resolve: {
|
resolve: {
|
||||||
extensions: [".ts", ".js", ".json"],
|
extensions: [".ts", ".js", ".json"],
|
||||||
@@ -176,8 +171,6 @@ const createWebpackConfig = ({
|
|||||||
"@lit-labs/virtualizer/layouts/grid.js",
|
"@lit-labs/virtualizer/layouts/grid.js",
|
||||||
"@lit-labs/virtualizer/polyfills/resize-observer-polyfill/ResizeObserver":
|
"@lit-labs/virtualizer/polyfills/resize-observer-polyfill/ResizeObserver":
|
||||||
"@lit-labs/virtualizer/polyfills/resize-observer-polyfill/ResizeObserver.js",
|
"@lit-labs/virtualizer/polyfills/resize-observer-polyfill/ResizeObserver.js",
|
||||||
"@lit-labs/observers/resize-controller":
|
|
||||||
"@lit-labs/observers/resize-controller.js",
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
output: {
|
output: {
|
||||||
@@ -185,12 +178,11 @@ const createWebpackConfig = ({
|
|||||||
filename: ({ chunk }) =>
|
filename: ({ chunk }) =>
|
||||||
!isProdBuild || isStatsBuild || dontHash.has(chunk.name)
|
!isProdBuild || isStatsBuild || dontHash.has(chunk.name)
|
||||||
? "[name].js"
|
? "[name].js"
|
||||||
: "[name].[contenthash].js",
|
: "[name]-[contenthash].js",
|
||||||
chunkFilename:
|
chunkFilename:
|
||||||
isProdBuild && !isStatsBuild ? "[name].[contenthash].js" : "[name].js",
|
isProdBuild && !isStatsBuild ? "[id]-[contenthash].js" : "[name].js",
|
||||||
assetModuleFilename:
|
assetModuleFilename:
|
||||||
isProdBuild && !isStatsBuild ? "[id].[contenthash][ext]" : "[id][ext]",
|
isProdBuild && !isStatsBuild ? "[id]-[contenthash][ext]" : "[id][ext]",
|
||||||
crossOriginLoading: "use-credentials",
|
|
||||||
hashFunction: "xxhash64",
|
hashFunction: "xxhash64",
|
||||||
hashDigest: "base64url",
|
hashDigest: "base64url",
|
||||||
hashDigestLength: 11, // full length of 64 bit base64url
|
hashDigestLength: 11, // full length of 64 bit base64url
|
||||||
|
Binary file not shown.
Before Width: | Height: | Size: 9.8 KiB |
@@ -1,4 +1,4 @@
|
|||||||
import "../../../src/resources/safari-14-attachshadow-patch";
|
import "../../../src/resources/safari-14-attachshadow-patch";
|
||||||
|
import "../../../src/resources/ha-style";
|
||||||
|
import "../../../src/resources/roboto";
|
||||||
import "./layout/hc-connect";
|
import "./layout/hc-connect";
|
||||||
|
|
||||||
import("../../../src/resources/ha-style");
|
|
||||||
|
@@ -3,7 +3,7 @@ import { mdiCast, mdiCastConnected } from "@mdi/js";
|
|||||||
import "@polymer/paper-item/paper-icon-item";
|
import "@polymer/paper-item/paper-icon-item";
|
||||||
import "@polymer/paper-listbox/paper-listbox";
|
import "@polymer/paper-listbox/paper-listbox";
|
||||||
import { Auth, Connection } from "home-assistant-js-websocket";
|
import { Auth, Connection } from "home-assistant-js-websocket";
|
||||||
import { CSSResultGroup, LitElement, TemplateResult, css, html } from "lit";
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import { CastManager } from "../../../../src/cast/cast_manager";
|
import { CastManager } from "../../../../src/cast/cast_manager";
|
||||||
import {
|
import {
|
||||||
@@ -22,27 +22,26 @@ import "../../../../src/components/ha-svg-icon";
|
|||||||
import {
|
import {
|
||||||
getLegacyLovelaceCollection,
|
getLegacyLovelaceCollection,
|
||||||
getLovelaceCollection,
|
getLovelaceCollection,
|
||||||
|
LovelaceConfig,
|
||||||
} from "../../../../src/data/lovelace";
|
} from "../../../../src/data/lovelace";
|
||||||
import { isStrategyDashboard } from "../../../../src/data/lovelace/config/types";
|
|
||||||
import { LovelaceViewConfig } from "../../../../src/data/lovelace/config/view";
|
|
||||||
import "../../../../src/layouts/hass-loading-screen";
|
import "../../../../src/layouts/hass-loading-screen";
|
||||||
import { generateDefaultViewConfig } from "../../../../src/panels/lovelace/common/generate-lovelace-config";
|
import { generateDefaultViewConfig } from "../../../../src/panels/lovelace/common/generate-lovelace-config";
|
||||||
import "./hc-layout";
|
import "./hc-layout";
|
||||||
|
|
||||||
@customElement("hc-cast")
|
@customElement("hc-cast")
|
||||||
class HcCast extends LitElement {
|
class HcCast extends LitElement {
|
||||||
@property({ attribute: false }) public auth!: Auth;
|
@property() public auth!: Auth;
|
||||||
|
|
||||||
@property({ attribute: false }) public connection!: Connection;
|
@property() public connection!: Connection;
|
||||||
|
|
||||||
@property({ attribute: false }) public castManager!: CastManager;
|
@property() public castManager!: CastManager;
|
||||||
|
|
||||||
@state() private askWrite = false;
|
@state() private askWrite = false;
|
||||||
|
|
||||||
@state() private lovelaceViews?: LovelaceViewConfig[] | null;
|
@state() private lovelaceConfig?: LovelaceConfig | null;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
if (this.lovelaceViews === undefined) {
|
if (this.lovelaceConfig === undefined) {
|
||||||
return html`<hass-loading-screen no-toolbar></hass-loading-screen>`;
|
return html`<hass-loading-screen no-toolbar></hass-loading-screen>`;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -73,44 +72,43 @@ class HcCast extends LitElement {
|
|||||||
${error
|
${error
|
||||||
? html` <div class="card-content">${error}</div> `
|
? html` <div class="card-content">${error}</div> `
|
||||||
: !this.castManager.status
|
: !this.castManager.status
|
||||||
? html`
|
? html`
|
||||||
<p class="center-item">
|
<p class="center-item">
|
||||||
<mwc-button raised @click=${this._handleLaunch}>
|
<mwc-button raised @click=${this._handleLaunch}>
|
||||||
<ha-svg-icon .path=${mdiCast}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiCast}></ha-svg-icon>
|
||||||
Start Casting
|
Start Casting
|
||||||
</mwc-button>
|
</mwc-button>
|
||||||
</p>
|
</p>
|
||||||
`
|
`
|
||||||
: html`
|
: html`
|
||||||
<div class="section-header">PICK A VIEW</div>
|
<div class="section-header">PICK A VIEW</div>
|
||||||
<paper-listbox
|
<paper-listbox
|
||||||
attr-for-selected="data-path"
|
attr-for-selected="data-path"
|
||||||
.selected=${this.castManager.status.lovelacePath || ""}
|
.selected=${this.castManager.status.lovelacePath || ""}
|
||||||
>
|
>
|
||||||
${(
|
${(this.lovelaceConfig
|
||||||
this.lovelaceViews ?? [
|
? this.lovelaceConfig.views
|
||||||
generateDefaultViewConfig({}, {}, {}, {}, () => ""),
|
: [generateDefaultViewConfig({}, {}, {}, {}, () => "")]
|
||||||
]
|
).map(
|
||||||
).map(
|
(view, idx) => html`
|
||||||
(view, idx) => html`
|
<paper-icon-item
|
||||||
<paper-icon-item
|
@click=${this._handlePickView}
|
||||||
@click=${this._handlePickView}
|
data-path=${view.path || idx}
|
||||||
data-path=${view.path || idx}
|
>
|
||||||
>
|
${view.icon
|
||||||
${view.icon
|
? html`
|
||||||
? html`
|
<ha-icon
|
||||||
<ha-icon
|
.icon=${view.icon}
|
||||||
.icon=${view.icon}
|
slot="item-icon"
|
||||||
slot="item-icon"
|
></ha-icon>
|
||||||
></ha-icon>
|
`
|
||||||
`
|
: ""}
|
||||||
: ""}
|
${view.title || view.path}
|
||||||
${view.title || view.path}
|
</paper-icon-item>
|
||||||
</paper-icon-item>
|
`
|
||||||
`
|
)}
|
||||||
)}
|
</paper-listbox>
|
||||||
</paper-listbox>
|
`}
|
||||||
`}
|
|
||||||
<div class="card-actions">
|
<div class="card-actions">
|
||||||
${this.castManager.status
|
${this.castManager.status
|
||||||
? html`
|
? html`
|
||||||
@@ -138,15 +136,11 @@ class HcCast extends LitElement {
|
|||||||
llColl.refresh().then(
|
llColl.refresh().then(
|
||||||
() => {
|
() => {
|
||||||
llColl.subscribe((config) => {
|
llColl.subscribe((config) => {
|
||||||
if (isStrategyDashboard(config)) {
|
this.lovelaceConfig = config;
|
||||||
this.lovelaceViews = null;
|
|
||||||
} else {
|
|
||||||
this.lovelaceViews = config.views;
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
async () => {
|
async () => {
|
||||||
this.lovelaceViews = null;
|
this.lovelaceConfig = null;
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -165,7 +159,9 @@ class HcCast extends LitElement {
|
|||||||
toggleAttribute(
|
toggleAttribute(
|
||||||
this,
|
this,
|
||||||
"hide-icons",
|
"hide-icons",
|
||||||
this.lovelaceViews ? !this.lovelaceViews.some((view) => view.icon) : true
|
this.lovelaceConfig
|
||||||
|
? !this.lovelaceConfig.views.some((view) => view.icon)
|
||||||
|
: true
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -241,8 +237,6 @@ class HcCast extends LitElement {
|
|||||||
|
|
||||||
mwc-button ha-svg-icon {
|
mwc-button ha-svg-icon {
|
||||||
margin-right: 8px;
|
margin-right: 8px;
|
||||||
margin-inline-end: 8px;
|
|
||||||
margin-inline-start: initial;
|
|
||||||
height: 18px;
|
height: 18px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1,5 +1,6 @@
|
|||||||
import "@material/mwc-button";
|
import "@material/mwc-button";
|
||||||
import { mdiCastConnected, mdiCast } from "@mdi/js";
|
import { mdiCastConnected, mdiCast } from "@mdi/js";
|
||||||
|
import "@polymer/paper-input/paper-input";
|
||||||
import {
|
import {
|
||||||
Auth,
|
Auth,
|
||||||
Connection,
|
Connection,
|
||||||
@@ -23,7 +24,6 @@ import "../../../../src/components/ha-svg-icon";
|
|||||||
import "../../../../src/layouts/hass-loading-screen";
|
import "../../../../src/layouts/hass-loading-screen";
|
||||||
import { registerServiceWorker } from "../../../../src/util/register-service-worker";
|
import { registerServiceWorker } from "../../../../src/util/register-service-worker";
|
||||||
import "./hc-layout";
|
import "./hc-layout";
|
||||||
import "../../../../src/components/ha-textfield";
|
|
||||||
|
|
||||||
const seeFAQ = (qid) => html`
|
const seeFAQ = (qid) => html`
|
||||||
See <a href="./faq.html${qid ? `#${qid}` : ""}">the FAQ</a> for more
|
See <a href="./faq.html${qid ? `#${qid}` : ""}">the FAQ</a> for more
|
||||||
@@ -33,13 +33,13 @@ const translateErr = (err) =>
|
|||||||
err === ERR_CANNOT_CONNECT
|
err === ERR_CANNOT_CONNECT
|
||||||
? "Unable to connect"
|
? "Unable to connect"
|
||||||
: err === ERR_HASS_HOST_REQUIRED
|
: err === ERR_HASS_HOST_REQUIRED
|
||||||
? "Please enter a Home Assistant URL."
|
? "Please enter a Home Assistant URL."
|
||||||
: err === ERR_INVALID_HTTPS_TO_HTTP
|
: err === ERR_INVALID_HTTPS_TO_HTTP
|
||||||
? html`
|
? html`
|
||||||
Cannot connect to Home Assistant instances over "http://".
|
Cannot connect to Home Assistant instances over "http://".
|
||||||
${seeFAQ("https")}
|
${seeFAQ("https")}
|
||||||
`
|
`
|
||||||
: `Unknown error (${err}).`;
|
: `Unknown error (${err}).`;
|
||||||
|
|
||||||
const INTRO = html`
|
const INTRO = html`
|
||||||
<p>
|
<p>
|
||||||
@@ -116,11 +116,13 @@ export class HcConnect extends LitElement {
|
|||||||
To get started, enter your Home Assistant URL and click authorize.
|
To get started, enter your Home Assistant URL and click authorize.
|
||||||
If you want a preview instead, click the show demo button.
|
If you want a preview instead, click the show demo button.
|
||||||
</p>
|
</p>
|
||||||
<ha-textfield
|
<p>
|
||||||
label="Home Assistant URL"
|
<paper-input
|
||||||
placeholder="https://abcdefghijklmnop.ui.nabu.casa"
|
label="Home Assistant URL"
|
||||||
@keydown=${this._handleInputKeyDown}
|
placeholder="https://abcdefghijklmnop.ui.nabu.casa"
|
||||||
></ha-textfield>
|
@keydown=${this._handleInputKeyDown}
|
||||||
|
></paper-input>
|
||||||
|
</p>
|
||||||
${this.error ? html` <p class="error">${this.error}</p> ` : ""}
|
${this.error ? html` <p class="error">${this.error}</p> ` : ""}
|
||||||
</div>
|
</div>
|
||||||
<div class="card-actions">
|
<div class="card-actions">
|
||||||
@@ -194,7 +196,7 @@ export class HcConnect extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private async _handleConnect() {
|
private async _handleConnect() {
|
||||||
const inputEl = this.shadowRoot!.querySelector("ha-textfield")!;
|
const inputEl = this.shadowRoot!.querySelector("paper-input")!;
|
||||||
const value = inputEl.value || "";
|
const value = inputEl.value || "";
|
||||||
this.error = undefined;
|
this.error = undefined;
|
||||||
|
|
||||||
@@ -313,10 +315,6 @@ export class HcConnect extends LitElement {
|
|||||||
.spacer {
|
.spacer {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
ha-textfield {
|
|
||||||
width: 100%;
|
|
||||||
}
|
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -10,13 +10,13 @@ import "../../../../src/components/ha-card";
|
|||||||
|
|
||||||
@customElement("hc-layout")
|
@customElement("hc-layout")
|
||||||
class HcLayout extends LitElement {
|
class HcLayout extends LitElement {
|
||||||
@property() public subtitle?: string;
|
@property() public subtitle?: string | undefined;
|
||||||
|
|
||||||
@property({ attribute: false }) public auth?: Auth;
|
@property() public auth?: Auth;
|
||||||
|
|
||||||
@property({ attribute: false }) public connection?: Connection;
|
@property() public connection?: Connection;
|
||||||
|
|
||||||
@property({ attribute: false }) public user?: HassUser;
|
@property() public user?: HassUser;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
return html`
|
return html`
|
||||||
|
@@ -1,5 +1,7 @@
|
|||||||
import { LovelaceCardConfig } from "../../../../src/data/lovelace/config/card";
|
import {
|
||||||
import { LovelaceConfig } from "../../../../src/data/lovelace/config/types";
|
LovelaceCardConfig,
|
||||||
|
LovelaceConfig,
|
||||||
|
} from "../../../../src/data/lovelace";
|
||||||
import { castContext } from "../cast_context";
|
import { castContext } from "../cast_context";
|
||||||
|
|
||||||
export const castDemoLovelace: () => LovelaceConfig = () => {
|
export const castDemoLovelace: () => LovelaceConfig = () => {
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
import { html, nothing } from "lit";
|
import { html, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import { mockHistory } from "../../../../demo/src/stubs/history";
|
import { mockHistory } from "../../../../demo/src/stubs/history";
|
||||||
import { LovelaceConfig } from "../../../../src/data/lovelace/config/types";
|
import { LovelaceConfig } from "../../../../src/data/lovelace";
|
||||||
import {
|
import {
|
||||||
MockHomeAssistant,
|
MockHomeAssistant,
|
||||||
provideHass,
|
provideHass,
|
||||||
|
@@ -12,8 +12,8 @@ class HcLaunchScreen extends LitElement {
|
|||||||
return html`
|
return html`
|
||||||
<div class="container">
|
<div class="container">
|
||||||
<img
|
<img
|
||||||
alt="Nabu Casa logo on left, Home Assistant logo on right, and red heart in center"
|
alt="Home Assistant logo on left, Nabu Casa logo on right, and red heart in center"
|
||||||
src="https://cast.home-assistant.io/images/nabu-loves-hass.png"
|
src="https://www.home-assistant.io/images/blog/2018-09-thinking-big/social.png"
|
||||||
/>
|
/>
|
||||||
<div class="status">
|
<div class="status">
|
||||||
${this.hass ? "Connected" : "Not Connected"}
|
${this.hass ? "Connected" : "Not Connected"}
|
||||||
@@ -28,23 +28,23 @@ class HcLaunchScreen extends LitElement {
|
|||||||
:host {
|
:host {
|
||||||
display: block;
|
display: block;
|
||||||
height: 100vh;
|
height: 100vh;
|
||||||
background-color: #f2f4f9;
|
padding-top: 64px;
|
||||||
|
background-color: white;
|
||||||
font-size: 24px;
|
font-size: 24px;
|
||||||
}
|
}
|
||||||
.container {
|
.container {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
align-items: center;
|
|
||||||
height: 100%;
|
|
||||||
justify-content: space-evenly;
|
|
||||||
}
|
}
|
||||||
img {
|
img {
|
||||||
max-width: 80%;
|
width: 717px;
|
||||||
object-fit: cover;
|
height: 376px;
|
||||||
|
display: block;
|
||||||
|
margin: 0 auto;
|
||||||
}
|
}
|
||||||
.status {
|
.status {
|
||||||
color: #1d2126;
|
padding-right: 54px;
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
import { customElement, property, query } from "lit/decorators";
|
import { customElement, property, query } from "lit/decorators";
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
import { LovelaceConfig } from "../../../../src/data/lovelace/config/types";
|
import { LovelaceConfig } from "../../../../src/data/lovelace";
|
||||||
import { Lovelace } from "../../../../src/panels/lovelace/types";
|
import { Lovelace } from "../../../../src/panels/lovelace/types";
|
||||||
import "../../../../src/panels/lovelace/views/hui-view";
|
import "../../../../src/panels/lovelace/views/hui-view";
|
||||||
import { HomeAssistant } from "../../../../src/types";
|
import { HomeAssistant } from "../../../../src/types";
|
||||||
@@ -14,10 +14,9 @@ import "./hc-launch-screen";
|
|||||||
class HcLovelace extends LitElement {
|
class HcLovelace extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property({ attribute: false })
|
@property({ attribute: false }) public lovelaceConfig!: LovelaceConfig;
|
||||||
public lovelaceConfig!: LovelaceConfig;
|
|
||||||
|
|
||||||
@property() public viewPath?: string | number | null;
|
@property() public viewPath?: string | number;
|
||||||
|
|
||||||
@property() public urlPath: string | null = null;
|
@property() public urlPath: string | null = null;
|
||||||
|
|
||||||
@@ -93,9 +92,6 @@ class HcLovelace extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private get _viewIndex() {
|
private get _viewIndex() {
|
||||||
if (this.viewPath === null) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
const selectedView = this.viewPath;
|
const selectedView = this.viewPath;
|
||||||
const selectedViewInt = parseInt(selectedView as string, 10);
|
const selectedViewInt = parseInt(selectedView as string, 10);
|
||||||
for (let i = 0; i < this.lovelaceConfig.views.length; i++) {
|
for (let i = 0; i < this.lovelaceConfig.views.length; i++) {
|
||||||
|
@@ -21,26 +21,18 @@ import {
|
|||||||
import { atLeastVersion } from "../../../../src/common/config/version";
|
import { atLeastVersion } from "../../../../src/common/config/version";
|
||||||
import { isNavigationClick } from "../../../../src/common/dom/is-navigation-click";
|
import { isNavigationClick } from "../../../../src/common/dom/is-navigation-click";
|
||||||
import {
|
import {
|
||||||
|
fetchResources,
|
||||||
getLegacyLovelaceCollection,
|
getLegacyLovelaceCollection,
|
||||||
getLovelaceCollection,
|
getLovelaceCollection,
|
||||||
} from "../../../../src/data/lovelace";
|
|
||||||
import {
|
|
||||||
isStrategyDashboard,
|
|
||||||
LegacyLovelaceConfig,
|
LegacyLovelaceConfig,
|
||||||
LovelaceConfig,
|
LovelaceConfig,
|
||||||
LovelaceDashboardStrategyConfig,
|
} from "../../../../src/data/lovelace";
|
||||||
} from "../../../../src/data/lovelace/config/types";
|
|
||||||
import { fetchResources } from "../../../../src/data/lovelace/resource";
|
|
||||||
import { loadLovelaceResources } from "../../../../src/panels/lovelace/common/load-resources";
|
import { loadLovelaceResources } from "../../../../src/panels/lovelace/common/load-resources";
|
||||||
import { HassElement } from "../../../../src/state/hass-element";
|
import { HassElement } from "../../../../src/state/hass-element";
|
||||||
import { castContext } from "../cast_context";
|
import { castContext } from "../cast_context";
|
||||||
import "./hc-launch-screen";
|
import "./hc-launch-screen";
|
||||||
|
|
||||||
const DEFAULT_CONFIG: LovelaceDashboardStrategyConfig = {
|
const DEFAULT_STRATEGY = "original-states";
|
||||||
strategy: {
|
|
||||||
type: "original-states",
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
let resourcesLoaded = false;
|
let resourcesLoaded = false;
|
||||||
@customElement("hc-main")
|
@customElement("hc-main")
|
||||||
@@ -51,10 +43,10 @@ export class HcMain extends HassElement {
|
|||||||
|
|
||||||
@state() private _lovelacePath: string | number | null = null;
|
@state() private _lovelacePath: string | number | null = null;
|
||||||
|
|
||||||
@state() private _urlPath?: string | null;
|
|
||||||
|
|
||||||
@state() private _error?: string;
|
@state() private _error?: string;
|
||||||
|
|
||||||
|
@state() private _urlPath?: string | null;
|
||||||
|
|
||||||
private _hassUUID?: string;
|
private _hassUUID?: string;
|
||||||
|
|
||||||
private _unsubLovelace?: UnsubscribeFunc;
|
private _unsubLovelace?: UnsubscribeFunc;
|
||||||
@@ -81,7 +73,7 @@ export class HcMain extends HassElement {
|
|||||||
|
|
||||||
if (
|
if (
|
||||||
!this._lovelaceConfig ||
|
!this._lovelaceConfig ||
|
||||||
this._urlPath === undefined ||
|
this._lovelacePath === null ||
|
||||||
// Guard against part of HA not being loaded yet.
|
// Guard against part of HA not being loaded yet.
|
||||||
!this.hass ||
|
!this.hass ||
|
||||||
!this.hass.states ||
|
!this.hass.states ||
|
||||||
@@ -99,18 +91,16 @@ export class HcMain extends HassElement {
|
|||||||
<hc-lovelace
|
<hc-lovelace
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.lovelaceConfig=${this._lovelaceConfig}
|
.lovelaceConfig=${this._lovelaceConfig}
|
||||||
.urlPath=${this._urlPath}
|
|
||||||
.viewPath=${this._lovelacePath}
|
.viewPath=${this._lovelacePath}
|
||||||
@config-refresh=${this._generateDefaultLovelaceConfig}
|
.urlPath=${this._urlPath}
|
||||||
|
@config-refresh=${this._generateLovelaceConfig}
|
||||||
></hc-lovelace>
|
></hc-lovelace>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected firstUpdated(changedProps) {
|
protected firstUpdated(changedProps) {
|
||||||
super.firstUpdated(changedProps);
|
super.firstUpdated(changedProps);
|
||||||
import("./hc-lovelace");
|
import("../second-load");
|
||||||
import("../../../../src/resources/ha-style");
|
|
||||||
|
|
||||||
window.addEventListener("location-changed", () => {
|
window.addEventListener("location-changed", () => {
|
||||||
const panelPath = `/${this._urlPath || "lovelace"}/`;
|
const panelPath = `/${this._urlPath || "lovelace"}/`;
|
||||||
if (location.pathname.startsWith(panelPath)) {
|
if (location.pathname.startsWith(panelPath)) {
|
||||||
@@ -205,6 +195,7 @@ export class HcMain extends HassElement {
|
|||||||
expires_in: 0,
|
expires_in: 0,
|
||||||
}),
|
}),
|
||||||
});
|
});
|
||||||
|
this._hassUUID = msg.hassUUID;
|
||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
const errorMessage = this._getErrorMessage(err);
|
const errorMessage = this._getErrorMessage(err);
|
||||||
this._error = errorMessage;
|
this._error = errorMessage;
|
||||||
@@ -224,17 +215,6 @@ export class HcMain extends HassElement {
|
|||||||
this.hass.connection.close();
|
this.hass.connection.close();
|
||||||
}
|
}
|
||||||
this.initializeHass(auth, connection);
|
this.initializeHass(auth, connection);
|
||||||
if (this._hassUUID !== msg.hassUUID) {
|
|
||||||
this._hassUUID = msg.hassUUID;
|
|
||||||
this._lovelaceConfig = undefined;
|
|
||||||
this._urlPath = undefined;
|
|
||||||
this._lovelacePath = null;
|
|
||||||
if (this._unsubLovelace) {
|
|
||||||
this._unsubLovelace();
|
|
||||||
this._unsubLovelace = undefined;
|
|
||||||
}
|
|
||||||
resourcesLoaded = false;
|
|
||||||
}
|
|
||||||
this._error = undefined;
|
this._error = undefined;
|
||||||
this._sendStatus();
|
this._sendStatus();
|
||||||
}
|
}
|
||||||
@@ -243,7 +223,7 @@ export class HcMain extends HassElement {
|
|||||||
this._showDemo = false;
|
this._showDemo = false;
|
||||||
// We should not get this command before we are connected.
|
// We should not get this command before we are connected.
|
||||||
// Means a client got out of sync. Let's send status to them.
|
// Means a client got out of sync. Let's send status to them.
|
||||||
if (!this.hass?.connected) {
|
if (!this.hass) {
|
||||||
this._sendStatus(msg.senderId!);
|
this._sendStatus(msg.senderId!);
|
||||||
this._error = "Cannot show Lovelace because we're not connected.";
|
this._error = "Cannot show Lovelace because we're not connected.";
|
||||||
this._sendError(
|
this._sendError(
|
||||||
@@ -270,7 +250,7 @@ export class HcMain extends HassElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
this._error = undefined;
|
this._error = undefined;
|
||||||
if (msg.urlPath === "lovelace" || msg.urlPath === undefined) {
|
if (msg.urlPath === "lovelace") {
|
||||||
msg.urlPath = null;
|
msg.urlPath = null;
|
||||||
}
|
}
|
||||||
this._lovelacePath = msg.viewPath;
|
this._lovelacePath = msg.viewPath;
|
||||||
@@ -280,12 +260,13 @@ export class HcMain extends HassElement {
|
|||||||
{
|
{
|
||||||
strategy: {
|
strategy: {
|
||||||
type: "energy",
|
type: "energy",
|
||||||
|
show_date_selection: true,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
};
|
};
|
||||||
this._urlPath = "energy";
|
this._urlPath = "energy";
|
||||||
this._lovelacePath = null;
|
this._lovelacePath = 0;
|
||||||
this._sendStatus();
|
this._sendStatus();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -294,7 +275,6 @@ export class HcMain extends HassElement {
|
|||||||
this._lovelaceConfig = undefined;
|
this._lovelaceConfig = undefined;
|
||||||
if (this._unsubLovelace) {
|
if (this._unsubLovelace) {
|
||||||
this._unsubLovelace();
|
this._unsubLovelace();
|
||||||
this._unsubLovelace = undefined;
|
|
||||||
}
|
}
|
||||||
const llColl = atLeastVersion(this.hass.connection.haVersion, 0, 107)
|
const llColl = atLeastVersion(this.hass.connection.haVersion, 0, 107)
|
||||||
? getLovelaceCollection(this.hass.connection, msg.urlPath)
|
? getLovelaceCollection(this.hass.connection, msg.urlPath)
|
||||||
@@ -303,20 +283,9 @@ export class HcMain extends HassElement {
|
|||||||
// configuration.
|
// configuration.
|
||||||
try {
|
try {
|
||||||
await llColl.refresh();
|
await llColl.refresh();
|
||||||
this._unsubLovelace = llColl.subscribe(async (rawConfig) => {
|
this._unsubLovelace = llColl.subscribe((lovelaceConfig) =>
|
||||||
if (isStrategyDashboard(rawConfig)) {
|
this._handleNewLovelaceConfig(lovelaceConfig)
|
||||||
const { generateLovelaceDashboardStrategy } = await import(
|
);
|
||||||
"../../../../src/panels/lovelace/strategies/get-strategy"
|
|
||||||
);
|
|
||||||
const config = await generateLovelaceDashboardStrategy(
|
|
||||||
rawConfig.strategy,
|
|
||||||
this.hass!
|
|
||||||
);
|
|
||||||
this._handleNewLovelaceConfig(config);
|
|
||||||
} else {
|
|
||||||
this._handleNewLovelaceConfig(rawConfig);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
} catch (err: any) {
|
} catch (err: any) {
|
||||||
if (
|
if (
|
||||||
atLeastVersion(this.hass.connection.haVersion, 0, 107) &&
|
atLeastVersion(this.hass.connection.haVersion, 0, 107) &&
|
||||||
@@ -330,7 +299,7 @@ export class HcMain extends HassElement {
|
|||||||
}
|
}
|
||||||
// Generate a Lovelace config.
|
// Generate a Lovelace config.
|
||||||
this._unsubLovelace = () => undefined;
|
this._unsubLovelace = () => undefined;
|
||||||
await this._generateDefaultLovelaceConfig();
|
await this._generateLovelaceConfig();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (!resourcesLoaded) {
|
if (!resourcesLoaded) {
|
||||||
@@ -339,20 +308,22 @@ export class HcMain extends HassElement {
|
|||||||
? await fetchResources(this.hass!.connection)
|
? await fetchResources(this.hass!.connection)
|
||||||
: (this._lovelaceConfig as LegacyLovelaceConfig).resources;
|
: (this._lovelaceConfig as LegacyLovelaceConfig).resources;
|
||||||
if (resources) {
|
if (resources) {
|
||||||
loadLovelaceResources(resources, this.hass!);
|
loadLovelaceResources(resources, this.hass!.auth.data.hassUrl);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
this._sendStatus();
|
this._sendStatus();
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _generateDefaultLovelaceConfig() {
|
private async _generateLovelaceConfig() {
|
||||||
const { generateLovelaceDashboardStrategy } = await import(
|
const { generateLovelaceDashboardStrategy } = await import(
|
||||||
"../../../../src/panels/lovelace/strategies/get-strategy"
|
"../../../../src/panels/lovelace/strategies/get-strategy"
|
||||||
);
|
);
|
||||||
this._handleNewLovelaceConfig(
|
this._handleNewLovelaceConfig(
|
||||||
await generateLovelaceDashboardStrategy(
|
await generateLovelaceDashboardStrategy(
|
||||||
DEFAULT_CONFIG.strategy,
|
{
|
||||||
|
type: DEFAULT_STRATEGY,
|
||||||
|
},
|
||||||
this.hass!
|
this.hass!
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
3
cast/src/receiver/second-load.ts
Normal file
3
cast/src/receiver/second-load.ts
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
import "../../../src/resources/ha-style";
|
||||||
|
import "../../../src/resources/roboto";
|
||||||
|
import "./layout/hc-lovelace";
|
@@ -3,15 +3,6 @@ import { DemoConfig } from "../types";
|
|||||||
|
|
||||||
export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) =>
|
export const demoEntitiesArsaboo: DemoConfig["entities"] = (localize) =>
|
||||||
convertEntities({
|
convertEntities({
|
||||||
"todo.shopping_list": {
|
|
||||||
entity_id: "todo.shopping_list",
|
|
||||||
state: "2",
|
|
||||||
attributes: {
|
|
||||||
supported_features: 15,
|
|
||||||
friendly_name: "Shopping List",
|
|
||||||
icon: "mdi:cart",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"zone.home": {
|
"zone.home": {
|
||||||
entity_id: "zone.home",
|
entity_id: "zone.home",
|
||||||
state: "zoning",
|
state: "zoning",
|
||||||
|
@@ -4,7 +4,6 @@ import { energyEntities } from "../stubs/entities";
|
|||||||
import { DemoConfig } from "./types";
|
import { DemoConfig } from "./types";
|
||||||
|
|
||||||
export const demoConfigs: Array<() => Promise<DemoConfig>> = [
|
export const demoConfigs: Array<() => Promise<DemoConfig>> = [
|
||||||
() => import("./sections").then((mod) => mod.demoSections),
|
|
||||||
() => import("./arsaboo").then((mod) => mod.demoArsaboo),
|
() => import("./arsaboo").then((mod) => mod.demoArsaboo),
|
||||||
() => import("./teachingbirds").then((mod) => mod.demoTeachingbirds),
|
() => import("./teachingbirds").then((mod) => mod.demoTeachingbirds),
|
||||||
() => import("./kernehed").then((mod) => mod.demoKernehed),
|
() => import("./kernehed").then((mod) => mod.demoKernehed),
|
||||||
|
@@ -3,15 +3,6 @@ import { DemoConfig } from "../types";
|
|||||||
|
|
||||||
export const demoEntitiesJimpower: DemoConfig["entities"] = () =>
|
export const demoEntitiesJimpower: DemoConfig["entities"] = () =>
|
||||||
convertEntities({
|
convertEntities({
|
||||||
"todo.shopping_list": {
|
|
||||||
entity_id: "todo.shopping_list",
|
|
||||||
state: "2",
|
|
||||||
attributes: {
|
|
||||||
supported_features: 15,
|
|
||||||
friendly_name: "Shopping List",
|
|
||||||
icon: "mdi:cart",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"zone.powertec": {
|
"zone.powertec": {
|
||||||
entity_id: "zone.powertec",
|
entity_id: "zone.powertec",
|
||||||
state: "zoning",
|
state: "zoning",
|
||||||
|
@@ -4,11 +4,16 @@ export const demoThemeJimpower = () => ({
|
|||||||
"primary-color": "#5294E2",
|
"primary-color": "#5294E2",
|
||||||
"label-badge-red": "var(--accent-color)",
|
"label-badge-red": "var(--accent-color)",
|
||||||
"paper-tabs-selection-bar-color": "green",
|
"paper-tabs-selection-bar-color": "green",
|
||||||
|
"paper-slider-knob-color": "var(--accent-color)",
|
||||||
"light-primary-color": "var(--accent-color)",
|
"light-primary-color": "var(--accent-color)",
|
||||||
"primary-background-color": "#383C45",
|
"primary-background-color": "#383C45",
|
||||||
"primary-text-color": "#FFFFFF",
|
"primary-text-color": "#FFFFFF",
|
||||||
"paper-item-selected_-_background-color": "#434954",
|
"paper-item-selected_-_background-color": "#434954",
|
||||||
|
"paper-slider-active-color": "var(--accent-color)",
|
||||||
"secondary-background-color": "#383C45",
|
"secondary-background-color": "#383C45",
|
||||||
|
"paper-slider-container-color":
|
||||||
|
"linear-gradient(var(--primary-background-color), var(--secondary-background-color)) no-repeat",
|
||||||
|
"paper-slider-disabled-active-color": "var(--disabled-text-color)",
|
||||||
"disabled-text-color": "#7F848E",
|
"disabled-text-color": "#7F848E",
|
||||||
"paper-item-icon_-_color": "green",
|
"paper-item-icon_-_color": "green",
|
||||||
"paper-grey-200": "#414A59",
|
"paper-grey-200": "#414A59",
|
||||||
@@ -27,10 +32,14 @@ export const demoThemeJimpower = () => ({
|
|||||||
"switch-unchecked-button-color": "var(--disabled-text-color)",
|
"switch-unchecked-button-color": "var(--disabled-text-color)",
|
||||||
"label-badge-border-color": "green",
|
"label-badge-border-color": "green",
|
||||||
"paper-listbox-color": "var(--primary-color)",
|
"paper-listbox-color": "var(--primary-color)",
|
||||||
|
"paper-slider-disabled-secondary-color": "var(--disabled-text-color)",
|
||||||
"card-background-color": "#434954",
|
"card-background-color": "#434954",
|
||||||
"label-badge-text-color": "var(--primary-text-color)",
|
"label-badge-text-color": "var(--primary-text-color)",
|
||||||
|
"paper-slider-knob-start-color": "var(--accent-color)",
|
||||||
"switch-unchecked-track-color": "var(--disabled-text-color)",
|
"switch-unchecked-track-color": "var(--disabled-text-color)",
|
||||||
"dark-primary-color": "var(--accent-color)",
|
"dark-primary-color": "var(--accent-color)",
|
||||||
|
"paper-slider-secondary-color": "var(--secondary-background-color)",
|
||||||
|
"paper-slider-pin-color": "var(--accent-color)",
|
||||||
"paper-item-icon-active-color": "#F9C536",
|
"paper-item-icon-active-color": "#F9C536",
|
||||||
"accent-color": "#E45E65",
|
"accent-color": "#E45E65",
|
||||||
"table-row-alternative-background-color": "#3E424B",
|
"table-row-alternative-background-color": "#3E424B",
|
||||||
|
@@ -3,15 +3,6 @@ import { DemoConfig } from "../types";
|
|||||||
|
|
||||||
export const demoEntitiesKernehed: DemoConfig["entities"] = () =>
|
export const demoEntitiesKernehed: DemoConfig["entities"] = () =>
|
||||||
convertEntities({
|
convertEntities({
|
||||||
"todo.shopping_list": {
|
|
||||||
entity_id: "todo.shopping_list",
|
|
||||||
state: "2",
|
|
||||||
attributes: {
|
|
||||||
supported_features: 15,
|
|
||||||
friendly_name: "Shopping List",
|
|
||||||
icon: "mdi:cart",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"zone.anna": {
|
"zone.anna": {
|
||||||
entity_id: "zone.anna",
|
entity_id: "zone.anna",
|
||||||
state: "zoning",
|
state: "zoning",
|
||||||
|
@@ -5,12 +5,17 @@ export const demoThemeKernehed = () => ({
|
|||||||
"primary-color": "#2980b9",
|
"primary-color": "#2980b9",
|
||||||
"label-badge-red": "var(--accent-color)",
|
"label-badge-red": "var(--accent-color)",
|
||||||
"paper-tabs-selection-bar-color": "green",
|
"paper-tabs-selection-bar-color": "green",
|
||||||
|
"paper-slider-knob-color": "var(--accent-color)",
|
||||||
"primary-text-color": "#FFFFFF",
|
"primary-text-color": "#FFFFFF",
|
||||||
"light-primary-color": "var(--accent-color)",
|
"light-primary-color": "var(--accent-color)",
|
||||||
"primary-background-color": "#222222",
|
"primary-background-color": "#222222",
|
||||||
"sidebar-icon-color": "#777777",
|
"sidebar-icon-color": "#777777",
|
||||||
"paper-item-selected_-_background-color": "#292929",
|
"paper-item-selected_-_background-color": "#292929",
|
||||||
|
"paper-slider-active-color": "var(--accent-color)",
|
||||||
"secondary-background-color": "#222222",
|
"secondary-background-color": "#222222",
|
||||||
|
"paper-slider-container-color":
|
||||||
|
"linear-gradient(var(--primary-background-color), var(--secondary-background-color)) no-repeat",
|
||||||
|
"paper-slider-disabled-active-color": "var(--disabled-text-color)",
|
||||||
"disabled-text-color": "#777777",
|
"disabled-text-color": "#777777",
|
||||||
"paper-item-icon_-_color": "green",
|
"paper-item-icon_-_color": "green",
|
||||||
"paper-grey-200": "#222222",
|
"paper-grey-200": "#222222",
|
||||||
@@ -28,10 +33,14 @@ export const demoThemeKernehed = () => ({
|
|||||||
"switch-unchecked-button-color": "var(--disabled-text-color)",
|
"switch-unchecked-button-color": "var(--disabled-text-color)",
|
||||||
"label-badge-border-color": "green",
|
"label-badge-border-color": "green",
|
||||||
"paper-listbox-color": "#777777",
|
"paper-listbox-color": "#777777",
|
||||||
|
"paper-slider-disabled-secondary-color": "var(--disabled-text-color)",
|
||||||
"card-background-color": "#292929",
|
"card-background-color": "#292929",
|
||||||
"label-badge-text-color": "var(--primary-text-color)",
|
"label-badge-text-color": "var(--primary-text-color)",
|
||||||
|
"paper-slider-knob-start-color": "var(--accent-color)",
|
||||||
"switch-unchecked-track-color": "var(--disabled-text-color)",
|
"switch-unchecked-track-color": "var(--disabled-text-color)",
|
||||||
"dark-primary-color": "var(--accent-color)",
|
"dark-primary-color": "var(--accent-color)",
|
||||||
|
"paper-slider-secondary-color": "var(--secondary-background-color)",
|
||||||
|
"paper-slider-pin-color": "var(--accent-color)",
|
||||||
"paper-item-icon-active-color": "#b58e31",
|
"paper-item-icon-active-color": "#b58e31",
|
||||||
"accent-color": "#2980b9",
|
"accent-color": "#2980b9",
|
||||||
"table-row-alternative-background-color": "#292929",
|
"table-row-alternative-background-color": "#292929",
|
||||||
|
@@ -1,16 +0,0 @@
|
|||||||
import { html } from "lit";
|
|
||||||
import { DemoConfig } from "../types";
|
|
||||||
|
|
||||||
export const demoLovelaceDescription: DemoConfig["description"] = (
|
|
||||||
localize
|
|
||||||
) => html`
|
|
||||||
<p>
|
|
||||||
${localize("ui.panel.page-demo.config.sections.description", {
|
|
||||||
blog_post: html`<a
|
|
||||||
href="https://www.home-assistant.io/blog/2024/03/04/dashboard-chapter-1/"
|
|
||||||
target="_blank"
|
|
||||||
>${localize("ui.panel.page-demo.config.sections.description_blog_post")}
|
|
||||||
</a>`,
|
|
||||||
})}
|
|
||||||
</p>
|
|
||||||
`;
|
|
@@ -1,474 +0,0 @@
|
|||||||
import { convertEntities } from "../../../../src/fake_data/entity";
|
|
||||||
import { DemoConfig } from "../types";
|
|
||||||
|
|
||||||
export const demoEntitiesSections: DemoConfig["entities"] = () =>
|
|
||||||
convertEntities({
|
|
||||||
"cover.living_room_garden_shutter": {
|
|
||||||
entity_id: "cover.living_room_garden_shutter",
|
|
||||||
state: "open",
|
|
||||||
attributes: {
|
|
||||||
current_position: 100,
|
|
||||||
device_class: "shutter",
|
|
||||||
friendly_name: "Living room garden shutter",
|
|
||||||
supported_features: 15,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"cover.living_room_graveyard_shutter": {
|
|
||||||
entity_id: "cover.living_room_graveyard_shutter",
|
|
||||||
state: "open",
|
|
||||||
attributes: {
|
|
||||||
current_position: 100,
|
|
||||||
device_class: "shutter",
|
|
||||||
friendly_name: "Living room graveyard shutter",
|
|
||||||
supported_features: 15,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"cover.living_room_left_shutter": {
|
|
||||||
entity_id: "cover.living_room_left_shutter",
|
|
||||||
state: "open",
|
|
||||||
attributes: {
|
|
||||||
current_position: 100,
|
|
||||||
device_class: "shutter",
|
|
||||||
friendly_name: "Living room left shutter",
|
|
||||||
supported_features: 15,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"cover.living_room_right_shutter": {
|
|
||||||
entity_id: "cover.living_room_right_shutter",
|
|
||||||
state: "open",
|
|
||||||
attributes: {
|
|
||||||
current_position: 100,
|
|
||||||
device_class: "shutter",
|
|
||||||
friendly_name: "Living room right shutter",
|
|
||||||
supported_features: 15,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"light.floor_lamp": {
|
|
||||||
entity_id: "light.floor_lamp",
|
|
||||||
state: "on",
|
|
||||||
attributes: {
|
|
||||||
min_color_temp_kelvin: 2000,
|
|
||||||
max_color_temp_kelvin: 6535,
|
|
||||||
min_mireds: 153,
|
|
||||||
max_mireds: 500,
|
|
||||||
supported_color_modes: ["color_temp", "xy"],
|
|
||||||
color_mode: "color_temp",
|
|
||||||
brightness: 178,
|
|
||||||
color_temp_kelvin: 2583,
|
|
||||||
color_temp: 387,
|
|
||||||
hs_color: [28.664, 69.597],
|
|
||||||
rgb_color: [255, 162, 77],
|
|
||||||
xy_color: [0.538, 0.389],
|
|
||||||
icon: "mdi:floor-lamp",
|
|
||||||
friendly_name: "Floor lamp",
|
|
||||||
supported_features: 44,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"light.living_room_spotlights": {
|
|
||||||
entity_id: "light.living_room_spotlights",
|
|
||||||
state: "on",
|
|
||||||
attributes: {
|
|
||||||
supported_color_modes: ["brightness"],
|
|
||||||
color_mode: "brightness",
|
|
||||||
brightness: 126,
|
|
||||||
icon: "mdi:ceiling-light-multiple",
|
|
||||||
friendly_name: "Living room spotlights",
|
|
||||||
supported_features: 32,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"light.bar_lamp": {
|
|
||||||
entity_id: "light.bar_lamp",
|
|
||||||
state: "on",
|
|
||||||
attributes: {
|
|
||||||
min_color_temp_kelvin: 2202,
|
|
||||||
max_color_temp_kelvin: 4504,
|
|
||||||
min_mireds: 222,
|
|
||||||
max_mireds: 454,
|
|
||||||
effect_list: ["None", "candle"],
|
|
||||||
supported_color_modes: ["color_temp"],
|
|
||||||
effect: null,
|
|
||||||
color_mode: null,
|
|
||||||
brightness: null,
|
|
||||||
color_temp_kelvin: null,
|
|
||||||
color_temp: null,
|
|
||||||
hs_color: null,
|
|
||||||
rgb_color: null,
|
|
||||||
xy_color: null,
|
|
||||||
mode: "normal",
|
|
||||||
dynamics: "none",
|
|
||||||
icon: "mdi:lightbulb-variant",
|
|
||||||
friendly_name: "Bar lamp",
|
|
||||||
supported_features: 44,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sensor.living_room_temperature": {
|
|
||||||
entity_id: "sensor.living_room_temperature",
|
|
||||||
state: "22.8",
|
|
||||||
attributes: {
|
|
||||||
state_class: "measurement",
|
|
||||||
unit_of_measurement: "°C",
|
|
||||||
device_class: "temperature",
|
|
||||||
friendly_name: "Living room Temperature",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"media_player.living_room_nest_mini": {
|
|
||||||
entity_id: "media_player.living_room_nest_mini",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
device_class: "speaker",
|
|
||||||
friendly_name: "Living room Nest Mini",
|
|
||||||
supported_features: 152461,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"cover.kitchen_shutter": {
|
|
||||||
entity_id: "cover.kitchen_shutter",
|
|
||||||
state: "open",
|
|
||||||
attributes: {
|
|
||||||
current_position: 100,
|
|
||||||
device_class: "shutter",
|
|
||||||
friendly_name: "Kitchen shutter ",
|
|
||||||
supported_features: 15,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"light.kitchen_spotlights": {
|
|
||||||
entity_id: "light.kitchen_spotlights",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
supported_color_modes: ["brightness"],
|
|
||||||
color_mode: null,
|
|
||||||
brightness: null,
|
|
||||||
icon: "mdi:ceiling-light-multiple",
|
|
||||||
friendly_name: "Kitchen spotlights ",
|
|
||||||
supported_features: 32,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"light.worktop_spotlights": {
|
|
||||||
entity_id: "light.worktop_spotlights",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
supported_color_modes: ["brightness"],
|
|
||||||
color_mode: null,
|
|
||||||
brightness: null,
|
|
||||||
icon: "mdi:ceiling-light-multiple",
|
|
||||||
friendly_name: "Worktop spotlights ",
|
|
||||||
supported_features: 32,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"binary_sensor.fridge_door": {
|
|
||||||
entity_id: "binary_sensor.fridge_door",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
device_class: "door",
|
|
||||||
icon: "mdi:fridge",
|
|
||||||
friendly_name: "Fridge door",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"media_player.kitchen_nest_audio": {
|
|
||||||
entity_id: "media_player.kitchen_nest_audio",
|
|
||||||
state: "on",
|
|
||||||
attributes: {
|
|
||||||
device_class: "speaker",
|
|
||||||
friendly_name: "Kitchen Nest Audio",
|
|
||||||
supported_features: 152461,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"binary_sensor.tesla_wall_connector_vehicle_connected": {
|
|
||||||
entity_id: "binary_sensor.tesla_wall_connector_vehicle_connected",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
device_class: "plug",
|
|
||||||
friendly_name: "Wall Connector Vehicle connected",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sensor.tesla_wall_connector_session_energy": {
|
|
||||||
entity_id: "sensor.tesla_wall_connector_session_energy",
|
|
||||||
state: "16.3",
|
|
||||||
attributes: {
|
|
||||||
state_class: "total_increasing",
|
|
||||||
unit_of_measurement: "kWh",
|
|
||||||
device_class: "energy",
|
|
||||||
friendly_name: "Tesla Wall Connector Session energy",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sensor.electric_meter_power": {
|
|
||||||
entity_id: "sensor.electric_meter_power",
|
|
||||||
state: "797.86",
|
|
||||||
attributes: {
|
|
||||||
state_class: "measurement",
|
|
||||||
unit_of_measurement: "W",
|
|
||||||
device_class: "power",
|
|
||||||
icon: "mdi:meter-electric",
|
|
||||||
friendly_name: "Electric meter Power",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sensor.eletric_meter_voltage": {
|
|
||||||
entity_id: "sensor.eletric_meter_voltage",
|
|
||||||
state: "232.19",
|
|
||||||
attributes: {
|
|
||||||
state_class: "measurement",
|
|
||||||
unit_of_measurement: "V",
|
|
||||||
device_class: "voltage",
|
|
||||||
friendly_name: "Electric meter voltage",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sensor.electricity_maps_grid_fossil_fuel_percentage": {
|
|
||||||
entity_id: "sensor.electricity_maps_grid_fossil_fuel_percentage",
|
|
||||||
state: "9.84",
|
|
||||||
attributes: {
|
|
||||||
state_class: "measurement",
|
|
||||||
country_code: "FR",
|
|
||||||
unit_of_measurement: "%",
|
|
||||||
attribution: "Data provided by Electricity Maps",
|
|
||||||
icon: "mdi:barrel",
|
|
||||||
friendly_name: "Electricity Maps Grid fossil fuel percentage",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sensor.electricity_maps_co2_intensity": {
|
|
||||||
entity_id: "sensor.electricity_maps_co2_intensity",
|
|
||||||
state: "62.0",
|
|
||||||
attributes: {
|
|
||||||
state_class: "measurement",
|
|
||||||
country_code: "FR",
|
|
||||||
unit_of_measurement: "gCO2eq/kWh",
|
|
||||||
attribution: "Data provided by Electricity Maps",
|
|
||||||
friendly_name: "Electricity Maps CO2 intensity",
|
|
||||||
icon: "mdi:molecule-co2",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sun.sun": {
|
|
||||||
entity_id: "sun.sun",
|
|
||||||
state: "above_horizon",
|
|
||||||
attributes: {
|
|
||||||
next_dawn: "2024-03-05T05:50:21.964405+00:00",
|
|
||||||
next_dusk: "2024-03-04T18:08:54.311334+00:00",
|
|
||||||
next_midnight: "2024-03-05T00:00:00+00:00",
|
|
||||||
next_noon: "2024-03-05T12:00:05+00:00",
|
|
||||||
next_rising: "2024-03-05T06:23:42.739159+00:00",
|
|
||||||
next_setting: "2024-03-04T17:35:26.271171+00:00",
|
|
||||||
elevation: 30.38,
|
|
||||||
azimuth: 204.42,
|
|
||||||
rising: false,
|
|
||||||
friendly_name: "Sun",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sensor.rain": {
|
|
||||||
entity_id: "sensor.moon_phase",
|
|
||||||
state: "7.2",
|
|
||||||
attributes: {
|
|
||||||
state_class: "total_increasing",
|
|
||||||
unit_of_measurement: "mm",
|
|
||||||
device_class: "precipitation",
|
|
||||||
friendly_name: "Rain",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"climate.ground_floor": {
|
|
||||||
entity_id: "climate.ground_floor",
|
|
||||||
state: "heat",
|
|
||||||
attributes: {
|
|
||||||
hvac_modes: ["auto", "heat", "off"],
|
|
||||||
min_temp: 7,
|
|
||||||
max_temp: 35,
|
|
||||||
preset_modes: [
|
|
||||||
"comfort",
|
|
||||||
"away",
|
|
||||||
"eco",
|
|
||||||
"frost_protection",
|
|
||||||
"external",
|
|
||||||
"home",
|
|
||||||
],
|
|
||||||
current_temperature: 20.8,
|
|
||||||
temperature: 21,
|
|
||||||
preset_mode: "comfort",
|
|
||||||
icon: "mdi:home-floor-0",
|
|
||||||
friendly_name: "Ground floor Thermostat",
|
|
||||||
supported_features: 401,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"climate.first_floor": {
|
|
||||||
entity_id: "climate.first_floor",
|
|
||||||
state: "heat",
|
|
||||||
attributes: {
|
|
||||||
hvac_modes: ["auto", "heat", "off"],
|
|
||||||
min_temp: 7,
|
|
||||||
max_temp: 35,
|
|
||||||
preset_modes: [
|
|
||||||
"comfort",
|
|
||||||
"away",
|
|
||||||
"eco",
|
|
||||||
"frost_protection",
|
|
||||||
"external",
|
|
||||||
"home",
|
|
||||||
],
|
|
||||||
current_temperature: 21.7,
|
|
||||||
temperature: 21,
|
|
||||||
preset_mode: "comfort",
|
|
||||||
icon: "mdi:home-floor-1",
|
|
||||||
friendly_name: "First floor Thermostat",
|
|
||||||
supported_features: 401,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"cover.study_shutter": {
|
|
||||||
entity_id: "cover.study_shutter",
|
|
||||||
state: "open",
|
|
||||||
attributes: {
|
|
||||||
current_position: 100,
|
|
||||||
device_class: "shutter",
|
|
||||||
friendly_name: "Study shutter",
|
|
||||||
supported_features: 15,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"light.study_spotlights": {
|
|
||||||
entity_id: "light.study_spotlights",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
supported_color_modes: ["brightness"],
|
|
||||||
color_mode: null,
|
|
||||||
brightness: null,
|
|
||||||
icon: "mdi:ceiling-light-multiple",
|
|
||||||
friendly_name: "Study spotlights",
|
|
||||||
supported_features: 32,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"media_player.study_nest_hub": {
|
|
||||||
entity_id: "media_player.study_nest_hub",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
friendly_name: "Study Nest Hub",
|
|
||||||
supported_features: 152461,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sensor.standing_desk_height": {
|
|
||||||
entity_id: "sensor.standing_desk_height",
|
|
||||||
state: "72",
|
|
||||||
attributes: {
|
|
||||||
unit_of_measurement: "cm",
|
|
||||||
icon: "mdi:tape-measure",
|
|
||||||
friendly_name: "Standing desk Height",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"light.outdoor_light": {
|
|
||||||
entity_id: "light.outdoor_light",
|
|
||||||
state: "on",
|
|
||||||
attributes: {
|
|
||||||
supported_color_modes: ["brightness"],
|
|
||||||
color_mode: null,
|
|
||||||
brightness: 255,
|
|
||||||
icon: "mdi:outdoor-lamp",
|
|
||||||
friendly_name: "Outdoor light",
|
|
||||||
supported_features: 32,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"light.flood_light": {
|
|
||||||
entity_id: "light.flood_light",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
effect_list: ["None", "candle"],
|
|
||||||
supported_color_modes: ["brightness"],
|
|
||||||
effect: null,
|
|
||||||
color_mode: null,
|
|
||||||
brightness: null,
|
|
||||||
mode: "normal",
|
|
||||||
dynamics: "none",
|
|
||||||
icon: "mdi:light-flood-down",
|
|
||||||
friendly_name: "Flood light",
|
|
||||||
supported_features: 44,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sensor.outdoor_motion_sensor_temperature": {
|
|
||||||
entity_id: "sensor.outdoor_motion_sensor_temperature",
|
|
||||||
state: "10.2",
|
|
||||||
attributes: {
|
|
||||||
state_class: "measurement",
|
|
||||||
unit_of_measurement: "°C",
|
|
||||||
device_class: "temperature",
|
|
||||||
friendly_name: "Outdoor motion sensor Temperature",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"binary_sensor.outdoor_motion_sensor_motion": {
|
|
||||||
entity_id: "binary_sensor.outdoor_motion_sensor_motion",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
device_class: "motion",
|
|
||||||
friendly_name: "Outdoor motion sensor Motion",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sensor.outdoor_motion_sensor_illuminance": {
|
|
||||||
entity_id: "sensor.outdoor_motion_sensor_illuminance",
|
|
||||||
state: "555",
|
|
||||||
attributes: {
|
|
||||||
state_class: "measurement",
|
|
||||||
light_level: 27444,
|
|
||||||
unit_of_measurement: "lx",
|
|
||||||
device_class: "illuminance",
|
|
||||||
friendly_name: "Outdoor motion sensor Illuminance",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"automation.home_assistant_auto_update": {
|
|
||||||
entity_id: "automation.home_assistant_auto_update",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
id: "1700669321947",
|
|
||||||
last_triggered: "2024-02-29T18:02:05.343139+00:00",
|
|
||||||
mode: "queued",
|
|
||||||
current: 0,
|
|
||||||
max: 50,
|
|
||||||
icon: "mdi:auto-mode",
|
|
||||||
friendly_name: "Home Assistant Auto-update",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"update.home_assistant_operating_system_update": {
|
|
||||||
entity_id: "update.home_assistant_operating_system_update",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
auto_update: false,
|
|
||||||
installed_version: "12.1",
|
|
||||||
in_progress: false,
|
|
||||||
latest_version: "12.1",
|
|
||||||
release_summary: null,
|
|
||||||
release_url:
|
|
||||||
"https://github.com/home-assistant/operating-system/commits/dev",
|
|
||||||
skipped_version: null,
|
|
||||||
title: "Home Assistant Operating System",
|
|
||||||
entity_picture:
|
|
||||||
"https://brands.home-assistant.io/homeassistant/icon.png",
|
|
||||||
friendly_name: "Home Assistant Operating System Update",
|
|
||||||
supported_features: 3,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"update.home_assistant_supervisor_update": {
|
|
||||||
entity_id: "update.home_assistant_supervisor_update",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
auto_update: true,
|
|
||||||
installed_version: "2024.02.2",
|
|
||||||
in_progress: false,
|
|
||||||
latest_version: "2024.02.2",
|
|
||||||
release_summary: null,
|
|
||||||
release_url:
|
|
||||||
"https://github.com/home-assistant/supervisor/commits/main",
|
|
||||||
skipped_version: null,
|
|
||||||
title: "Home Assistant Supervisor",
|
|
||||||
entity_picture: "https://brands.home-assistant.io/hassio/icon.png",
|
|
||||||
friendly_name: "Home Assistant Supervisor Update",
|
|
||||||
supported_features: 1,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"update.home_assistant_core_update": {
|
|
||||||
entity_id: "update.home_assistant_supervisor_update",
|
|
||||||
state: "off",
|
|
||||||
attributes: {
|
|
||||||
auto_update: false,
|
|
||||||
installed_version: "2024.4.0",
|
|
||||||
in_progress: false,
|
|
||||||
latest_version: "2024.4.0",
|
|
||||||
release_summary: null,
|
|
||||||
release_url: "https://github.com/home-assistant/core/commits/dev",
|
|
||||||
skipped_version: null,
|
|
||||||
title: "Home Assistant Core",
|
|
||||||
entity_picture:
|
|
||||||
"https://brands.home-assistant.io/homeassistant/icon.png",
|
|
||||||
friendly_name: "Home Assistant Core Update",
|
|
||||||
supported_features: 11,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
@@ -1,14 +0,0 @@
|
|||||||
import { DemoConfig } from "../types";
|
|
||||||
import { demoLovelaceDescription } from "./description";
|
|
||||||
import { demoEntitiesSections } from "./entities";
|
|
||||||
import { demoLovelaceSections } from "./lovelace";
|
|
||||||
|
|
||||||
export const demoSections: DemoConfig = {
|
|
||||||
authorName: "Home Assistant",
|
|
||||||
authorUrl: "https://github.com/home-assistant/frontend/",
|
|
||||||
name: "Home Demo",
|
|
||||||
description: demoLovelaceDescription,
|
|
||||||
lovelace: demoLovelaceSections,
|
|
||||||
entities: demoEntitiesSections,
|
|
||||||
theme: () => ({}),
|
|
||||||
};
|
|
@@ -1,281 +0,0 @@
|
|||||||
import { DemoConfig } from "../types";
|
|
||||||
|
|
||||||
export const demoLovelaceSections: DemoConfig["lovelace"] = () => ({
|
|
||||||
title: "Home Assistant Demo",
|
|
||||||
views: [
|
|
||||||
{
|
|
||||||
type: "sections",
|
|
||||||
title: "Demo",
|
|
||||||
path: "home",
|
|
||||||
icon: "mdi:home-assistant",
|
|
||||||
sections: [
|
|
||||||
{
|
|
||||||
title: "Welcome 👋",
|
|
||||||
cards: [{ type: "custom:ha-demo-card" }],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
cards: [
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "cover.living_room_garden_shutter",
|
|
||||||
name: "Garden",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "cover.living_room_graveyard_shutter",
|
|
||||||
name: "Rear",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "cover.living_room_left_shutter",
|
|
||||||
name: "Left",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "cover.living_room_right_shutter",
|
|
||||||
name: "Right",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "light.floor_lamp",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "light.living_room_spotlights",
|
|
||||||
name: "Spotlights",
|
|
||||||
features: [
|
|
||||||
{
|
|
||||||
type: "light-brightness",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "light.bar_lamp",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
graph: "line",
|
|
||||||
type: "sensor",
|
|
||||||
entity: "sensor.living_room_temperature",
|
|
||||||
detail: 1,
|
|
||||||
name: "Temperature",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "media_player.living_room_nest_mini",
|
|
||||||
name: "Nest Mini",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
title: "🛋️ Living room ",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "grid",
|
|
||||||
cards: [
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "cover.kitchen_shutter",
|
|
||||||
name: "Shutter",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "light.kitchen_spotlights",
|
|
||||||
name: "Spotlights",
|
|
||||||
features: [
|
|
||||||
{
|
|
||||||
type: "light-brightness",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "light.worktop_spotlights",
|
|
||||||
name: "Worktop",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "binary_sensor.fridge_door",
|
|
||||||
name: "Fridge",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "media_player.kitchen_nest_audio",
|
|
||||||
name: "Nest Audio",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
title: "👩🍳 Kitchen",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "grid",
|
|
||||||
cards: [
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "binary_sensor.tesla_wall_connector_vehicle_connected",
|
|
||||||
name: "EV",
|
|
||||||
icon: "mdi:car",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "sensor.tesla_wall_connector_session_energy",
|
|
||||||
name: "Last charge",
|
|
||||||
color: "green",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "sensor.electric_meter_power",
|
|
||||||
color: "deep-orange",
|
|
||||||
name: "Home power",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "sensor.eletric_meter_voltage",
|
|
||||||
name: "Voltage",
|
|
||||||
color: "deep-orange",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "sensor.electricity_maps_grid_fossil_fuel_percentage",
|
|
||||||
name: "Fossil fuel",
|
|
||||||
color: "brown",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "sensor.electricity_maps_co2_intensity",
|
|
||||||
name: "CO2 Intensity",
|
|
||||||
color: "dark-grey",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
title: "⚡️ Energy",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "grid",
|
|
||||||
cards: [
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "sun.sun",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "sensor.rain",
|
|
||||||
color: "blue",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
features: [
|
|
||||||
{
|
|
||||||
type: "target-temperature",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
type: "tile",
|
|
||||||
name: "Downstairs",
|
|
||||||
entity: "climate.ground_floor",
|
|
||||||
state_content: ["preset_mode", "current_temperature"],
|
|
||||||
},
|
|
||||||
{
|
|
||||||
features: [
|
|
||||||
{
|
|
||||||
type: "target-temperature",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
type: "tile",
|
|
||||||
name: "Upstairs",
|
|
||||||
entity: "climate.first_floor",
|
|
||||||
state_content: ["preset_mode", "current_temperature"],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
title: "🌤️ Climate",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "grid",
|
|
||||||
cards: [
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "cover.study_shutter",
|
|
||||||
name: "Shutter",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "light.study_spotlights",
|
|
||||||
name: "Spotlights",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "media_player.study_nest_hub",
|
|
||||||
name: "Nest Hub",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "sensor.standing_desk_height",
|
|
||||||
name: "Desk",
|
|
||||||
color: "brown",
|
|
||||||
icon: "mdi:desk",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
title: "🧑💻 Study",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "grid",
|
|
||||||
cards: [
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "light.outdoor_light",
|
|
||||||
name: "Door light",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "light.flood_light",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
graph: "line",
|
|
||||||
type: "sensor",
|
|
||||||
entity: "sensor.outdoor_motion_sensor_temperature",
|
|
||||||
detail: 1,
|
|
||||||
name: "Temperature",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "binary_sensor.outdoor_motion_sensor_motion",
|
|
||||||
name: "Motion",
|
|
||||||
color: "blue",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "sensor.outdoor_motion_sensor_illuminance",
|
|
||||||
color: "amber",
|
|
||||||
name: "Illuminance",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
title: "🌳 Outdoor",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "grid",
|
|
||||||
cards: [
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "automation.home_assistant_auto_update",
|
|
||||||
name: "Auto-update",
|
|
||||||
color: "green",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "update.home_assistant_operating_system_update",
|
|
||||||
name: "OS",
|
|
||||||
icon: "mdi:home-assistant",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "update.home_assistant_supervisor_update",
|
|
||||||
icon: "mdi:home-assistant",
|
|
||||||
name: "Supervisor",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
type: "tile",
|
|
||||||
entity: "update.home_assistant_core_update",
|
|
||||||
name: "Core",
|
|
||||||
icon: "mdi:home-assistant",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
title: "🎉 Updates",
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
});
|
|
@@ -3,15 +3,6 @@ import { DemoConfig } from "../types";
|
|||||||
|
|
||||||
export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
export const demoEntitiesTeachingbirds: DemoConfig["entities"] = () =>
|
||||||
convertEntities({
|
convertEntities({
|
||||||
"todo.shopping_list": {
|
|
||||||
entity_id: "todo.shopping_list",
|
|
||||||
state: "2",
|
|
||||||
attributes: {
|
|
||||||
supported_features: 15,
|
|
||||||
friendly_name: "Shopping List",
|
|
||||||
icon: "mdi:cart",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
"sensor.pollen_grabo": {
|
"sensor.pollen_grabo": {
|
||||||
entity_id: "sensor.pollen_grabo",
|
entity_id: "sensor.pollen_grabo",
|
||||||
state: "",
|
state: "",
|
||||||
|
@@ -220,8 +220,7 @@ export const demoLovelaceTeachingbirds: DemoConfig["lovelace"] = () => ({
|
|||||||
state_filter: ["on"],
|
state_filter: ["on"],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
type: "todo-list",
|
type: "shopping-list",
|
||||||
entity: "todo.shopping_list",
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
entities: [
|
entities: [
|
||||||
|
@@ -1,5 +1,6 @@
|
|||||||
export const demoThemeTeachingbirds = () => ({
|
export const demoThemeTeachingbirds = () => ({
|
||||||
"paper-card-header-color": "var(--paper-item-icon-color)",
|
"paper-card-header-color": "var(--paper-item-icon-color)",
|
||||||
|
"paper-slider-pin-color": "var(--primary-color)",
|
||||||
"paper-listbox-background-color": "#202020",
|
"paper-listbox-background-color": "#202020",
|
||||||
"paper-grey-50": "var(--primary-text-color)",
|
"paper-grey-50": "var(--primary-text-color)",
|
||||||
"paper-item-icon-color": "#d3d3d3",
|
"paper-item-icon-color": "#d3d3d3",
|
||||||
@@ -7,6 +8,8 @@ export const demoThemeTeachingbirds = () => ({
|
|||||||
"primary-color": "#389638",
|
"primary-color": "#389638",
|
||||||
"light-primary-color": "#6f956f",
|
"light-primary-color": "#6f956f",
|
||||||
"label-badge-red": "var(--primary-color)",
|
"label-badge-red": "var(--primary-color)",
|
||||||
|
"paper-slider-secondary-color": "var(--light-primary-color)",
|
||||||
|
"paper-slider-knob-color": "var(--primary-color)",
|
||||||
"paper-listbox-color": "#FFFFFF",
|
"paper-listbox-color": "#FFFFFF",
|
||||||
"paper-toggle-button-checked-bar-color": "var(--light-primary-color)",
|
"paper-toggle-button-checked-bar-color": "var(--light-primary-color)",
|
||||||
"switch-unchecked-track-color": "var(--primary-text-color)",
|
"switch-unchecked-track-color": "var(--primary-text-color)",
|
||||||
@@ -14,7 +17,9 @@ export const demoThemeTeachingbirds = () => ({
|
|||||||
"label-badge-text-color": "var(--text-primary-color)",
|
"label-badge-text-color": "var(--text-primary-color)",
|
||||||
"primary-background-color": "#303030",
|
"primary-background-color": "#303030",
|
||||||
"sidebar-icon-color": "var(--paper-item-icon-color)",
|
"sidebar-icon-color": "var(--paper-item-icon-color)",
|
||||||
|
"paper-slider-active-color": "#d8bf50",
|
||||||
"secondary-background-color": "#2b2b2b",
|
"secondary-background-color": "#2b2b2b",
|
||||||
|
"paper-slider-knob-start-color": "var(--primary-color)",
|
||||||
"paper-item-icon-active-color": "#d8bf50",
|
"paper-item-icon-active-color": "#d8bf50",
|
||||||
"switch-checked-color": "var(--primary-color)",
|
"switch-checked-color": "var(--primary-color)",
|
||||||
"secondary-text-color": "#389638",
|
"secondary-text-color": "#389638",
|
||||||
|
@@ -1,6 +1,5 @@
|
|||||||
import { TemplateResult } from "lit";
|
|
||||||
import { LocalizeFunc } from "../../../src/common/translations/localize";
|
import { LocalizeFunc } from "../../../src/common/translations/localize";
|
||||||
import { LovelaceConfig } from "../../../src/data/lovelace/config/types";
|
import { LovelaceConfig } from "../../../src/data/lovelace";
|
||||||
import { Entity } from "../../../src/fake_data/entity";
|
import { Entity } from "../../../src/fake_data/entity";
|
||||||
|
|
||||||
export interface DemoConfig {
|
export interface DemoConfig {
|
||||||
@@ -8,9 +7,6 @@ export interface DemoConfig {
|
|||||||
name: string;
|
name: string;
|
||||||
authorName: string;
|
authorName: string;
|
||||||
authorUrl: string;
|
authorUrl: string;
|
||||||
description?:
|
|
||||||
| string
|
|
||||||
| ((localize: LocalizeFunc) => string | TemplateResult<1>);
|
|
||||||
lovelace: (localize: LocalizeFunc) => LovelaceConfig;
|
lovelace: (localize: LocalizeFunc) => LovelaceConfig;
|
||||||
entities: (localize: LocalizeFunc) => Entity[];
|
entities: (localize: LocalizeFunc) => Entity[];
|
||||||
theme: () => Record<string, string> | null;
|
theme: () => Record<string, string> | null;
|
||||||
|
@@ -4,7 +4,7 @@ import { customElement, property, state } from "lit/decorators";
|
|||||||
import { until } from "lit/directives/until";
|
import { until } from "lit/directives/until";
|
||||||
import "../../../src/components/ha-card";
|
import "../../../src/components/ha-card";
|
||||||
import "../../../src/components/ha-circular-progress";
|
import "../../../src/components/ha-circular-progress";
|
||||||
import { LovelaceCardConfig } from "../../../src/data/lovelace/config/card";
|
import { LovelaceCardConfig } from "../../../src/data/lovelace";
|
||||||
import { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
import { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
||||||
import { Lovelace, LovelaceCard } from "../../../src/panels/lovelace/types";
|
import { Lovelace, LovelaceCard } from "../../../src/panels/lovelace/types";
|
||||||
import {
|
import {
|
||||||
@@ -39,51 +39,31 @@ export class HADemoCard extends LitElement implements LovelaceCard {
|
|||||||
<div class="picker">
|
<div class="picker">
|
||||||
<div class="label">
|
<div class="label">
|
||||||
${this._switching
|
${this._switching
|
||||||
? html`
|
? html`<ha-circular-progress active></ha-circular-progress>`
|
||||||
<ha-circular-progress indeterminate></ha-circular-progress>
|
|
||||||
`
|
|
||||||
: until(
|
: until(
|
||||||
selectedDemoConfig.then(
|
selectedDemoConfig.then(
|
||||||
(conf) => html`
|
(conf) => html`
|
||||||
${conf.name}
|
${conf.name}
|
||||||
<small>
|
<small>
|
||||||
${this.hass.localize(
|
<a target="_blank" href=${conf.authorUrl}>
|
||||||
"ui.panel.page-demo.cards.demo.demo_by",
|
${this.hass.localize(
|
||||||
{
|
"ui.panel.page-demo.cards.demo.demo_by",
|
||||||
name: html`
|
"name",
|
||||||
<a target="_blank" href=${conf.authorUrl}>
|
conf.authorName
|
||||||
${conf.authorName}
|
)}
|
||||||
</a>
|
</a>
|
||||||
`,
|
|
||||||
}
|
|
||||||
)}
|
|
||||||
</small>
|
</small>
|
||||||
`
|
`
|
||||||
),
|
),
|
||||||
""
|
""
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<mwc-button @click=${this._nextConfig} .disabled=${this._switching}>
|
<mwc-button @click=${this._nextConfig} .disabled=${this._switching}>
|
||||||
${this.hass.localize("ui.panel.page-demo.cards.demo.next_demo")}
|
${this.hass.localize("ui.panel.page-demo.cards.demo.next_demo")}
|
||||||
</mwc-button>
|
</mwc-button>
|
||||||
</div>
|
</div>
|
||||||
<div class="content">
|
<div class="content small-hidden">
|
||||||
<p class="small-hidden">
|
${this.hass.localize("ui.panel.page-demo.cards.demo.introduction")}
|
||||||
${this.hass.localize("ui.panel.page-demo.cards.demo.introduction")}
|
|
||||||
</p>
|
|
||||||
${until(
|
|
||||||
selectedDemoConfig.then((conf) => {
|
|
||||||
if (typeof conf.description === "function") {
|
|
||||||
return conf.description(this.hass.localize);
|
|
||||||
}
|
|
||||||
if (conf.description) {
|
|
||||||
return html`<p>${conf.description}</p>`;
|
|
||||||
}
|
|
||||||
return nothing;
|
|
||||||
}),
|
|
||||||
nothing
|
|
||||||
)}
|
|
||||||
</div>
|
</div>
|
||||||
<div class="actions small-hidden">
|
<div class="actions small-hidden">
|
||||||
<a href="https://www.home-assistant.io" target="_blank">
|
<a href="https://www.home-assistant.io" target="_blank">
|
||||||
@@ -127,7 +107,6 @@ export class HADemoCard extends LitElement implements LovelaceCard {
|
|||||||
css`
|
css`
|
||||||
a {
|
a {
|
||||||
color: var(--primary-color);
|
color: var(--primary-color);
|
||||||
display: inline-block;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.actions a {
|
.actions a {
|
||||||
@@ -135,11 +114,7 @@ export class HADemoCard extends LitElement implements LovelaceCard {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.content {
|
.content {
|
||||||
padding: 0 16px;
|
padding: 16px;
|
||||||
}
|
|
||||||
|
|
||||||
.content p {
|
|
||||||
margin: 16px 0;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.picker {
|
.picker {
|
||||||
@@ -162,8 +137,9 @@ export class HADemoCard extends LitElement implements LovelaceCard {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.actions {
|
.actions {
|
||||||
padding: 0px 8px 4px 8px;
|
padding-left: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@media only screen and (max-width: 500px) {
|
@media only screen and (max-width: 500px) {
|
||||||
.small-hidden {
|
.small-hidden {
|
||||||
display: none;
|
display: none;
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
|
import "../../src/resources/ha-style";
|
||||||
|
import "../../src/resources/roboto";
|
||||||
import "../../src/resources/safari-14-attachshadow-patch";
|
import "../../src/resources/safari-14-attachshadow-patch";
|
||||||
import "./ha-demo";
|
import "./ha-demo";
|
||||||
|
|
||||||
import("../../src/resources/ha-style");
|
|
||||||
|
@@ -17,14 +17,12 @@ import { energyEntities } from "./stubs/entities";
|
|||||||
import { mockEntityRegistry } from "./stubs/entity_registry";
|
import { mockEntityRegistry } from "./stubs/entity_registry";
|
||||||
import { mockEvents } from "./stubs/events";
|
import { mockEvents } from "./stubs/events";
|
||||||
import { mockFrontend } from "./stubs/frontend";
|
import { mockFrontend } from "./stubs/frontend";
|
||||||
import { mockIcons } from "./stubs/icons";
|
|
||||||
import { mockHistory } from "./stubs/history";
|
import { mockHistory } from "./stubs/history";
|
||||||
import { mockLovelace } from "./stubs/lovelace";
|
import { mockLovelace } from "./stubs/lovelace";
|
||||||
import { mockMediaPlayer } from "./stubs/media_player";
|
import { mockMediaPlayer } from "./stubs/media_player";
|
||||||
import { mockPersistentNotification } from "./stubs/persistent_notification";
|
import { mockPersistentNotification } from "./stubs/persistent_notification";
|
||||||
import { mockRecorder } from "./stubs/recorder";
|
import { mockRecorder } from "./stubs/recorder";
|
||||||
import { mockTodo } from "./stubs/todo";
|
import { mockShoppingList } from "./stubs/shopping_list";
|
||||||
import { mockSensor } from "./stubs/sensor";
|
|
||||||
import { mockSystemLog } from "./stubs/system_log";
|
import { mockSystemLog } from "./stubs/system_log";
|
||||||
import { mockTemplate } from "./stubs/template";
|
import { mockTemplate } from "./stubs/template";
|
||||||
import { mockTranslations } from "./stubs/translations";
|
import { mockTranslations } from "./stubs/translations";
|
||||||
@@ -51,14 +49,12 @@ export class HaDemo extends HomeAssistantAppEl {
|
|||||||
mockTranslations(hass);
|
mockTranslations(hass);
|
||||||
mockHistory(hass);
|
mockHistory(hass);
|
||||||
mockRecorder(hass);
|
mockRecorder(hass);
|
||||||
mockTodo(hass);
|
mockShoppingList(hass);
|
||||||
mockSensor(hass);
|
|
||||||
mockSystemLog(hass);
|
mockSystemLog(hass);
|
||||||
mockTemplate(hass);
|
mockTemplate(hass);
|
||||||
mockEvents(hass);
|
mockEvents(hass);
|
||||||
mockMediaPlayer(hass);
|
mockMediaPlayer(hass);
|
||||||
mockFrontend(hass);
|
mockFrontend(hass);
|
||||||
mockIcons(hass);
|
|
||||||
mockEnergy(hass);
|
mockEnergy(hass);
|
||||||
mockPersistentNotification(hass);
|
mockPersistentNotification(hass);
|
||||||
mockConfigEntries(hass);
|
mockConfigEntries(hass);
|
||||||
@@ -72,8 +68,6 @@ export class HaDemo extends HomeAssistantAppEl {
|
|||||||
id: "sensor.co2_intensity",
|
id: "sensor.co2_intensity",
|
||||||
name: null,
|
name: null,
|
||||||
icon: null,
|
icon: null,
|
||||||
labels: [],
|
|
||||||
categories: {},
|
|
||||||
platform: "co2signal",
|
platform: "co2signal",
|
||||||
hidden_by: null,
|
hidden_by: null,
|
||||||
entity_category: null,
|
entity_category: null,
|
||||||
@@ -90,8 +84,6 @@ export class HaDemo extends HomeAssistantAppEl {
|
|||||||
id: "sensor.co2_intensity",
|
id: "sensor.co2_intensity",
|
||||||
name: null,
|
name: null,
|
||||||
icon: null,
|
icon: null,
|
||||||
labels: [],
|
|
||||||
categories: {},
|
|
||||||
platform: "co2signal",
|
platform: "co2signal",
|
||||||
hidden_by: null,
|
hidden_by: null,
|
||||||
entity_category: null,
|
entity_category: null,
|
||||||
|
@@ -74,9 +74,19 @@
|
|||||||
<body>
|
<body>
|
||||||
<div id="ha-launch-screen">
|
<div id="ha-launch-screen">
|
||||||
<div class="ha-launch-screen-spacer"></div>
|
<div class="ha-launch-screen-spacer"></div>
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 240 240">
|
<svg
|
||||||
<path fill="#18BCF2" d="M240 224.762a15 15 0 0 1-15 15H15a15 15 0 0 1-15-15v-90c0-8.25 4.77-19.769 10.61-25.609l98.78-98.7805c5.83-5.83 15.38-5.83 21.21 0l98.79 98.7895c5.83 5.83 10.61 17.36 10.61 25.61v90-.01Z"/>
|
viewBox="0 0 240 240"
|
||||||
<path fill="#F2F4F9" d="m107.27 239.762-40.63-40.63c-2.09.72-4.32 1.13-6.64 1.13-11.3 0-20.5-9.2-20.5-20.5s9.2-20.5 20.5-20.5 20.5 9.2 20.5 20.5c0 2.33-.41 4.56-1.13 6.65l31.63 31.63v-115.88c-6.8-3.3395-11.5-10.3195-11.5-18.3895 0-11.3 9.2-20.5 20.5-20.5s20.5 9.2 20.5 20.5c0 8.07-4.7 15.05-11.5 18.3895v81.27l31.46-31.46c-.62-1.96-.96-4.04-.96-6.2 0-11.3 9.2-20.5 20.5-20.5s20.5 9.2 20.5 20.5-9.2 20.5-20.5 20.5c-2.5 0-4.88-.47-7.09-1.29L129 208.892v30.88z"/>
|
fill="none"
|
||||||
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
d="M240 224.762C240 233.012 233.25 239.762 225 239.762H15C6.75 239.762 0 233.012 0 224.762V134.762C0 126.512 4.77 114.993 10.61 109.153L109.39 10.3725C115.22 4.5425 124.77 4.5425 130.6 10.3725L229.39 109.162C235.22 114.992 240 126.522 240 134.772V224.772V224.762Z"
|
||||||
|
fill="#F2F4F9"
|
||||||
|
/>
|
||||||
|
<path
|
||||||
|
d="M229.39 109.153L130.61 10.3725C124.78 4.5425 115.23 4.5425 109.4 10.3725L10.61 109.153C4.78 114.983 0 126.512 0 134.762V224.762C0 233.012 6.75 239.762 15 239.762H107.27L66.64 199.132C64.55 199.852 62.32 200.262 60 200.262C48.7 200.262 39.5 191.062 39.5 179.762C39.5 168.462 48.7 159.262 60 159.262C71.3 159.262 80.5 168.462 80.5 179.762C80.5 182.092 80.09 184.322 79.37 186.412L111 218.042V102.162C104.2 98.8225 99.5 91.8425 99.5 83.7725C99.5 72.4725 108.7 63.2725 120 63.2725C131.3 63.2725 140.5 72.4725 140.5 83.7725C140.5 91.8425 135.8 98.8225 129 102.162V183.432L160.46 151.972C159.84 150.012 159.5 147.932 159.5 145.772C159.5 134.472 168.7 125.272 180 125.272C191.3 125.272 200.5 134.472 200.5 145.772C200.5 157.072 191.3 166.272 180 166.272C177.5 166.272 175.12 165.802 172.91 164.982L129 208.892V239.772H225C233.25 239.772 240 233.022 240 224.772V134.772C240 126.522 235.23 115.002 229.39 109.162V109.153Z"
|
||||||
|
fill="#18BCF2"
|
||||||
|
/>
|
||||||
</svg>
|
</svg>
|
||||||
<div id="ha-launch-screen-info-box" class="ha-launch-screen-spacer"></div>
|
<div id="ha-launch-screen-info-box" class="ha-launch-screen-spacer"></div>
|
||||||
</div>
|
</div>
|
||||||
|
@@ -4,11 +4,4 @@ import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
|||||||
export const mockAreaRegistry = (
|
export const mockAreaRegistry = (
|
||||||
hass: MockHomeAssistant,
|
hass: MockHomeAssistant,
|
||||||
data: AreaRegistryEntry[] = []
|
data: AreaRegistryEntry[] = []
|
||||||
) => {
|
) => hass.mockWS("config/area_registry/list", () => data);
|
||||||
hass.mockWS("config/area_registry/list", () => data);
|
|
||||||
const areas = {};
|
|
||||||
data.forEach((area) => {
|
|
||||||
areas[area.area_id] = area;
|
|
||||||
});
|
|
||||||
hass.updateHass({ areas });
|
|
||||||
};
|
|
||||||
|
@@ -10,7 +10,6 @@ export const mockConfigEntries = (hass: MockHomeAssistant) => {
|
|||||||
supports_options: false,
|
supports_options: false,
|
||||||
supports_remove_device: false,
|
supports_remove_device: false,
|
||||||
supports_unload: true,
|
supports_unload: true,
|
||||||
supports_reconfigure: true,
|
|
||||||
pref_disable_new_entities: false,
|
pref_disable_new_entities: false,
|
||||||
pref_disable_polling: false,
|
pref_disable_polling: false,
|
||||||
disabled_by: null,
|
disabled_by: null,
|
||||||
|
@@ -4,11 +4,4 @@ import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
|||||||
export const mockDeviceRegistry = (
|
export const mockDeviceRegistry = (
|
||||||
hass: MockHomeAssistant,
|
hass: MockHomeAssistant,
|
||||||
data: DeviceRegistryEntry[] = []
|
data: DeviceRegistryEntry[] = []
|
||||||
) => {
|
) => hass.mockWS("config/device_registry/list", () => data);
|
||||||
hass.mockWS("config/device_registry/list", () => data);
|
|
||||||
const devices = {};
|
|
||||||
data.forEach((device) => {
|
|
||||||
devices[device.id] = device;
|
|
||||||
});
|
|
||||||
hass.updateHass({ devices });
|
|
||||||
};
|
|
||||||
|
@@ -1,7 +0,0 @@
|
|||||||
import { FloorRegistryEntry } from "../../../src/data/floor_registry";
|
|
||||||
import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
|
||||||
|
|
||||||
export const mockFloorRegistry = (
|
|
||||||
hass: MockHomeAssistant,
|
|
||||||
data: FloorRegistryEntry[] = []
|
|
||||||
) => hass.mockWS("config/floor_registry/list", () => data);
|
|
@@ -1,33 +0,0 @@
|
|||||||
import { IconCategory } from "../../../src/data/icons";
|
|
||||||
import { ENTITY_COMPONENT_ICONS } from "../../../src/fake_data/entity_component_icons";
|
|
||||||
import { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
|
||||||
|
|
||||||
export const mockIcons = (hass: MockHomeAssistant) => {
|
|
||||||
hass.mockWS(
|
|
||||||
"frontend/get_icons",
|
|
||||||
async ({
|
|
||||||
category,
|
|
||||||
integration,
|
|
||||||
}: {
|
|
||||||
category: IconCategory;
|
|
||||||
integration?: string;
|
|
||||||
}) => {
|
|
||||||
if (integration) {
|
|
||||||
try {
|
|
||||||
const response = await fetch(
|
|
||||||
`https://raw.githubusercontent.com/home-assistant/core/dev/homeassistant/components/${integration}/icons.json`
|
|
||||||
).then((resp) => resp.json());
|
|
||||||
return { resources: { [integration]: response[category] || {} } };
|
|
||||||
} catch {
|
|
||||||
return { resources: {} };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (category === "entity_component") {
|
|
||||||
return {
|
|
||||||
resources: ENTITY_COMPONENT_ICONS,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
return { resources: {} };
|
|
||||||
}
|
|
||||||
);
|
|
||||||
};
|
|
@@ -1,7 +0,0 @@
|
|||||||
import { LabelRegistryEntry } from "../../../src/data/label_registry";
|
|
||||||
import type { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
|
||||||
|
|
||||||
export const mockLabelRegistry = (
|
|
||||||
hass: MockHomeAssistant,
|
|
||||||
data: LabelRegistryEntry[] = []
|
|
||||||
) => hass.mockWS("config/label_registry/list", () => data);
|
|
@@ -43,8 +43,8 @@ const generateMeanStatistics = (
|
|||||||
period === "day"
|
period === "day"
|
||||||
? addDays(currentDate, 1)
|
? addDays(currentDate, 1)
|
||||||
: period === "month"
|
: period === "month"
|
||||||
? addMonths(currentDate, 1)
|
? addMonths(currentDate, 1)
|
||||||
: addHours(currentDate, 1);
|
: addHours(currentDate, 1);
|
||||||
}
|
}
|
||||||
return statistics;
|
return statistics;
|
||||||
};
|
};
|
||||||
@@ -80,8 +80,8 @@ const generateSumStatistics = (
|
|||||||
period === "day"
|
period === "day"
|
||||||
? addDays(currentDate, 1)
|
? addDays(currentDate, 1)
|
||||||
: period === "month"
|
: period === "month"
|
||||||
? addMonths(currentDate, 1)
|
? addMonths(currentDate, 1)
|
||||||
: addHours(currentDate, 1);
|
: addHours(currentDate, 1);
|
||||||
}
|
}
|
||||||
return statistics;
|
return statistics;
|
||||||
};
|
};
|
||||||
|
@@ -1,58 +0,0 @@
|
|||||||
import { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
|
||||||
|
|
||||||
export const mockSensor = (hass: MockHomeAssistant) => {
|
|
||||||
hass.mockWS("sensor/numeric_device_classes", () => [
|
|
||||||
{
|
|
||||||
numeric_device_classes: [
|
|
||||||
"volume_storage",
|
|
||||||
"gas",
|
|
||||||
"data_size",
|
|
||||||
"irradiance",
|
|
||||||
"wind_speed",
|
|
||||||
"volatile_organic_compounds",
|
|
||||||
"volatile_organic_compounds_parts",
|
|
||||||
"voltage",
|
|
||||||
"frequency",
|
|
||||||
"precipitation_intensity",
|
|
||||||
"volume",
|
|
||||||
"precipitation",
|
|
||||||
"battery",
|
|
||||||
"nitrogen_dioxide",
|
|
||||||
"speed",
|
|
||||||
"signal_strength",
|
|
||||||
"pm1",
|
|
||||||
"nitrous_oxide",
|
|
||||||
"atmospheric_pressure",
|
|
||||||
"data_rate",
|
|
||||||
"temperature",
|
|
||||||
"power_factor",
|
|
||||||
"aqi",
|
|
||||||
"current",
|
|
||||||
"volume_flow_rate",
|
|
||||||
"humidity",
|
|
||||||
"duration",
|
|
||||||
"ozone",
|
|
||||||
"distance",
|
|
||||||
"pressure",
|
|
||||||
"pm25",
|
|
||||||
"weight",
|
|
||||||
"energy",
|
|
||||||
"carbon_monoxide",
|
|
||||||
"apparent_power",
|
|
||||||
"illuminance",
|
|
||||||
"energy_storage",
|
|
||||||
"moisture",
|
|
||||||
"power",
|
|
||||||
"water",
|
|
||||||
"carbon_dioxide",
|
|
||||||
"ph",
|
|
||||||
"reactive_power",
|
|
||||||
"monetary",
|
|
||||||
"nitrogen_monoxide",
|
|
||||||
"pm10",
|
|
||||||
"sound_pressure",
|
|
||||||
"sulphur_dioxide",
|
|
||||||
],
|
|
||||||
},
|
|
||||||
]);
|
|
||||||
};
|
|
44
demo/src/stubs/shopping_list.ts
Normal file
44
demo/src/stubs/shopping_list.ts
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
import { ShoppingListItem } from "../../../src/data/shopping-list";
|
||||||
|
import { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
||||||
|
|
||||||
|
let items: ShoppingListItem[] = [
|
||||||
|
{
|
||||||
|
id: 12,
|
||||||
|
name: "Milk",
|
||||||
|
complete: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 13,
|
||||||
|
name: "Eggs",
|
||||||
|
complete: false,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: 14,
|
||||||
|
name: "Oranges",
|
||||||
|
complete: true,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
export const mockShoppingList = (hass: MockHomeAssistant) => {
|
||||||
|
hass.mockWS("shopping_list/items", () => items);
|
||||||
|
hass.mockWS("shopping_list/items/add", (msg) => {
|
||||||
|
const item: ShoppingListItem = {
|
||||||
|
id: new Date().getTime(),
|
||||||
|
complete: false,
|
||||||
|
name: msg.name,
|
||||||
|
};
|
||||||
|
items.push(item);
|
||||||
|
hass.mockEvent("shopping_list_updated");
|
||||||
|
return item;
|
||||||
|
});
|
||||||
|
hass.mockWS("shopping_list/items/update", ({ type, item_id, ...updates }) => {
|
||||||
|
items = items.map((item) =>
|
||||||
|
item.id === item_id ? { ...item, ...updates } : item
|
||||||
|
);
|
||||||
|
hass.mockEvent("shopping_list_updated");
|
||||||
|
});
|
||||||
|
hass.mockWS("shopping_list/items/clear", () => {
|
||||||
|
items = items.filter((item) => !item.complete);
|
||||||
|
hass.mockEvent("shopping_list_updated");
|
||||||
|
});
|
||||||
|
};
|
@@ -1,25 +0,0 @@
|
|||||||
import { TodoItem, TodoItemStatus } from "../../../src/data/todo";
|
|
||||||
import { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
|
||||||
|
|
||||||
export const mockTodo = (hass: MockHomeAssistant) => {
|
|
||||||
hass.mockWS("todo/item/list", () => ({
|
|
||||||
items: [
|
|
||||||
{
|
|
||||||
uid: "12",
|
|
||||||
summary: "Milk",
|
|
||||||
status: TodoItemStatus.NeedsAction,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
uid: "13",
|
|
||||||
summary: "Eggs",
|
|
||||||
status: TodoItemStatus.NeedsAction,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
uid: "14",
|
|
||||||
summary: "Oranges",
|
|
||||||
status: TodoItemStatus.Completed,
|
|
||||||
},
|
|
||||||
] as TodoItem[],
|
|
||||||
}));
|
|
||||||
hass.mockWS("todo/item/subscribe", (_msg, _hass) => () => {});
|
|
||||||
};
|
|
@@ -1,5 +1,5 @@
|
|||||||
import { Button } from "@material/mwc-button";
|
import { Button } from "@material/mwc-button";
|
||||||
import { html, LitElement, css, TemplateResult, nothing } from "lit";
|
import { html, LitElement, css, TemplateResult } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import { applyThemesOnElement } from "../../../src/common/dom/apply_themes_on_element";
|
import { applyThemesOnElement } from "../../../src/common/dom/apply_themes_on_element";
|
||||||
import { fireEvent } from "../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../src/common/dom/fire_event";
|
||||||
@@ -9,9 +9,9 @@ import "../../../src/components/ha-card";
|
|||||||
class DemoBlackWhiteRow extends LitElement {
|
class DemoBlackWhiteRow extends LitElement {
|
||||||
@property() title!: string;
|
@property() title!: string;
|
||||||
|
|
||||||
@property() value?: any;
|
@property() value!: any;
|
||||||
|
|
||||||
@property({ type: Boolean }) public disabled = false;
|
@property() disabled = false;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
return html`
|
return html`
|
||||||
@@ -45,9 +45,7 @@ class DemoBlackWhiteRow extends LitElement {
|
|||||||
</mwc-button>
|
</mwc-button>
|
||||||
</div>
|
</div>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
${this.value
|
<pre>${JSON.stringify(this.value, undefined, 2)}</pre>
|
||||||
? html`<pre>${JSON.stringify(this.value, undefined, 2)}</pre>`
|
|
||||||
: nothing}
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
|
@@ -11,11 +11,11 @@ export interface DemoCardConfig {
|
|||||||
|
|
||||||
@customElement("demo-card")
|
@customElement("demo-card")
|
||||||
class DemoCard extends LitElement {
|
class DemoCard extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property() public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property({ attribute: false }) public config!: DemoCardConfig;
|
@property() public config!: DemoCardConfig;
|
||||||
|
|
||||||
@property({ type: Boolean }) public showConfig = false;
|
@property() public showConfig = false;
|
||||||
|
|
||||||
@state() private _size?: number;
|
@state() private _size?: number;
|
||||||
|
|
||||||
|
@@ -10,9 +10,9 @@ import "../ha-demo-options";
|
|||||||
|
|
||||||
@customElement("demo-cards")
|
@customElement("demo-cards")
|
||||||
class DemoCards extends LitElement {
|
class DemoCards extends LitElement {
|
||||||
@property({ attribute: false }) public configs!: DemoCardConfig[];
|
@property() public configs!: DemoCardConfig[];
|
||||||
|
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property() public hass!: HomeAssistant;
|
||||||
|
|
||||||
@state() private _showConfig = false;
|
@state() private _showConfig = false;
|
||||||
|
|
||||||
|
@@ -8,11 +8,11 @@ import { HomeAssistant } from "../../../src/types";
|
|||||||
|
|
||||||
@customElement("demo-more-info")
|
@customElement("demo-more-info")
|
||||||
class DemoMoreInfo extends LitElement {
|
class DemoMoreInfo extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property() public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property() public entityId!: string;
|
@property() public entityId!: string;
|
||||||
|
|
||||||
@property({ type: Boolean }) public showConfig = false;
|
@property() public showConfig!: boolean;
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const state = this._getState(this.entityId, this.hass.states);
|
const state = this._getState(this.entityId, this.hass.states);
|
||||||
@@ -23,7 +23,7 @@ class DemoMoreInfo extends LitElement {
|
|||||||
<state-card-content
|
<state-card-content
|
||||||
.stateObj=${state}
|
.stateObj=${state}
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
inDialog
|
in-dialog
|
||||||
></state-card-content>
|
></state-card-content>
|
||||||
|
|
||||||
<more-info-content
|
<more-info-content
|
||||||
|
@@ -1,19 +1,19 @@
|
|||||||
import { LitElement, css, html } from "lit";
|
import { LitElement, css, html } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import { applyThemesOnElement } from "../../../src/common/dom/apply_themes_on_element";
|
import { applyThemesOnElement } from "../../../src/common/dom/apply_themes_on_element";
|
||||||
import "../../../src/components/ha-formfield";
|
import "../../../src/components/ha-formfield";
|
||||||
import "../../../src/components/ha-switch";
|
import "../../../src/components/ha-switch";
|
||||||
import { HomeAssistant } from "../../../src/types";
|
|
||||||
import "../ha-demo-options";
|
|
||||||
import "./demo-more-info";
|
import "./demo-more-info";
|
||||||
|
import "../ha-demo-options";
|
||||||
|
import { HomeAssistant } from "../../../src/types";
|
||||||
|
|
||||||
@customElement("demo-more-infos")
|
@customElement("demo-more-infos")
|
||||||
class DemoMoreInfos extends LitElement {
|
class DemoMoreInfos extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property() public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property({ type: Array }) public entities!: string[];
|
@property() public entities!: [];
|
||||||
|
|
||||||
@state() private _showConfig = false;
|
@property({ attribute: false }) _showConfig: boolean = false;
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
return html`
|
return html`
|
||||||
|
@@ -509,7 +509,7 @@ export default {
|
|||||||
away_mode: "on",
|
away_mode: "on",
|
||||||
aux_heat: "off",
|
aux_heat: "off",
|
||||||
unit_of_measurement: "°C",
|
unit_of_measurement: "°C",
|
||||||
friendly_name: "HVAC",
|
friendly_name: "Hvac",
|
||||||
supported_features: 3833,
|
supported_features: 3833,
|
||||||
},
|
},
|
||||||
last_changed: "2018-07-19T10:44:46.200650+00:00",
|
last_changed: "2018-07-19T10:44:46.200650+00:00",
|
||||||
|
@@ -17,7 +17,6 @@ export const basicTrace: DemoTrace = {
|
|||||||
{
|
{
|
||||||
path: "trigger/0",
|
path: "trigger/0",
|
||||||
timestamp: "2021-03-25T04:36:51.223693+00:00",
|
timestamp: "2021-03-25T04:36:51.223693+00:00",
|
||||||
changed_variables: {},
|
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
"condition/0": [
|
"condition/0": [
|
||||||
|
@@ -17,7 +17,6 @@ export const motionLightTrace: DemoTrace = {
|
|||||||
{
|
{
|
||||||
path: "trigger/0",
|
path: "trigger/0",
|
||||||
timestamp: "2021-03-25T04:36:51.223693+00:00",
|
timestamp: "2021-03-25T04:36:51.223693+00:00",
|
||||||
changed_variables: {},
|
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
"action/0": [
|
"action/0": [
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
|
import "../../src/resources/ha-style";
|
||||||
|
import "../../src/resources/roboto";
|
||||||
import "./ha-gallery";
|
import "./ha-gallery";
|
||||||
|
|
||||||
import("../../src/resources/ha-style");
|
|
||||||
|
|
||||||
document.body.appendChild(document.createElement("ha-gallery"));
|
document.body.appendChild(document.createElement("ha-gallery"));
|
||||||
|
@@ -1,14 +1,14 @@
|
|||||||
|
import { mdiMenu } from "@mdi/js";
|
||||||
import "@material/mwc-drawer";
|
import "@material/mwc-drawer";
|
||||||
import "@material/mwc-top-app-bar-fixed";
|
import "@material/mwc-top-app-bar-fixed";
|
||||||
import { mdiMenu } from "@mdi/js";
|
import { html, css, LitElement, PropertyValues } from "lit";
|
||||||
import { LitElement, PropertyValues, css, html } from "lit";
|
import { customElement, property, query } from "lit/decorators";
|
||||||
import { customElement, query, state } from "lit/decorators";
|
|
||||||
import { dynamicElement } from "../../src/common/dom/dynamic-element-directive";
|
|
||||||
import { HaExpansionPanel } from "../../src/components/ha-expansion-panel";
|
|
||||||
import "../../src/components/ha-icon-button";
|
import "../../src/components/ha-icon-button";
|
||||||
import "../../src/managers/notification-manager";
|
import "../../src/managers/notification-manager";
|
||||||
|
import { HaExpansionPanel } from "../../src/components/ha-expansion-panel";
|
||||||
import { haStyle } from "../../src/resources/styles";
|
import { haStyle } from "../../src/resources/styles";
|
||||||
import { PAGES, SIDEBAR } from "../build/import-pages";
|
import { PAGES, SIDEBAR } from "../build/import-pages";
|
||||||
|
import { dynamicElement } from "../../src/common/dom/dynamic-element-directive";
|
||||||
import "./components/page-description";
|
import "./components/page-description";
|
||||||
|
|
||||||
const GITHUB_DEMO_URL =
|
const GITHUB_DEMO_URL =
|
||||||
@@ -24,7 +24,7 @@ const FAKE_HASS = {
|
|||||||
|
|
||||||
@customElement("ha-gallery")
|
@customElement("ha-gallery")
|
||||||
class HaGallery extends LitElement {
|
class HaGallery extends LitElement {
|
||||||
@state() private _page =
|
@property() private _page =
|
||||||
document.location.hash.substring(1) ||
|
document.location.hash.substring(1) ||
|
||||||
`${SIDEBAR[0].category}/${SIDEBAR[0].pages![0]}`;
|
`${SIDEBAR[0].category}/${SIDEBAR[0].pages![0]}`;
|
||||||
|
|
||||||
|
@@ -21,10 +21,10 @@ const ENTITIES = [
|
|||||||
}),
|
}),
|
||||||
];
|
];
|
||||||
|
|
||||||
const conditions: Condition[] = [
|
const conditions = [
|
||||||
{ condition: "and", conditions: [] },
|
{ condition: "and" },
|
||||||
{ condition: "not", conditions: [] },
|
{ condition: "not" },
|
||||||
{ condition: "or", conditions: [] },
|
{ condition: "or" },
|
||||||
{ condition: "state", entity_id: "light.kitchen", state: "on" },
|
{ condition: "state", entity_id: "light.kitchen", state: "on" },
|
||||||
{
|
{
|
||||||
condition: "numeric_state",
|
condition: "numeric_state",
|
||||||
@@ -34,11 +34,11 @@ const conditions: Condition[] = [
|
|||||||
above: 20,
|
above: 20,
|
||||||
},
|
},
|
||||||
{ condition: "sun", after: "sunset" },
|
{ condition: "sun", after: "sunset" },
|
||||||
{ condition: "sun", after: "sunrise", before_offset: 3600 },
|
{ condition: "sun", after: "sunrise", offset: "-01:00" },
|
||||||
{ condition: "zone", entity_id: "device_tracker.person", zone: "zone.home" },
|
{ condition: "zone", entity_id: "device_tracker.person", zone: "zone.home" },
|
||||||
{ condition: "trigger", id: "motion" },
|
{ condition: "trigger", id: "motion" },
|
||||||
{ condition: "time" },
|
{ condition: "time" },
|
||||||
{ condition: "template", value_template: "" },
|
{ condition: "template" },
|
||||||
];
|
];
|
||||||
|
|
||||||
const initialCondition: Condition = {
|
const initialCondition: Condition = {
|
||||||
|
@@ -80,7 +80,7 @@ const SCHEMAS: { name: string; conditions: ConditionWithShorthand[] }[] = [
|
|||||||
];
|
];
|
||||||
|
|
||||||
@customElement("demo-automation-editor-condition")
|
@customElement("demo-automation-editor-condition")
|
||||||
export class DemoAutomationEditorCondition extends LitElement {
|
class DemoHaAutomationEditorCondition extends LitElement {
|
||||||
@state() private hass!: HomeAssistant;
|
@state() private hass!: HomeAssistant;
|
||||||
|
|
||||||
@state() private _disabled = false;
|
@state() private _disabled = false;
|
||||||
@@ -155,6 +155,6 @@ export class DemoAutomationEditorCondition extends LitElement {
|
|||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface HTMLElementTagNameMap {
|
interface HTMLElementTagNameMap {
|
||||||
"demo-automation-editor-condition": DemoAutomationEditorCondition;
|
"demo-ha-automation-editor-condition": DemoHaAutomationEditorCondition;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -126,7 +126,7 @@ const SCHEMAS: { name: string; triggers: Trigger[] }[] = [
|
|||||||
];
|
];
|
||||||
|
|
||||||
@customElement("demo-automation-editor-trigger")
|
@customElement("demo-automation-editor-trigger")
|
||||||
export class DemoAutomationEditorTrigger extends LitElement {
|
class DemoHaAutomationEditorTrigger extends LitElement {
|
||||||
@state() private hass!: HomeAssistant;
|
@state() private hass!: HomeAssistant;
|
||||||
|
|
||||||
@state() private _disabled = false;
|
@state() private _disabled = false;
|
||||||
@@ -201,6 +201,6 @@ export class DemoAutomationEditorTrigger extends LitElement {
|
|||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface HTMLElementTagNameMap {
|
interface HTMLElementTagNameMap {
|
||||||
"demo-automation-editor-trigger": DemoAutomationEditorTrigger;
|
"demo-ha-automation-editor-trigger": DemoHaAutomationEditorTrigger;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -3,6 +3,7 @@
|
|||||||
import { css, html, LitElement, nothing } from "lit";
|
import { css, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import "../../../../src/components/ha-card";
|
import "../../../../src/components/ha-card";
|
||||||
|
import "../../../../src/components/trace/hat-script-graph";
|
||||||
import "../../../../src/components/trace/hat-trace-timeline";
|
import "../../../../src/components/trace/hat-trace-timeline";
|
||||||
import { provideHass } from "../../../../src/fake_data/provide_hass";
|
import { provideHass } from "../../../../src/fake_data/provide_hass";
|
||||||
import { HomeAssistant } from "../../../../src/types";
|
import { HomeAssistant } from "../../../../src/types";
|
||||||
@@ -55,7 +56,6 @@ export class DemoAutomationTraceTimeline extends LitElement {
|
|||||||
super.firstUpdated(changedProps);
|
super.firstUpdated(changedProps);
|
||||||
const hass = provideHass(this);
|
const hass = provideHass(this);
|
||||||
hass.updateTranslations(null, "en");
|
hass.updateTranslations(null, "en");
|
||||||
hass.updateTranslations("config", "en");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles() {
|
static get styles() {
|
||||||
|
@@ -60,7 +60,6 @@ export class DemoAutomationTrace extends LitElement {
|
|||||||
super.firstUpdated(changedProps);
|
super.firstUpdated(changedProps);
|
||||||
const hass = provideHass(this);
|
const hass = provideHass(this);
|
||||||
hass.updateTranslations(null, "en");
|
hass.updateTranslations(null, "en");
|
||||||
hass.updateTranslations("config", "en");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles() {
|
static get styles() {
|
||||||
|
@@ -1,10 +1,8 @@
|
|||||||
import { css, html, LitElement, TemplateResult, nothing } from "lit";
|
import { css, html, LitElement, TemplateResult } from "lit";
|
||||||
import { customElement } from "lit/decorators";
|
import { customElement } from "lit/decorators";
|
||||||
import "../../../../src/components/ha-card";
|
import "../../../../src/components/ha-card";
|
||||||
import "../../../../src/components/chips/ha-chip-set";
|
import "../../../../src/components/ha-chip";
|
||||||
import "../../../../src/components/chips/ha-assist-chip";
|
import "../../../../src/components/ha-chip-set";
|
||||||
import "../../../../src/components/chips/ha-input-chip";
|
|
||||||
import "../../../../src/components/chips/ha-filter-chip";
|
|
||||||
import "../../../../src/components/ha-svg-icon";
|
import "../../../../src/components/ha-svg-icon";
|
||||||
import { mdiHomeAssistant } from "../../../../src/resources/home-assistant-logo-svg";
|
import { mdiHomeAssistant } from "../../../../src/resources/home-assistant-logo-svg";
|
||||||
|
|
||||||
@@ -12,6 +10,10 @@ const chips: {
|
|||||||
icon?: string;
|
icon?: string;
|
||||||
content?: string;
|
content?: string;
|
||||||
}[] = [
|
}[] = [
|
||||||
|
{},
|
||||||
|
{
|
||||||
|
icon: mdiHomeAssistant,
|
||||||
|
},
|
||||||
{
|
{
|
||||||
content: "Content",
|
content: "Content",
|
||||||
},
|
},
|
||||||
@@ -27,73 +29,31 @@ export class DemoHaChips extends LitElement {
|
|||||||
return html`
|
return html`
|
||||||
<ha-card header="ha-chip demo">
|
<ha-card header="ha-chip demo">
|
||||||
<div class="card-content">
|
<div class="card-content">
|
||||||
<p>Action chip</p>
|
${chips.map(
|
||||||
|
(chip) => html`
|
||||||
|
<ha-chip .hasIcon=${chip.icon !== undefined}>
|
||||||
|
${chip.icon
|
||||||
|
? html`<ha-svg-icon slot="icon" .path=${chip.icon}>
|
||||||
|
</ha-svg-icon>`
|
||||||
|
: ""}
|
||||||
|
${chip.content}
|
||||||
|
</ha-chip>
|
||||||
|
`
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</ha-card>
|
||||||
|
<ha-card header="ha-chip-set demo">
|
||||||
|
<div class="card-content">
|
||||||
<ha-chip-set>
|
<ha-chip-set>
|
||||||
${chips.map(
|
${chips.map(
|
||||||
(chip) => html`
|
(chip) => html`
|
||||||
<ha-assist-chip .label=${chip.content}>
|
<ha-chip .hasIcon=${chip.icon !== undefined}>
|
||||||
${chip.icon
|
|
||||||
? html`<ha-svg-icon slot="icon" .path=${chip.icon}>
|
|
||||||
</ha-svg-icon>`
|
|
||||||
: nothing}
|
|
||||||
</ha-assist-chip>
|
|
||||||
`
|
|
||||||
)}
|
|
||||||
${chips.map(
|
|
||||||
(chip) => html`
|
|
||||||
<ha-assist-chip .label=${chip.content} selected>
|
|
||||||
${chip.icon
|
|
||||||
? html`<ha-svg-icon slot="icon" .path=${chip.icon}>
|
|
||||||
</ha-svg-icon>`
|
|
||||||
: nothing}
|
|
||||||
</ha-assist-chip>
|
|
||||||
`
|
|
||||||
)}
|
|
||||||
</ha-chip-set>
|
|
||||||
<p>Filter chip</p>
|
|
||||||
<ha-chip-set>
|
|
||||||
${chips.map(
|
|
||||||
(chip) => html`
|
|
||||||
<ha-filter-chip .label=${chip.content}>
|
|
||||||
${chip.icon
|
|
||||||
? html`<ha-svg-icon slot="icon" .path=${chip.icon}>
|
|
||||||
</ha-svg-icon>`
|
|
||||||
: nothing}
|
|
||||||
</ha-filter-chip>
|
|
||||||
`
|
|
||||||
)}
|
|
||||||
${chips.map(
|
|
||||||
(chip) => html`
|
|
||||||
<ha-filter-chip .label=${chip.content} selected>
|
|
||||||
${chip.icon
|
|
||||||
? html`<ha-svg-icon slot="icon" .path=${chip.icon}>
|
|
||||||
</ha-svg-icon>`
|
|
||||||
: nothing}
|
|
||||||
</ha-filter-chip>
|
|
||||||
`
|
|
||||||
)}
|
|
||||||
</ha-chip-set>
|
|
||||||
<p>Input chip</p>
|
|
||||||
<ha-chip-set>
|
|
||||||
${chips.map(
|
|
||||||
(chip) => html`
|
|
||||||
<ha-input-chip .label=${chip.content}>
|
|
||||||
${chip.icon
|
${chip.icon
|
||||||
? html`<ha-svg-icon slot="icon" .path=${chip.icon}>
|
? html`<ha-svg-icon slot="icon" .path=${chip.icon}>
|
||||||
</ha-svg-icon>`
|
</ha-svg-icon>`
|
||||||
: ""}
|
: ""}
|
||||||
${chip.content}
|
${chip.content}
|
||||||
</ha-input-chip>
|
</ha-chip>
|
||||||
`
|
|
||||||
)}
|
|
||||||
${chips.map(
|
|
||||||
(chip) => html`
|
|
||||||
<ha-input-chip .label=${chip.content} selected>
|
|
||||||
${chip.icon
|
|
||||||
? html`<ha-svg-icon slot="icon" .path=${chip.icon}>
|
|
||||||
</ha-svg-icon>`
|
|
||||||
: nothing}
|
|
||||||
</ha-input-chip>
|
|
||||||
`
|
`
|
||||||
)}
|
)}
|
||||||
</ha-chip-set>
|
</ha-chip-set>
|
||||||
@@ -108,10 +68,12 @@ export class DemoHaChips extends LitElement {
|
|||||||
max-width: 600px;
|
max-width: 600px;
|
||||||
margin: 24px auto;
|
margin: 24px auto;
|
||||||
}
|
}
|
||||||
|
ha-chip {
|
||||||
|
margin-bottom: 4px;
|
||||||
|
}
|
||||||
.card-content {
|
.card-content {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
align-items: flex-start;
|
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
@@ -1,4 +0,0 @@
|
|||||||
---
|
|
||||||
title: Circular Progress
|
|
||||||
subtitle: Can be used to indicate an ongoing task.
|
|
||||||
---
|
|
@@ -1,64 +0,0 @@
|
|||||||
import { html, css, LitElement, TemplateResult } from "lit";
|
|
||||||
import { customElement, property } from "lit/decorators";
|
|
||||||
import "../../../../src/components/ha-bar";
|
|
||||||
import "../../../../src/components/ha-card";
|
|
||||||
import "../../../../src/components/ha-circular-progress";
|
|
||||||
import "@material/web/progress/circular-progress";
|
|
||||||
import { HomeAssistant } from "../../../../src/types";
|
|
||||||
|
|
||||||
@customElement("demo-components-ha-circular-progress")
|
|
||||||
export class DemoHaCircularProgress extends LitElement {
|
|
||||||
@property({ attribute: false }) hass!: HomeAssistant;
|
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
|
||||||
return html`<ha-card header="Basic circular progress">
|
|
||||||
<div class="card-content">
|
|
||||||
<ha-circular-progress indeterminate></ha-circular-progress></div
|
|
||||||
></ha-card>
|
|
||||||
<ha-card header="Different circular progress sizes">
|
|
||||||
<div class="card-content">
|
|
||||||
<ha-circular-progress
|
|
||||||
indeterminate
|
|
||||||
size="tiny"
|
|
||||||
></ha-circular-progress>
|
|
||||||
<ha-circular-progress
|
|
||||||
indeterminate
|
|
||||||
size="small"
|
|
||||||
></ha-circular-progress>
|
|
||||||
<ha-circular-progress
|
|
||||||
indeterminate
|
|
||||||
size="medium"
|
|
||||||
></ha-circular-progress>
|
|
||||||
<ha-circular-progress
|
|
||||||
indeterminate
|
|
||||||
size="large"
|
|
||||||
></ha-circular-progress></div
|
|
||||||
></ha-card>
|
|
||||||
<ha-card header="Circular progress with an aria-label">
|
|
||||||
<div class="card-content">
|
|
||||||
<ha-circular-progress
|
|
||||||
indeterminate
|
|
||||||
aria-label="Doing something..."
|
|
||||||
></ha-circular-progress>
|
|
||||||
<ha-circular-progress
|
|
||||||
indeterminate
|
|
||||||
.ariaLabel=${"Doing something..."}
|
|
||||||
></ha-circular-progress></div
|
|
||||||
></ha-card>`;
|
|
||||||
}
|
|
||||||
|
|
||||||
static get styles() {
|
|
||||||
return css`
|
|
||||||
ha-card {
|
|
||||||
max-width: 600px;
|
|
||||||
margin: 24px auto;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
declare global {
|
|
||||||
interface HTMLElementTagNameMap {
|
|
||||||
"demo-components-ha-circular-progress": DemoHaCircularProgress;
|
|
||||||
}
|
|
||||||
}
|
|
@@ -59,7 +59,7 @@ export class DemoHaBarButton extends LitElement {
|
|||||||
<ha-control-button
|
<ha-control-button
|
||||||
class=${ifDefined(btn.class)}
|
class=${ifDefined(btn.class)}
|
||||||
label=${ifDefined(btn.label)}
|
label=${ifDefined(btn.label)}
|
||||||
?disabled=${btn.disabled}
|
disabled=${ifDefined(btn.disabled)}
|
||||||
>
|
>
|
||||||
<ha-svg-icon .path=${btn.icon || mdiLightbulb}></ha-svg-icon>
|
<ha-svg-icon .path=${btn.icon || mdiLightbulb}></ha-svg-icon>
|
||||||
</ha-control-button>
|
</ha-control-button>
|
||||||
@@ -162,7 +162,7 @@ export class DemoHaBarButton extends LitElement {
|
|||||||
}
|
}
|
||||||
.custom-group {
|
.custom-group {
|
||||||
--control-button-group-thickness: 100px;
|
--control-button-group-thickness: 100px;
|
||||||
--control-button-group-border-radius: 36px;
|
--control-button-group-border-radius: 18px;
|
||||||
--control-button-group-spacing: 20px;
|
--control-button-group-spacing: 20px;
|
||||||
}
|
}
|
||||||
.custom-group ha-control-button {
|
.custom-group ha-control-button {
|
||||||
|
@@ -49,11 +49,11 @@ export class DemoHaCircularSlider extends LitElement {
|
|||||||
<div class="field">
|
<div class="field">
|
||||||
<p>Current</p>
|
<p>Current</p>
|
||||||
<ha-slider
|
<ha-slider
|
||||||
labeled
|
|
||||||
min="10"
|
min="10"
|
||||||
max="30"
|
max="30"
|
||||||
.value=${this.current}
|
.value=${this.current}
|
||||||
@change=${this._currentChanged}
|
@change=${this._currentChanged}
|
||||||
|
pin
|
||||||
></ha-slider>
|
></ha-slider>
|
||||||
<p>${this.current} °C</p>
|
<p>${this.current} °C</p>
|
||||||
</div>
|
</div>
|
||||||
|
@@ -11,7 +11,6 @@ const buttons: {
|
|||||||
min?: number;
|
min?: number;
|
||||||
max?: number;
|
max?: number;
|
||||||
step?: number;
|
step?: number;
|
||||||
unit?: string;
|
|
||||||
class?: string;
|
class?: string;
|
||||||
}[] = [
|
}[] = [
|
||||||
{
|
{
|
||||||
@@ -30,11 +29,6 @@ const buttons: {
|
|||||||
label: "Custom",
|
label: "Custom",
|
||||||
class: "custom",
|
class: "custom",
|
||||||
},
|
},
|
||||||
{
|
|
||||||
id: "unit",
|
|
||||||
label: "With unit",
|
|
||||||
unit: "m",
|
|
||||||
},
|
|
||||||
];
|
];
|
||||||
|
|
||||||
@customElement("demo-components-ha-control-number-buttons")
|
@customElement("demo-components-ha-control-number-buttons")
|
||||||
@@ -56,7 +50,6 @@ export class DemoHarControlNumberButtons extends LitElement {
|
|||||||
<pre>Config: ${JSON.stringify(config)}</pre>
|
<pre>Config: ${JSON.stringify(config)}</pre>
|
||||||
<ha-control-number-buttons
|
<ha-control-number-buttons
|
||||||
.value=${this.value}
|
.value=${this.value}
|
||||||
.unit=${config.unit}
|
|
||||||
.min=${config.min}
|
.min=${config.min}
|
||||||
.max=${config.max}
|
.max=${config.max}
|
||||||
.step=${config.step}
|
.step=${config.step}
|
||||||
@@ -94,7 +87,7 @@ export class DemoHarControlNumberButtons extends LitElement {
|
|||||||
--control-number-buttons-background-color: #2196f3;
|
--control-number-buttons-background-color: #2196f3;
|
||||||
--control-number-buttons-background-opacity: 0.1;
|
--control-number-buttons-background-opacity: 0.1;
|
||||||
--control-number-buttons-thickness: 100px;
|
--control-number-buttons-thickness: 100px;
|
||||||
--control-number-buttons-border-radius: 36px;
|
--control-number-buttons-border-radius: 24px;
|
||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
@@ -135,7 +135,7 @@ export class DemoHaControlSelect extends LitElement {
|
|||||||
class=${ifDefined(config.class)}
|
class=${ifDefined(config.class)}
|
||||||
@value-changed=${this.handleValueChanged}
|
@value-changed=${this.handleValueChanged}
|
||||||
aria-labelledby=${id}
|
aria-labelledby=${id}
|
||||||
?disabled=${config.disabled}
|
disabled=${ifDefined(config.disabled)}
|
||||||
>
|
>
|
||||||
</ha-control-select>
|
</ha-control-select>
|
||||||
</div>
|
</div>
|
||||||
@@ -156,7 +156,7 @@ export class DemoHaControlSelect extends LitElement {
|
|||||||
class=${ifDefined(config.class)}
|
class=${ifDefined(config.class)}
|
||||||
@value-changed=${this.handleValueChanged}
|
@value-changed=${this.handleValueChanged}
|
||||||
aria-labelledby=${id}
|
aria-labelledby=${id}
|
||||||
?disabled=${config.disabled}
|
disabled=${ifDefined(config.disabled)}
|
||||||
>
|
>
|
||||||
</ha-control-select>
|
</ha-control-select>
|
||||||
`;
|
`;
|
||||||
@@ -186,8 +186,8 @@ export class DemoHaControlSelect extends LitElement {
|
|||||||
.custom {
|
.custom {
|
||||||
--mdc-icon-size: 24px;
|
--mdc-icon-size: 24px;
|
||||||
--control-select-color: var(--state-fan-active-color);
|
--control-select-color: var(--state-fan-active-color);
|
||||||
--control-select-thickness: 130px;
|
--control-select-thickness: 100px;
|
||||||
--control-select-border-radius: 48px;
|
--control-select-border-radius: 24px;
|
||||||
}
|
}
|
||||||
.vertical-selects {
|
.vertical-selects {
|
||||||
height: 300px;
|
height: 300px;
|
||||||
|
@@ -9,7 +9,6 @@ const sliders: {
|
|||||||
id: string;
|
id: string;
|
||||||
label: string;
|
label: string;
|
||||||
mode?: "start" | "end" | "cursor";
|
mode?: "start" | "end" | "cursor";
|
||||||
unit?: string;
|
|
||||||
class?: string;
|
class?: string;
|
||||||
}[] = [
|
}[] = [
|
||||||
{
|
{
|
||||||
@@ -32,21 +31,18 @@ const sliders: {
|
|||||||
label: "Slider (start mode) and custom style",
|
label: "Slider (start mode) and custom style",
|
||||||
mode: "start",
|
mode: "start",
|
||||||
class: "custom",
|
class: "custom",
|
||||||
unit: "mm",
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "slider-end-custom",
|
id: "slider-end-custom",
|
||||||
label: "Slider (end mode) and custom style",
|
label: "Slider (end mode) and custom style",
|
||||||
mode: "end",
|
mode: "end",
|
||||||
class: "custom",
|
class: "custom",
|
||||||
unit: "mm",
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "slider-cursor-custom",
|
id: "slider-cursor-custom",
|
||||||
label: "Slider (cursor mode) and custom style",
|
label: "Slider (cursor mode) and custom style",
|
||||||
mode: "cursor",
|
mode: "cursor",
|
||||||
class: "custom",
|
class: "custom",
|
||||||
unit: "mm",
|
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -97,7 +93,6 @@ export class DemoHaBarSlider extends LitElement {
|
|||||||
@value-changed=${this.handleValueChanged}
|
@value-changed=${this.handleValueChanged}
|
||||||
@slider-moved=${this.handleSliderMoved}
|
@slider-moved=${this.handleSliderMoved}
|
||||||
aria-labelledby=${id}
|
aria-labelledby=${id}
|
||||||
.unit=${config.unit}
|
|
||||||
>
|
>
|
||||||
</ha-control-slider>
|
</ha-control-slider>
|
||||||
</div>
|
</div>
|
||||||
@@ -119,7 +114,6 @@ export class DemoHaBarSlider extends LitElement {
|
|||||||
@value-changed=${this.handleValueChanged}
|
@value-changed=${this.handleValueChanged}
|
||||||
@slider-moved=${this.handleSliderMoved}
|
@slider-moved=${this.handleSliderMoved}
|
||||||
aria-label=${label}
|
aria-label=${label}
|
||||||
.unit=${config.unit}
|
|
||||||
>
|
>
|
||||||
</ha-control-slider>
|
</ha-control-slider>
|
||||||
`;
|
`;
|
||||||
@@ -150,8 +144,8 @@ export class DemoHaBarSlider extends LitElement {
|
|||||||
--control-slider-color: #ffcf4c;
|
--control-slider-color: #ffcf4c;
|
||||||
--control-slider-background: #ffcf4c;
|
--control-slider-background: #ffcf4c;
|
||||||
--control-slider-background-opacity: 0.2;
|
--control-slider-background-opacity: 0.2;
|
||||||
--control-slider-thickness: 130px;
|
--control-slider-thickness: 100px;
|
||||||
--control-slider-border-radius: 48px;
|
--control-slider-border-radius: 24px;
|
||||||
}
|
}
|
||||||
.vertical-sliders {
|
.vertical-sliders {
|
||||||
height: 300px;
|
height: 300px;
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user