mirror of
https://github.com/home-assistant/frontend.git
synced 2025-08-24 08:39:26 +00:00
Compare commits
190 Commits
compress_d
...
20230329.0
Author | SHA1 | Date | |
---|---|---|---|
![]() |
a4e36d6145 | ||
![]() |
bdfdab439a | ||
![]() |
8c59537032 | ||
![]() |
f5a4affdec | ||
![]() |
f19fdeacba | ||
![]() |
e0f7544d2f | ||
![]() |
4d2d7cd125 | ||
![]() |
0f5320c6fb | ||
![]() |
4d52913a01 | ||
![]() |
0f97a76428 | ||
![]() |
f2cf598f98 | ||
![]() |
a6f9482bf6 | ||
![]() |
48c74c8660 | ||
![]() |
3a700aebcc | ||
![]() |
cd6aac85d2 | ||
![]() |
77b227a7d1 | ||
![]() |
b785fedef2 | ||
![]() |
66a202be7e | ||
![]() |
0ad02013ba | ||
![]() |
1b2eaedba0 | ||
![]() |
8ea350a488 | ||
![]() |
8ff56bd8f5 | ||
![]() |
dd7ec07f29 | ||
![]() |
520f489830 | ||
![]() |
395358b192 | ||
![]() |
a19ff5aef5 | ||
![]() |
34f8b48fbe | ||
![]() |
01f8b4e1c4 | ||
![]() |
9e1cdf8215 | ||
![]() |
551127b844 | ||
![]() |
afe42629b2 | ||
![]() |
848d12e6fe | ||
![]() |
65bd373af4 | ||
![]() |
91c099632d | ||
![]() |
9053bc7b78 | ||
![]() |
9c4b0259a8 | ||
![]() |
7930f3879d | ||
![]() |
d9dbb69e62 | ||
![]() |
74cfccaac7 | ||
![]() |
4ba7e5cf0f | ||
![]() |
fc76d8f1cf | ||
![]() |
3eb07e9bc3 | ||
![]() |
4e6ed61e2b | ||
![]() |
ca6d1544d1 | ||
![]() |
81e9bc894b | ||
![]() |
173d13ae66 | ||
![]() |
886ae791bc | ||
![]() |
50b7e72688 | ||
![]() |
d07ae1cf48 | ||
![]() |
da2de3c7d2 | ||
![]() |
7c62b08fdd | ||
![]() |
4f5fca7c60 | ||
![]() |
45c153d374 | ||
![]() |
cd2996734c | ||
![]() |
4abc2a65cb | ||
![]() |
89decd2f31 | ||
![]() |
04a16812d3 | ||
![]() |
cadbc501e2 | ||
![]() |
0d95d856c1 | ||
![]() |
d272783258 | ||
![]() |
65d3af6fd6 | ||
![]() |
24c3ddb96b | ||
![]() |
c9d709152a | ||
![]() |
d5bc892bae | ||
![]() |
a5ea7b33b0 | ||
![]() |
c29568d164 | ||
![]() |
57e1769c06 | ||
![]() |
abd2070011 | ||
![]() |
840450d9e5 | ||
![]() |
178163cbc7 | ||
![]() |
dcca02477a | ||
![]() |
c06990f309 | ||
![]() |
1c9e3915e8 | ||
![]() |
30b8dc258a | ||
![]() |
f31043cfdc | ||
![]() |
c0c83d3721 | ||
![]() |
dd08909fef | ||
![]() |
952028a7be | ||
![]() |
b0f3006c4b | ||
![]() |
27098c5f3f | ||
![]() |
fec061f5d1 | ||
![]() |
db08c5029b | ||
![]() |
c5be2acd46 | ||
![]() |
0232c11bc2 | ||
![]() |
dfd7acd713 | ||
![]() |
24f1677809 | ||
![]() |
7b2afa3df9 | ||
![]() |
13a3db4141 | ||
![]() |
aeb7f8ff36 | ||
![]() |
db62e9f922 | ||
![]() |
1861547d9b | ||
![]() |
43684795a4 | ||
![]() |
0158d7e3e5 | ||
![]() |
77dcace95e | ||
![]() |
60d106d9c3 | ||
![]() |
020aab0584 | ||
![]() |
27808c9853 | ||
![]() |
1a9b9da0db | ||
![]() |
fe19963ea9 | ||
![]() |
3afe1f83c7 | ||
![]() |
6cb4b5f429 | ||
![]() |
7b676cbd91 | ||
![]() |
bf1cabca6e | ||
![]() |
effb4b9f7a | ||
![]() |
1bc7bb0169 | ||
![]() |
3abc5c42d0 | ||
![]() |
cc011a4d47 | ||
![]() |
ca0338436c | ||
![]() |
84d31cb9d5 | ||
![]() |
69264b3448 | ||
![]() |
01a098e5aa | ||
![]() |
11b1e056dc | ||
![]() |
4e99e32ca2 | ||
![]() |
a8d7f8a7fe | ||
![]() |
9602c68640 | ||
![]() |
9a0699ba84 | ||
![]() |
47cfc3d27c | ||
![]() |
77d395b0c9 | ||
![]() |
baee563bdd | ||
![]() |
bc5345af95 | ||
![]() |
52e2033f52 | ||
![]() |
dfa26ee7a1 | ||
![]() |
0cde0fc9a9 | ||
![]() |
4ab1723c99 | ||
![]() |
4f1c4bdcb9 | ||
![]() |
55edeb474e | ||
![]() |
4d19e3ad63 | ||
![]() |
0b3e954752 | ||
![]() |
f193563649 | ||
![]() |
77c3b5b5a5 | ||
![]() |
9bb36e38e6 | ||
![]() |
1741b051fc | ||
![]() |
98e21370fd | ||
![]() |
40b9e62a87 | ||
![]() |
8e8fb4ea13 | ||
![]() |
6dc7508460 | ||
![]() |
9055099a18 | ||
![]() |
6b4e8c1e06 | ||
![]() |
cbeee01d75 | ||
![]() |
76bdf96c6d | ||
![]() |
645eea0bd4 | ||
![]() |
7453ea1e40 | ||
![]() |
b64fc8ec3b | ||
![]() |
3c73fcb0bf | ||
![]() |
c5c668bd7b | ||
![]() |
be909d0a9c | ||
![]() |
19efe9dcdd | ||
![]() |
a57609380a | ||
![]() |
84affcce33 | ||
![]() |
d113d44d5a | ||
![]() |
afe1b7ef59 | ||
![]() |
000e2ef7fc | ||
![]() |
15394516af | ||
![]() |
a0033d9112 | ||
![]() |
895ebcdb2d | ||
![]() |
e7044ce40c | ||
![]() |
a0e71b9478 | ||
![]() |
db1f81e0ef | ||
![]() |
657a74e8ad | ||
![]() |
f803bbadc9 | ||
![]() |
2ab8bba4b0 | ||
![]() |
1a1c00ba62 | ||
![]() |
7b4850b5d6 | ||
![]() |
42688e100c | ||
![]() |
d32cde3cf8 | ||
![]() |
34c15d96c1 | ||
![]() |
0d869d53e3 | ||
![]() |
ccf670465b | ||
![]() |
4a9ec7233d | ||
![]() |
c747ab7605 | ||
![]() |
9c703ab469 | ||
![]() |
e3b797e85c | ||
![]() |
a5541996d7 | ||
![]() |
ea95fb98aa | ||
![]() |
4b428a60cd | ||
![]() |
9ce4563dd4 | ||
![]() |
4031d9cc78 | ||
![]() |
20f2f5b317 | ||
![]() |
6af57fa2cd | ||
![]() |
17e6e132d1 | ||
![]() |
67d064db28 | ||
![]() |
e0df5e4631 | ||
![]() |
a8a85a2af6 | ||
![]() |
702c24d908 | ||
![]() |
df8f46388f | ||
![]() |
dd6437376d | ||
![]() |
971d2ff1c2 | ||
![]() |
e6de8ec94d | ||
![]() |
0e06267055 | ||
![]() |
63a35c9d68 |
@@ -119,7 +119,6 @@
|
|||||||
"lit/no-template-map": "off",
|
"lit/no-template-map": "off",
|
||||||
"lit/no-native-attributes": "warn",
|
"lit/no-native-attributes": "warn",
|
||||||
"lit/no-this-assign-in-render": "warn",
|
"lit/no-this-assign-in-render": "warn",
|
||||||
"lit/prefer-nothing": "warn",
|
|
||||||
"lit-a11y/click-events-have-key-events": ["off"],
|
"lit-a11y/click-events-have-key-events": ["off"],
|
||||||
"lit-a11y/no-autofocus": "off",
|
"lit-a11y/no-autofocus": "off",
|
||||||
"lit-a11y/alt-text": "warn",
|
"lit-a11y/alt-text": "warn",
|
||||||
|
13
.github/dependabot.yml
vendored
13
.github/dependabot.yml
vendored
@@ -6,16 +6,3 @@ updates:
|
|||||||
interval: weekly
|
interval: weekly
|
||||||
time: "06:00"
|
time: "06:00"
|
||||||
open-pull-requests-limit: 10
|
open-pull-requests-limit: 10
|
||||||
- package-ecosystem: "npm"
|
|
||||||
directory: "/"
|
|
||||||
schedule:
|
|
||||||
interval: "daily"
|
|
||||||
time: "03:00"
|
|
||||||
open-pull-requests-limit: 10
|
|
||||||
labels:
|
|
||||||
- "dependencies"
|
|
||||||
ignore:
|
|
||||||
# Ignore rollup and plugins until everything else is updated
|
|
||||||
- dependency-name: "*rollup*"
|
|
||||||
- dependency-name: "@rollup/*"
|
|
||||||
- dependency-name: "serve"
|
|
||||||
|
4
.github/workflows/cast_deployment.yaml
vendored
4
.github/workflows/cast_deployment.yaml
vendored
@@ -22,7 +22,7 @@ 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@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
with:
|
with:
|
||||||
ref: dev
|
ref: dev
|
||||||
|
|
||||||
@@ -58,7 +58,7 @@ 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@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
with:
|
with:
|
||||||
ref: master
|
ref: master
|
||||||
|
|
||||||
|
8
.github/workflows/ci.yaml
vendored
8
.github/workflows/ci.yaml
vendored
@@ -25,7 +25,7 @@ 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@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
- name: Set up Node ${{ env.NODE_VERSION }}
|
- name: Set up Node ${{ env.NODE_VERSION }}
|
||||||
uses: actions/setup-node@v3.6.0
|
uses: actions/setup-node@v3.6.0
|
||||||
with:
|
with:
|
||||||
@@ -48,7 +48,7 @@ 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@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
- name: Set up Node ${{ env.NODE_VERSION }}
|
- name: Set up Node ${{ env.NODE_VERSION }}
|
||||||
uses: actions/setup-node@v3.6.0
|
uses: actions/setup-node@v3.6.0
|
||||||
with:
|
with:
|
||||||
@@ -66,7 +66,7 @@ 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@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
- name: Set up Node ${{ env.NODE_VERSION }}
|
- name: Set up Node ${{ env.NODE_VERSION }}
|
||||||
uses: actions/setup-node@v3.6.0
|
uses: actions/setup-node@v3.6.0
|
||||||
with:
|
with:
|
||||||
@@ -84,7 +84,7 @@ 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@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
- name: Set up Node ${{ env.NODE_VERSION }}
|
- name: Set up Node ${{ env.NODE_VERSION }}
|
||||||
uses: actions/setup-node@v3.6.0
|
uses: actions/setup-node@v3.6.0
|
||||||
with:
|
with:
|
||||||
|
2
.github/workflows/codeql-analysis.yml
vendored
2
.github/workflows/codeql-analysis.yml
vendored
@@ -23,7 +23,7 @@ jobs:
|
|||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@v3.3.0
|
uses: actions/checkout@v3.5.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.
|
||||||
|
4
.github/workflows/demo_deployment.yaml
vendored
4
.github/workflows/demo_deployment.yaml
vendored
@@ -23,7 +23,7 @@ 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@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
with:
|
with:
|
||||||
ref: dev
|
ref: dev
|
||||||
|
|
||||||
@@ -59,7 +59,7 @@ 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@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
with:
|
with:
|
||||||
ref: master
|
ref: master
|
||||||
|
|
||||||
|
2
.github/workflows/design_deployment.yaml
vendored
2
.github/workflows/design_deployment.yaml
vendored
@@ -17,7 +17,7 @@ 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@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
|
|
||||||
- name: Set up Node ${{ env.NODE_VERSION }}
|
- name: Set up Node ${{ env.NODE_VERSION }}
|
||||||
uses: actions/setup-node@v3.6.0
|
uses: actions/setup-node@v3.6.0
|
||||||
|
2
.github/workflows/design_preview.yaml
vendored
2
.github/workflows/design_preview.yaml
vendored
@@ -22,7 +22,7 @@ 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@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
|
|
||||||
- name: Set up Node ${{ env.NODE_VERSION }}
|
- name: Set up Node ${{ env.NODE_VERSION }}
|
||||||
uses: actions/setup-node@v3.6.0
|
uses: actions/setup-node@v3.6.0
|
||||||
|
2
.github/workflows/nightly.yaml
vendored
2
.github/workflows/nightly.yaml
vendored
@@ -21,7 +21,7 @@ jobs:
|
|||||||
contents: write
|
contents: write
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
|
|
||||||
- name: Set up Python ${{ env.PYTHON_VERSION }}
|
- name: Set up Python ${{ env.PYTHON_VERSION }}
|
||||||
uses: actions/setup-python@v4
|
uses: actions/setup-python@v4
|
||||||
|
2
.github/workflows/release.yaml
vendored
2
.github/workflows/release.yaml
vendored
@@ -24,7 +24,7 @@ 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@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
|
|
||||||
- name: Verify version
|
- name: Verify version
|
||||||
uses: home-assistant/actions/helpers/verify-version@master
|
uses: home-assistant/actions/helpers/verify-version@master
|
||||||
|
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@v7.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
@@ -16,7 +16,7 @@ jobs:
|
|||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout the repository
|
- name: Checkout the repository
|
||||||
uses: actions/checkout@v3.3.0
|
uses: actions/checkout@v3.5.0
|
||||||
|
|
||||||
- name: Upload Translations
|
- name: Upload Translations
|
||||||
run: |
|
run: |
|
||||||
|
823
.yarn/releases/yarn-3.3.1.cjs
vendored
823
.yarn/releases/yarn-3.3.1.cjs
vendored
File diff suppressed because one or more lines are too long
873
.yarn/releases/yarn-3.5.0.cjs
vendored
Executable file
873
.yarn/releases/yarn-3.5.0.cjs
vendored
Executable file
File diff suppressed because one or more lines are too long
@@ -1,3 +1,5 @@
|
|||||||
|
defaultSemverRangePrefix: ""
|
||||||
|
|
||||||
nodeLinker: node-modules
|
nodeLinker: node-modules
|
||||||
|
|
||||||
plugins:
|
plugins:
|
||||||
@@ -6,4 +8,4 @@ plugins:
|
|||||||
- path: .yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
|
- path: .yarn/plugins/@yarnpkg/plugin-interactive-tools.cjs
|
||||||
spec: "@yarnpkg/plugin-interactive-tools"
|
spec: "@yarnpkg/plugin-interactive-tools"
|
||||||
|
|
||||||
yarnPath: .yarn/releases/yarn-3.3.1.cjs
|
yarnPath: .yarn/releases/yarn-3.5.0.cjs
|
||||||
|
@@ -2,6 +2,15 @@ const path = require("path");
|
|||||||
const env = require("./env.js");
|
const env = require("./env.js");
|
||||||
const paths = require("./paths.js");
|
const paths = require("./paths.js");
|
||||||
|
|
||||||
|
// 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
|
||||||
|
module.exports.sourceMapURL = () => {
|
||||||
|
const ref = env.version().endsWith("dev")
|
||||||
|
? process.env.GITHUB_SHA || "dev"
|
||||||
|
: env.version();
|
||||||
|
return `https://raw.githubusercontent.com/home-assistant/frontend/${ref}`;
|
||||||
|
};
|
||||||
|
|
||||||
// Files from NPM Packages that should not be imported
|
// Files from NPM Packages that should not be imported
|
||||||
// eslint-disable-next-line unused-imports/no-unused-vars
|
// eslint-disable-next-line unused-imports/no-unused-vars
|
||||||
module.exports.ignorePackages = ({ latestBuild }) => [
|
module.exports.ignorePackages = ({ latestBuild }) => [
|
||||||
@@ -53,13 +62,26 @@ module.exports.definedVars = ({ isProdBuild, latestBuild, defineOverlay }) => ({
|
|||||||
...defineOverlay,
|
...defineOverlay,
|
||||||
});
|
});
|
||||||
|
|
||||||
module.exports.terserOptions = (latestBuild) => ({
|
module.exports.htmlMinifierOptions = {
|
||||||
|
caseSensitive: true,
|
||||||
|
collapseWhitespace: true,
|
||||||
|
conservativeCollapse: true,
|
||||||
|
decodeEntities: true,
|
||||||
|
removeComments: true,
|
||||||
|
removeRedundantAttributes: true,
|
||||||
|
minifyCSS: {
|
||||||
|
compatibility: "*,-properties.zeroUnits",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports.terserOptions = ({ latestBuild, isTestBuild }) => ({
|
||||||
safari10: !latestBuild,
|
safari10: !latestBuild,
|
||||||
ecma: latestBuild ? undefined : 5,
|
ecma: latestBuild ? undefined : 5,
|
||||||
output: { comments: false },
|
format: { comments: false },
|
||||||
|
sourceMap: !isTestBuild,
|
||||||
});
|
});
|
||||||
|
|
||||||
module.exports.babelOptions = ({ latestBuild }) => ({
|
module.exports.babelOptions = ({ latestBuild, isProdBuild, isTestBuild }) => ({
|
||||||
babelrc: false,
|
babelrc: false,
|
||||||
compact: false,
|
compact: false,
|
||||||
presets: [
|
presets: [
|
||||||
@@ -67,7 +89,7 @@ module.exports.babelOptions = ({ latestBuild }) => ({
|
|||||||
"@babel/preset-env",
|
"@babel/preset-env",
|
||||||
{
|
{
|
||||||
useBuiltIns: "entry",
|
useBuiltIns: "entry",
|
||||||
corejs: { version: "3.28", proposals: true },
|
corejs: { version: "3.29", proposals: true },
|
||||||
bugfixes: true,
|
bugfixes: true,
|
||||||
},
|
},
|
||||||
],
|
],
|
||||||
@@ -93,20 +115,41 @@ module.exports.babelOptions = ({ latestBuild }) => ({
|
|||||||
"@babel/plugin-syntax-import-meta",
|
"@babel/plugin-syntax-import-meta",
|
||||||
"@babel/plugin-syntax-dynamic-import",
|
"@babel/plugin-syntax-dynamic-import",
|
||||||
"@babel/plugin-syntax-top-level-await",
|
"@babel/plugin-syntax-top-level-await",
|
||||||
|
// Support various proposals
|
||||||
"@babel/plugin-proposal-optional-chaining",
|
"@babel/plugin-proposal-optional-chaining",
|
||||||
"@babel/plugin-proposal-nullish-coalescing-operator",
|
"@babel/plugin-proposal-nullish-coalescing-operator",
|
||||||
["@babel/plugin-proposal-decorators", { decoratorsBeforeExport: true }],
|
["@babel/plugin-proposal-decorators", { decoratorsBeforeExport: true }],
|
||||||
["@babel/plugin-proposal-private-methods", { loose: true }],
|
["@babel/plugin-proposal-private-methods", { loose: true }],
|
||||||
["@babel/plugin-proposal-private-property-in-object", { loose: true }],
|
["@babel/plugin-proposal-private-property-in-object", { loose: true }],
|
||||||
["@babel/plugin-proposal-class-properties", { loose: true }],
|
["@babel/plugin-proposal-class-properties", { loose: true }],
|
||||||
|
// Minify template literals for production
|
||||||
|
isProdBuild && [
|
||||||
|
"template-html-minifier",
|
||||||
|
{
|
||||||
|
modules: {
|
||||||
|
lit: [
|
||||||
|
"html",
|
||||||
|
{ name: "svg", encapsulation: "svg" },
|
||||||
|
{ name: "css", encapsulation: "style" },
|
||||||
|
],
|
||||||
|
"@polymer/polymer/lib/utils/html-tag": ["html"],
|
||||||
|
},
|
||||||
|
strictCSS: true,
|
||||||
|
htmlMinifier: module.exports.htmlMinifierOptions,
|
||||||
|
failOnError: true, // we can turn this off in case of false positives
|
||||||
|
},
|
||||||
|
],
|
||||||
].filter(Boolean),
|
].filter(Boolean),
|
||||||
exclude: [
|
exclude: [
|
||||||
// \\ for Windows, / for Mac OS and Linux
|
// \\ for Windows, / for Mac OS and Linux
|
||||||
/node_modules[\\/]core-js/,
|
/node_modules[\\/]core-js/,
|
||||||
/node_modules[\\/]webpack[\\/]buildin/,
|
/node_modules[\\/]webpack[\\/]buildin/,
|
||||||
],
|
],
|
||||||
|
sourceMaps: !isTestBuild,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
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");
|
||||||
|
|
||||||
@@ -129,14 +172,17 @@ BundleConfig {
|
|||||||
latestBuild: boolean,
|
latestBuild: boolean,
|
||||||
// If we're doing a stats build (create nice chunk names)
|
// If we're doing a stats build (create nice chunk names)
|
||||||
isStatsBuild: boolean,
|
isStatsBuild: boolean,
|
||||||
|
// If it's just a test build in CI, skip time on source map generation
|
||||||
|
isTestBuild: boolean,
|
||||||
// Names of entrypoints that should not be hashed
|
// Names of entrypoints that should not be hashed
|
||||||
dontHash: Set<string>
|
dontHash: Set<string>
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
|
|
||||||
module.exports.config = {
|
module.exports.config = {
|
||||||
app({ isProdBuild, latestBuild, isStatsBuild, isWDS }) {
|
app({ isProdBuild, latestBuild, isStatsBuild, isTestBuild, isWDS }) {
|
||||||
return {
|
return {
|
||||||
|
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",
|
||||||
@@ -150,12 +196,14 @@ module.exports.config = {
|
|||||||
isProdBuild,
|
isProdBuild,
|
||||||
latestBuild,
|
latestBuild,
|
||||||
isStatsBuild,
|
isStatsBuild,
|
||||||
|
isTestBuild,
|
||||||
isWDS,
|
isWDS,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
demo({ isProdBuild, latestBuild, isStatsBuild }) {
|
demo({ isProdBuild, latestBuild, isStatsBuild }) {
|
||||||
return {
|
return {
|
||||||
|
name: "demo" + nameSuffix(latestBuild),
|
||||||
entry: {
|
entry: {
|
||||||
main: path.resolve(paths.demo_dir, "src/entrypoint.ts"),
|
main: path.resolve(paths.demo_dir, "src/entrypoint.ts"),
|
||||||
},
|
},
|
||||||
@@ -185,6 +233,7 @@ module.exports.config = {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
|
name: "cast" + nameSuffix(latestBuild),
|
||||||
entry,
|
entry,
|
||||||
outputPath: outputPath(paths.cast_output_root, latestBuild),
|
outputPath: outputPath(paths.cast_output_root, latestBuild),
|
||||||
publicPath: publicPath(latestBuild),
|
publicPath: publicPath(latestBuild),
|
||||||
@@ -196,8 +245,9 @@ module.exports.config = {
|
|||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
hassio({ isProdBuild, latestBuild }) {
|
hassio({ isProdBuild, latestBuild, isStatsBuild, isTestBuild }) {
|
||||||
return {
|
return {
|
||||||
|
name: "supervisor" + nameSuffix(latestBuild),
|
||||||
entry: {
|
entry: {
|
||||||
entrypoint: path.resolve(paths.hassio_dir, "src/entrypoint.ts"),
|
entrypoint: path.resolve(paths.hassio_dir, "src/entrypoint.ts"),
|
||||||
},
|
},
|
||||||
@@ -205,6 +255,8 @@ module.exports.config = {
|
|||||||
publicPath: publicPath(latestBuild, paths.hassio_publicPath),
|
publicPath: publicPath(latestBuild, paths.hassio_publicPath),
|
||||||
isProdBuild,
|
isProdBuild,
|
||||||
latestBuild,
|
latestBuild,
|
||||||
|
isStatsBuild,
|
||||||
|
isTestBuild,
|
||||||
isHassioBuild: true,
|
isHassioBuild: true,
|
||||||
defineOverlay: {
|
defineOverlay: {
|
||||||
__SUPERVISOR__: true,
|
__SUPERVISOR__: true,
|
||||||
@@ -214,6 +266,7 @@ module.exports.config = {
|
|||||||
|
|
||||||
gallery({ isProdBuild, latestBuild }) {
|
gallery({ isProdBuild, latestBuild }) {
|
||||||
return {
|
return {
|
||||||
|
name: "gallery" + nameSuffix(latestBuild),
|
||||||
entry: {
|
entry: {
|
||||||
entrypoint: path.resolve(paths.gallery_dir, "src/entrypoint.js"),
|
entrypoint: path.resolve(paths.gallery_dir, "src/entrypoint.js"),
|
||||||
},
|
},
|
||||||
|
@@ -17,7 +17,7 @@ module.exports = {
|
|||||||
isStatsBuild() {
|
isStatsBuild() {
|
||||||
return process.env.STATS === "1";
|
return process.env.STATS === "1";
|
||||||
},
|
},
|
||||||
isTest() {
|
isTestBuild() {
|
||||||
return process.env.IS_TEST === "true";
|
return process.env.IS_TEST === "true";
|
||||||
},
|
},
|
||||||
isNetlify() {
|
isNetlify() {
|
||||||
|
@@ -1,8 +1,7 @@
|
|||||||
// Run HA develop mode
|
// Run HA develop mode
|
||||||
|
|
||||||
const gulp = require("gulp");
|
const gulp = require("gulp");
|
||||||
|
|
||||||
const env = require("../env");
|
const env = require("../env");
|
||||||
|
|
||||||
require("./clean.js");
|
require("./clean.js");
|
||||||
require("./translations.js");
|
require("./translations.js");
|
||||||
require("./locale-data.js");
|
require("./locale-data.js");
|
||||||
@@ -50,7 +49,7 @@ gulp.task(
|
|||||||
"copy-static-app",
|
"copy-static-app",
|
||||||
env.useRollup() ? "rollup-prod-app" : "webpack-prod-app",
|
env.useRollup() ? "rollup-prod-app" : "webpack-prod-app",
|
||||||
// Don't compress running tests
|
// Don't compress running tests
|
||||||
...(env.isTest() ? [] : ["compress-app"]),
|
...(env.isTestBuild() ? [] : ["compress-app"]),
|
||||||
gulp.parallel(
|
gulp.parallel(
|
||||||
"gen-pages-prod",
|
"gen-pages-prod",
|
||||||
"gen-index-app-prod",
|
"gen-index-app-prod",
|
||||||
|
@@ -3,9 +3,10 @@ const gulp = require("gulp");
|
|||||||
const fs = require("fs-extra");
|
const fs = require("fs-extra");
|
||||||
const path = require("path");
|
const path = require("path");
|
||||||
const template = require("lodash.template");
|
const template = require("lodash.template");
|
||||||
const minify = require("html-minifier").minify;
|
const { minify } = require("html-minifier-terser");
|
||||||
const paths = require("../paths.js");
|
const paths = require("../paths.js");
|
||||||
const env = require("../env.js");
|
const env = require("../env.js");
|
||||||
|
const { htmlMinifierOptions, terserOptions } = require("../bundle.js");
|
||||||
|
|
||||||
const templatePath = (tpl) =>
|
const templatePath = (tpl) =>
|
||||||
path.resolve(paths.polymer_dir, "src/html/", `${tpl}.html.template`);
|
path.resolve(paths.polymer_dir, "src/html/", `${tpl}.html.template`);
|
||||||
@@ -39,10 +40,12 @@ const renderGalleryTemplate = (pth, data = {}) =>
|
|||||||
|
|
||||||
const minifyHtml = (content) =>
|
const minifyHtml = (content) =>
|
||||||
minify(content, {
|
minify(content, {
|
||||||
collapseWhitespace: true,
|
...htmlMinifierOptions,
|
||||||
minifyJS: true,
|
conservativeCollapse: false,
|
||||||
minifyCSS: true,
|
minifyJS: terserOptions({
|
||||||
removeComments: true,
|
latestBuild: false, // Shared scripts should be ES5
|
||||||
|
isTestBuild: true, // Don't need source maps
|
||||||
|
}),
|
||||||
});
|
});
|
||||||
|
|
||||||
const PAGES = ["onboarding", "authorize"];
|
const PAGES = ["onboarding", "authorize"];
|
||||||
@@ -63,7 +66,7 @@ gulp.task("gen-pages-dev", (done) => {
|
|||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task("gen-pages-prod", (done) => {
|
gulp.task("gen-pages-prod", async () => {
|
||||||
const latestManifest = require(path.resolve(
|
const latestManifest = require(path.resolve(
|
||||||
paths.app_output_latest,
|
paths.app_output_latest,
|
||||||
"manifest.json"
|
"manifest.json"
|
||||||
@@ -73,19 +76,23 @@ gulp.task("gen-pages-prod", (done) => {
|
|||||||
"manifest.json"
|
"manifest.json"
|
||||||
));
|
));
|
||||||
|
|
||||||
|
const minifiedHTML = [];
|
||||||
for (const page of PAGES) {
|
for (const page of PAGES) {
|
||||||
const content = renderTemplate(page, {
|
const content = renderTemplate(page, {
|
||||||
latestPageJS: latestManifest[`${page}.js`],
|
latestPageJS: latestManifest[`${page}.js`],
|
||||||
|
|
||||||
es5PageJS: es5Manifest[`${page}.js`],
|
es5PageJS: es5Manifest[`${page}.js`],
|
||||||
});
|
});
|
||||||
|
|
||||||
fs.outputFileSync(
|
minifiedHTML.push(
|
||||||
path.resolve(paths.app_output_root, `${page}.html`),
|
minifyHtml(content).then((minified) =>
|
||||||
minifyHtml(content)
|
fs.outputFileSync(
|
||||||
|
path.resolve(paths.app_output_root, `${page}.html`),
|
||||||
|
minified
|
||||||
|
)
|
||||||
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
done();
|
await Promise.all(minifiedHTML);
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task("gen-index-app-dev", (done) => {
|
gulp.task("gen-index-app-dev", (done) => {
|
||||||
@@ -118,7 +125,7 @@ gulp.task("gen-index-app-dev", (done) => {
|
|||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task("gen-index-app-prod", (done) => {
|
gulp.task("gen-index-app-prod", async () => {
|
||||||
const latestManifest = require(path.resolve(
|
const latestManifest = require(path.resolve(
|
||||||
paths.app_output_latest,
|
paths.app_output_latest,
|
||||||
"manifest.json"
|
"manifest.json"
|
||||||
@@ -136,13 +143,15 @@ gulp.task("gen-index-app-prod", (done) => {
|
|||||||
es5CoreJS: es5Manifest["core.js"],
|
es5CoreJS: es5Manifest["core.js"],
|
||||||
es5CustomPanelJS: es5Manifest["custom-panel.js"],
|
es5CustomPanelJS: es5Manifest["custom-panel.js"],
|
||||||
});
|
});
|
||||||
const minified = minifyHtml(content).replace(/#THEMEC/g, "{{ theme_color }}");
|
const minified = (await minifyHtml(content)).replace(
|
||||||
|
/#THEMEC/g,
|
||||||
|
"{{ theme_color }}"
|
||||||
|
);
|
||||||
|
|
||||||
fs.outputFileSync(
|
fs.outputFileSync(
|
||||||
path.resolve(paths.app_output_root, "index.html"),
|
path.resolve(paths.app_output_root, "index.html"),
|
||||||
minified
|
minified
|
||||||
);
|
);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task("gen-index-cast-dev", (done) => {
|
gulp.task("gen-index-cast-dev", (done) => {
|
||||||
@@ -244,7 +253,7 @@ gulp.task("gen-index-demo-dev", (done) => {
|
|||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task("gen-index-demo-prod", (done) => {
|
gulp.task("gen-index-demo-prod", async () => {
|
||||||
const latestManifest = require(path.resolve(
|
const latestManifest = require(path.resolve(
|
||||||
paths.demo_output_latest,
|
paths.demo_output_latest,
|
||||||
"manifest.json"
|
"manifest.json"
|
||||||
@@ -258,13 +267,12 @@ gulp.task("gen-index-demo-prod", (done) => {
|
|||||||
|
|
||||||
es5DemoJS: es5Manifest["main.js"],
|
es5DemoJS: es5Manifest["main.js"],
|
||||||
});
|
});
|
||||||
const minified = minifyHtml(content);
|
const minified = await minifyHtml(content);
|
||||||
|
|
||||||
fs.outputFileSync(
|
fs.outputFileSync(
|
||||||
path.resolve(paths.demo_output_root, "index.html"),
|
path.resolve(paths.demo_output_root, "index.html"),
|
||||||
minified
|
minified
|
||||||
);
|
);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task("gen-index-gallery-dev", (done) => {
|
gulp.task("gen-index-gallery-dev", (done) => {
|
||||||
@@ -279,7 +287,7 @@ gulp.task("gen-index-gallery-dev", (done) => {
|
|||||||
done();
|
done();
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task("gen-index-gallery-prod", (done) => {
|
gulp.task("gen-index-gallery-prod", async () => {
|
||||||
const latestManifest = require(path.resolve(
|
const latestManifest = require(path.resolve(
|
||||||
paths.gallery_output_latest,
|
paths.gallery_output_latest,
|
||||||
"manifest.json"
|
"manifest.json"
|
||||||
@@ -287,13 +295,12 @@ gulp.task("gen-index-gallery-prod", (done) => {
|
|||||||
const content = renderGalleryTemplate("index", {
|
const content = renderGalleryTemplate("index", {
|
||||||
latestGalleryJS: latestManifest["entrypoint.js"],
|
latestGalleryJS: latestManifest["entrypoint.js"],
|
||||||
});
|
});
|
||||||
const minified = minifyHtml(content);
|
const minified = await minifyHtml(content);
|
||||||
|
|
||||||
fs.outputFileSync(
|
fs.outputFileSync(
|
||||||
path.resolve(paths.gallery_output_root, "index.html"),
|
path.resolve(paths.gallery_output_root, "index.html"),
|
||||||
minified
|
minified
|
||||||
);
|
);
|
||||||
done();
|
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task("gen-index-hassio-dev", async () => {
|
gulp.task("gen-index-hassio-dev", async () => {
|
||||||
|
@@ -20,7 +20,7 @@ require("./rollup.js");
|
|||||||
|
|
||||||
gulp.task("gather-gallery-pages", async function gatherPages() {
|
gulp.task("gather-gallery-pages", async function gatherPages() {
|
||||||
const pageDir = path.resolve(paths.gallery_dir, "src/pages");
|
const pageDir = path.resolve(paths.gallery_dir, "src/pages");
|
||||||
const files = glob.sync(path.resolve(pageDir, "**/*"));
|
const files = await glob(path.resolve(pageDir, "**/*"));
|
||||||
|
|
||||||
const galleryBuild = path.resolve(paths.gallery_dir, "build");
|
const galleryBuild = path.resolve(paths.gallery_dir, "build");
|
||||||
fs.mkdirSync(galleryBuild, { recursive: true });
|
fs.mkdirSync(galleryBuild, { recursive: true });
|
||||||
|
@@ -1,7 +1,5 @@
|
|||||||
const gulp = require("gulp");
|
const gulp = require("gulp");
|
||||||
|
|
||||||
const env = require("../env");
|
const env = require("../env");
|
||||||
|
|
||||||
require("./clean.js");
|
require("./clean.js");
|
||||||
require("./gen-icons-json.js");
|
require("./gen-icons-json.js");
|
||||||
require("./webpack.js");
|
require("./webpack.js");
|
||||||
@@ -43,6 +41,6 @@ gulp.task(
|
|||||||
env.useRollup() ? "rollup-prod-hassio" : "webpack-prod-hassio",
|
env.useRollup() ? "rollup-prod-hassio" : "webpack-prod-hassio",
|
||||||
"gen-index-hassio-prod",
|
"gen-index-hassio-prod",
|
||||||
...// Don't compress running tests
|
...// Don't compress running tests
|
||||||
(env.isTest() ? [] : ["compress-hassio"])
|
(env.isTestBuild() ? [] : ["compress-hassio"])
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
@@ -5,6 +5,7 @@ const webpack = require("webpack");
|
|||||||
const WebpackDevServer = require("webpack-dev-server");
|
const WebpackDevServer = require("webpack-dev-server");
|
||||||
const log = require("fancy-log");
|
const log = require("fancy-log");
|
||||||
const path = require("path");
|
const path = require("path");
|
||||||
|
const env = require("../env");
|
||||||
const paths = require("../paths");
|
const paths = require("../paths");
|
||||||
const {
|
const {
|
||||||
createAppConfig,
|
createAppConfig,
|
||||||
@@ -104,6 +105,8 @@ gulp.task("webpack-prod-app", () =>
|
|||||||
prodBuild(
|
prodBuild(
|
||||||
bothBuilds(createAppConfig, {
|
bothBuilds(createAppConfig, {
|
||||||
isProdBuild: true,
|
isProdBuild: true,
|
||||||
|
isStatsBuild: env.isStatsBuild(),
|
||||||
|
isTestBuild: env.isTestBuild(),
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
@@ -161,6 +164,8 @@ gulp.task("webpack-prod-hassio", () =>
|
|||||||
prodBuild(
|
prodBuild(
|
||||||
bothBuilds(createHassioConfig, {
|
bothBuilds(createHassioConfig, {
|
||||||
isProdBuild: true,
|
isProdBuild: true,
|
||||||
|
isStatsBuild: env.isStatsBuild(),
|
||||||
|
isTestBuild: env.isTestBuild(),
|
||||||
})
|
})
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
@@ -1,30 +1 @@
|
|||||||
[
|
[]
|
||||||
{
|
|
||||||
"path": "M20,20H7A2,2 0 0,1 5,18V8.94L2.23,5.64C2.09,5.47 2,5.24 2,5A1,1 0 0,1 3,4H20A2,2 0 0,1 22,6V18A2,2 0 0,1 20,20M8.5,7A0.5,0.5 0 0,0 8,7.5V8.5A0.5,0.5 0 0,0 8.5,9H18.5A0.5,0.5 0 0,0 19,8.5V7.5A0.5,0.5 0 0,0 18.5,7H8.5M8.5,11A0.5,0.5 0 0,0 8,11.5V12.5A0.5,0.5 0 0,0 8.5,13H18.5A0.5,0.5 0 0,0 19,12.5V11.5A0.5,0.5 0 0,0 18.5,11H8.5M8.5,15A0.5,0.5 0 0,0 8,15.5V16.5A0.5,0.5 0 0,0 8.5,17H13.5A0.5,0.5 0 0,0 14,16.5V15.5A0.5,0.5 0 0,0 13.5,15H8.5Z",
|
|
||||||
"name": "android-messages"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"path": "M4,6H2V20A2,2 0 0,0 4,22H18V20H4V6M20,2H8A2,2 0 0,0 6,4V16A2,2 0 0,0 8,18H20A2,2 0 0,0 22,16V4A2,2 0 0,0 20,2M20,12L17.5,10.5L15,12V4H20V12Z",
|
|
||||||
"name": "book-variant-multiple"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"path": "M21,14H3V4H21M21,2H3C1.89,2 1,2.89 1,4V16A2,2 0 0,0 3,18H10L8,21V22H16V21L14,18H21A2,2 0 0,0 23,16V4C23,2.89 22.1,2 21,2Z",
|
|
||||||
"name": "desktop-mac"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"path": "M21,14V4H3V14H21M21,2A2,2 0 0,1 23,4V16A2,2 0 0,1 21,18H14L16,21V22H8V21L10,18H3C1.89,18 1,17.1 1,16V4C1,2.89 1.89,2 3,2H21M4,5H15V10H4V5M16,5H20V7H16V5M20,8V13H16V8H20M4,11H9V13H4V11M10,11H15V13H10V11Z",
|
|
||||||
"name": "desktop-mac-dashboard"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"path": "M22,24L16.75,19L17.38,21H4.5A2.5,2.5 0 0,1 2,18.5V3.5A2.5,2.5 0 0,1 4.5,1H19.5A2.5,2.5 0 0,1 22,3.5V24M12,6.8C9.32,6.8 7.44,7.95 7.44,7.95C8.47,7.03 10.27,6.5 10.27,6.5L10.1,6.33C8.41,6.36 6.88,7.53 6.88,7.53C5.16,11.12 5.27,14.22 5.27,14.22C6.67,16.03 8.75,15.9 8.75,15.9L9.46,15C8.21,14.73 7.42,13.62 7.42,13.62C7.42,13.62 9.3,14.9 12,14.9C14.7,14.9 16.58,13.62 16.58,13.62C16.58,13.62 15.79,14.73 14.54,15L15.25,15.9C15.25,15.9 17.33,16.03 18.73,14.22C18.73,14.22 18.84,11.12 17.12,7.53C17.12,7.53 15.59,6.36 13.9,6.33L13.73,6.5C13.73,6.5 15.53,7.03 16.56,7.95C16.56,7.95 14.68,6.8 12,6.8M9.93,10.59C10.58,10.59 11.11,11.16 11.1,11.86C11.1,12.55 10.58,13.13 9.93,13.13C9.29,13.13 8.77,12.55 8.77,11.86C8.77,11.16 9.28,10.59 9.93,10.59M14.1,10.59C14.75,10.59 15.27,11.16 15.27,11.86C15.27,12.55 14.75,13.13 14.1,13.13C13.46,13.13 12.94,12.55 12.94,11.86C12.94,11.16 13.45,10.59 14.1,10.59Z",
|
|
||||||
"name": "discord"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"path": "M8.06,7.78C7.5,7.78 7.17,7.73 7.08,7.64L6.66,13.73C7.19,14.05 7.88,14.3 8.72,14.5C9.56,14.71 10.78,14.77 12.38,14.67C13.97,14.58 15.63,14.23 17.34,13.64L16.55,4.22C15.67,5.09 14.38,5.91 12.66,6.66C11.13,7.31 9.81,7.69 8.72,7.78H8.06M7.97,5.34C7.28,5.94 7,6.34 7.13,6.56C7.22,6.78 7.7,6.84 8.58,6.75C9.67,6.66 10.91,6.31 12.28,5.72C13.22,5.31 14.03,4.88 14.72,4.41C15.41,3.94 15.88,3.55 16.13,3.23C16.38,2.92 16.47,2.7 16.41,2.58C16.34,2.42 16.03,2.34 15.47,2.34C14.34,2.34 12.94,2.7 11.25,3.42C9.81,4.05 8.72,4.69 7.97,5.34M17.34,2.2C17.41,2.33 17.44,2.47 17.44,2.63L18.61,17C18.61,18.73 18,20.09 16.83,21.07C15.64,22.05 14.03,22.55 12,22.55C10,22.55 8.4,22.04 7.2,21C6,20 5.39,18.64 5.39,16.92L6.09,6.47C6.09,6.22 6.2,5.94 6.42,5.63C6.64,5.31 6.84,5.06 7.03,4.88L7.36,4.59C8.33,3.78 9.5,3.08 10.88,2.5C11.81,2.08 12.73,1.77 13.62,1.57C14.5,1.37 15.3,1.3 16,1.38C16.71,1.46 17.16,1.73 17.34,2.2Z",
|
|
||||||
"name": "google-home"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"path": "M19.25,19H4.75V3H19.25M14,22H10V21H14M18,0H6A3,3 0 0,0 3,3V21A3,3 0 0,0 6,24H18A3,3 0 0,0 21,21V3A3,3 0 0,0 18,0Z",
|
|
||||||
"name": "tablet-android"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
|
@@ -39,11 +39,18 @@ const createRollupConfig = ({
|
|||||||
inputOptions: {
|
inputOptions: {
|
||||||
input: entry,
|
input: entry,
|
||||||
// Some entry points contain no JavaScript. This setting silences a warning about that.
|
// Some entry points contain no JavaScript. This setting silences a warning about that.
|
||||||
// https://rollupjs.org/guide/en/#preserveentrysignatures
|
// https://rollupjs.org/configuration-options/#preserveentrysignatures
|
||||||
preserveEntrySignatures: false,
|
preserveEntrySignatures: false,
|
||||||
plugins: [
|
plugins: [
|
||||||
ignore({
|
ignore({
|
||||||
files: bundle.emptyPackages({ latestBuild }),
|
files: bundle
|
||||||
|
.emptyPackages({ latestBuild })
|
||||||
|
// TEMP HACK: Makes Rollup build work again
|
||||||
|
.concat(
|
||||||
|
require.resolve(
|
||||||
|
"@webcomponents/scoped-custom-element-registry/scoped-custom-element-registry.min"
|
||||||
|
)
|
||||||
|
),
|
||||||
}),
|
}),
|
||||||
resolve({
|
resolve({
|
||||||
extensions,
|
extensions,
|
||||||
@@ -54,7 +61,7 @@ const createRollupConfig = ({
|
|||||||
commonjs(),
|
commonjs(),
|
||||||
json(),
|
json(),
|
||||||
babel({
|
babel({
|
||||||
...bundle.babelOptions({ latestBuild }),
|
...bundle.babelOptions({ latestBuild, isProdBuild }),
|
||||||
extensions,
|
extensions,
|
||||||
babelHelpers: isWDS ? "inline" : "bundled",
|
babelHelpers: isWDS ? "inline" : "bundled",
|
||||||
}),
|
}),
|
||||||
@@ -69,7 +76,7 @@ const createRollupConfig = ({
|
|||||||
}),
|
}),
|
||||||
!isWDS && worker(),
|
!isWDS && worker(),
|
||||||
!isWDS && dontHashPlugin({ dontHash }),
|
!isWDS && dontHashPlugin({ dontHash }),
|
||||||
!isWDS && isProdBuild && terser(bundle.terserOptions(latestBuild)),
|
!isWDS && isProdBuild && terser(bundle.terserOptions({ latestBuild })),
|
||||||
!isWDS &&
|
!isWDS &&
|
||||||
isStatsBuild &&
|
isStatsBuild &&
|
||||||
visualizer({
|
visualizer({
|
||||||
@@ -83,20 +90,20 @@ const createRollupConfig = ({
|
|||||||
* @type { import("rollup").OutputOptions }
|
* @type { import("rollup").OutputOptions }
|
||||||
*/
|
*/
|
||||||
outputOptions: {
|
outputOptions: {
|
||||||
// https://rollupjs.org/guide/en/#outputdir
|
// https://rollupjs.org/configuration-options/#output-dir
|
||||||
dir: outputPath,
|
dir: outputPath,
|
||||||
// https://rollupjs.org/guide/en/#outputformat
|
// https://rollupjs.org/configuration-options/#output-format
|
||||||
format: latestBuild ? "es" : "systemjs",
|
format: latestBuild ? "es" : "systemjs",
|
||||||
// https://rollupjs.org/guide/en/#outputexternallivebindings
|
// https://rollupjs.org/configuration-options/#output-externallivebindings
|
||||||
externalLiveBindings: false,
|
externalLiveBindings: false,
|
||||||
// https://rollupjs.org/guide/en/#outputentryfilenames
|
// https://rollupjs.org/configuration-options/#output-entryfilenames
|
||||||
// https://rollupjs.org/guide/en/#outputchunkfilenames
|
// https://rollupjs.org/configuration-options/#output-chunkfilenames
|
||||||
// https://rollupjs.org/guide/en/#outputassetfilenames
|
// https://rollupjs.org/configuration-options/#output-assetfilenames
|
||||||
entryFileNames:
|
entryFileNames:
|
||||||
isProdBuild && !isStatsBuild ? "[name]-[hash].js" : "[name].js",
|
isProdBuild && !isStatsBuild ? "[name]-[hash].js" : "[name].js",
|
||||||
chunkFileNames: isProdBuild && !isStatsBuild ? "c.[hash].js" : "[name].js",
|
chunkFileNames: isProdBuild && !isStatsBuild ? "c.[hash].js" : "[name].js",
|
||||||
assetFileNames: isProdBuild && !isStatsBuild ? "a.[hash].js" : "[name].js",
|
assetFileNames: isProdBuild && !isStatsBuild ? "a.[hash].js" : "[name].js",
|
||||||
// https://rollupjs.org/guide/en/#outputsourcemap
|
// https://rollupjs.org/configuration-options/#output-sourcemap
|
||||||
sourcemap: isProdBuild ? true : "inline",
|
sourcemap: isProdBuild ? true : "inline",
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
@@ -22,6 +22,7 @@ class LogStartCompilePlugin {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const createWebpackConfig = ({
|
const createWebpackConfig = ({
|
||||||
|
name,
|
||||||
entry,
|
entry,
|
||||||
outputPath,
|
outputPath,
|
||||||
publicPath,
|
publicPath,
|
||||||
@@ -29,6 +30,7 @@ const createWebpackConfig = ({
|
|||||||
isProdBuild,
|
isProdBuild,
|
||||||
latestBuild,
|
latestBuild,
|
||||||
isStatsBuild,
|
isStatsBuild,
|
||||||
|
isTestBuild,
|
||||||
isHassioBuild,
|
isHassioBuild,
|
||||||
dontHash,
|
dontHash,
|
||||||
}) => {
|
}) => {
|
||||||
@@ -37,10 +39,16 @@ const createWebpackConfig = ({
|
|||||||
}
|
}
|
||||||
const ignorePackages = bundle.ignorePackages({ latestBuild });
|
const ignorePackages = bundle.ignorePackages({ latestBuild });
|
||||||
return {
|
return {
|
||||||
|
name,
|
||||||
mode: isProdBuild ? "production" : "development",
|
mode: isProdBuild ? "production" : "development",
|
||||||
target: ["web", latestBuild ? "es2017" : "es5"],
|
target: ["web", latestBuild ? "es2017" : "es5"],
|
||||||
devtool: isProdBuild
|
// For tests/CI, source maps are skipped to gain build speed
|
||||||
? "cheap-module-source-map"
|
// For production, generate source maps for accurate stack traces without source code
|
||||||
|
// For development, generate "cheap" versions that can map to original line numbers
|
||||||
|
devtool: isTestBuild
|
||||||
|
? false
|
||||||
|
: isProdBuild
|
||||||
|
? "nosources-source-map"
|
||||||
: "eval-cheap-module-source-map",
|
: "eval-cheap-module-source-map",
|
||||||
entry,
|
entry,
|
||||||
node: false,
|
node: false,
|
||||||
@@ -51,7 +59,7 @@ const createWebpackConfig = ({
|
|||||||
use: {
|
use: {
|
||||||
loader: "babel-loader",
|
loader: "babel-loader",
|
||||||
options: {
|
options: {
|
||||||
...bundle.babelOptions({ latestBuild }),
|
...bundle.babelOptions({ latestBuild, isProdBuild, isTestBuild }),
|
||||||
cacheDirectory: !isProdBuild,
|
cacheDirectory: !isProdBuild,
|
||||||
cacheCompression: false,
|
cacheCompression: false,
|
||||||
},
|
},
|
||||||
@@ -68,7 +76,7 @@ const createWebpackConfig = ({
|
|||||||
new TerserPlugin({
|
new TerserPlugin({
|
||||||
parallel: true,
|
parallel: true,
|
||||||
extractComments: true,
|
extractComments: true,
|
||||||
terserOptions: bundle.terserOptions(latestBuild),
|
terserOptions: bundle.terserOptions({ latestBuild, isTestBuild }),
|
||||||
}),
|
}),
|
||||||
],
|
],
|
||||||
moduleIds: isProdBuild && !isStatsBuild ? "deterministic" : "named",
|
moduleIds: isProdBuild && !isStatsBuild ? "deterministic" : "named",
|
||||||
@@ -153,6 +161,22 @@ const createWebpackConfig = ({
|
|||||||
publicPath,
|
publicPath,
|
||||||
// To silence warning in worker plugin
|
// To silence warning in worker plugin
|
||||||
globalObject: "self",
|
globalObject: "self",
|
||||||
|
// Since production source maps don't include sources, we need to point to them elsewhere
|
||||||
|
// For dependencies, just provide the path (no source in browser)
|
||||||
|
// Otherwise, point to the raw code on GitHub for browser to load
|
||||||
|
devtoolModuleFilenameTemplate:
|
||||||
|
!isTestBuild && isProdBuild
|
||||||
|
? (info) => {
|
||||||
|
const sourcePath = info.resourcePath.replace(/^\.\//, "");
|
||||||
|
if (
|
||||||
|
sourcePath.startsWith("node_modules") ||
|
||||||
|
sourcePath.startsWith("webpack")
|
||||||
|
) {
|
||||||
|
return `no-source/${sourcePath}`;
|
||||||
|
}
|
||||||
|
return `${bundle.sourceMapURL()}/${sourcePath}`;
|
||||||
|
}
|
||||||
|
: undefined,
|
||||||
},
|
},
|
||||||
experiments: {
|
experiments: {
|
||||||
topLevelAwait: true,
|
topLevelAwait: true,
|
||||||
@@ -160,9 +184,14 @@ const createWebpackConfig = ({
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild }) =>
|
const createAppConfig = ({
|
||||||
|
isProdBuild,
|
||||||
|
latestBuild,
|
||||||
|
isStatsBuild,
|
||||||
|
isTestBuild,
|
||||||
|
}) =>
|
||||||
createWebpackConfig(
|
createWebpackConfig(
|
||||||
bundle.config.app({ isProdBuild, latestBuild, isStatsBuild })
|
bundle.config.app({ isProdBuild, latestBuild, isStatsBuild, isTestBuild })
|
||||||
);
|
);
|
||||||
|
|
||||||
const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) =>
|
const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) =>
|
||||||
@@ -173,8 +202,20 @@ const createDemoConfig = ({ isProdBuild, latestBuild, isStatsBuild }) =>
|
|||||||
const createCastConfig = ({ isProdBuild, latestBuild }) =>
|
const createCastConfig = ({ isProdBuild, latestBuild }) =>
|
||||||
createWebpackConfig(bundle.config.cast({ isProdBuild, latestBuild }));
|
createWebpackConfig(bundle.config.cast({ isProdBuild, latestBuild }));
|
||||||
|
|
||||||
const createHassioConfig = ({ isProdBuild, latestBuild }) =>
|
const createHassioConfig = ({
|
||||||
createWebpackConfig(bundle.config.hassio({ isProdBuild, latestBuild }));
|
isProdBuild,
|
||||||
|
latestBuild,
|
||||||
|
isStatsBuild,
|
||||||
|
isTestBuild,
|
||||||
|
}) =>
|
||||||
|
createWebpackConfig(
|
||||||
|
bundle.config.hassio({
|
||||||
|
isProdBuild,
|
||||||
|
latestBuild,
|
||||||
|
isStatsBuild,
|
||||||
|
isTestBuild,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
const createGalleryConfig = ({ isProdBuild, latestBuild }) =>
|
const createGalleryConfig = ({ isProdBuild, latestBuild }) =>
|
||||||
createWebpackConfig(bundle.config.gallery({ isProdBuild, latestBuild }));
|
createWebpackConfig(bundle.config.gallery({ isProdBuild, latestBuild }));
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { html, TemplateResult } 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";
|
import { LovelaceConfig } from "../../../../src/data/lovelace";
|
||||||
@@ -18,9 +18,9 @@ class HcDemo extends HassElement {
|
|||||||
|
|
||||||
@state() private _lovelaceConfig?: LovelaceConfig;
|
@state() private _lovelaceConfig?: LovelaceConfig;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this._lovelaceConfig) {
|
if (!this._lovelaceConfig) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<hc-lovelace
|
<hc-lovelace
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import { mdiTelevision } from "@mdi/js";
|
import { mdiTelevision } from "@mdi/js";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, state } from "lit/decorators";
|
import { customElement, state } from "lit/decorators";
|
||||||
import { CastManager } from "../../../src/cast/cast_manager";
|
import { CastManager } from "../../../src/cast/cast_manager";
|
||||||
import { castSendShowDemo } from "../../../src/cast/receiver_messages";
|
import { castSendShowDemo } from "../../../src/cast/receiver_messages";
|
||||||
@@ -20,12 +20,12 @@ class CastDemoRow extends LitElement implements LovelaceRow {
|
|||||||
// No config possible.
|
// No config possible.
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (
|
if (
|
||||||
!this._castManager ||
|
!this._castManager ||
|
||||||
this._castManager.castState === "NO_DEVICES_AVAILABLE"
|
this._castManager.castState === "NO_DEVICES_AVAILABLE"
|
||||||
) {
|
) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<ha-svg-icon .path=${mdiTelevision}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiTelevision}></ha-svg-icon>
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import "@material/mwc-button";
|
import "@material/mwc-button";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
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";
|
||||||
@@ -30,9 +30,9 @@ export class HADemoCard extends LitElement implements LovelaceCard {
|
|||||||
|
|
||||||
public setConfig(_config: LovelaceCardConfig) {}
|
public setConfig(_config: LovelaceCardConfig) {}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (this._hidden) {
|
if (this._hidden) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<ha-card>
|
<ha-card>
|
||||||
|
@@ -1,39 +1,19 @@
|
|||||||
import { HassEntity } from "home-assistant-js-websocket";
|
import { HassEntity } from "home-assistant-js-websocket";
|
||||||
|
import { HistoryStates } from "../../../src/data/history";
|
||||||
import { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
import { MockHomeAssistant } from "../../../src/fake_data/provide_hass";
|
||||||
|
|
||||||
interface HistoryQueryParams {
|
const generateStateHistory = (
|
||||||
filter_entity_id: string;
|
state: HassEntity,
|
||||||
end_time: string;
|
deltas,
|
||||||
}
|
start_date: Date,
|
||||||
|
end_date: Date
|
||||||
const parseQuery = <T>(queryString: string) => {
|
) => {
|
||||||
const query: any = {};
|
|
||||||
const items = queryString.split("&");
|
|
||||||
for (const item of items) {
|
|
||||||
const parts = item.split("=");
|
|
||||||
const key = decodeURIComponent(parts[0]);
|
|
||||||
const value = parts.length > 1 ? decodeURIComponent(parts[1]) : undefined;
|
|
||||||
query[key] = value;
|
|
||||||
}
|
|
||||||
return query as T;
|
|
||||||
};
|
|
||||||
|
|
||||||
const getTime = (minutesAgo) => {
|
|
||||||
const ts = new Date(Date.now() - minutesAgo * 60 * 1000);
|
|
||||||
return ts.toISOString();
|
|
||||||
};
|
|
||||||
|
|
||||||
const randomTimeAdjustment = (diff) => Math.random() * diff - diff / 2;
|
|
||||||
|
|
||||||
const maxTime = 1440;
|
|
||||||
|
|
||||||
const generateHistory = (state, deltas) => {
|
|
||||||
const changes =
|
const changes =
|
||||||
typeof deltas[0] === "object"
|
typeof deltas[0] === "object"
|
||||||
? deltas
|
? deltas
|
||||||
: deltas.map((st) => ({ state: st }));
|
: deltas.map((st) => ({ state: st }));
|
||||||
|
|
||||||
const timeDiff = 900 / changes.length;
|
const timeDiff = (end_date.getTime() - start_date.getTime()) / changes.length;
|
||||||
|
|
||||||
return changes.map((change, index) => {
|
return changes.map((change, index) => {
|
||||||
let attributes;
|
let attributes;
|
||||||
@@ -47,17 +27,13 @@ const generateHistory = (state, deltas) => {
|
|||||||
attributes = { ...state.attributes, ...change.attributes };
|
attributes = { ...state.attributes, ...change.attributes };
|
||||||
}
|
}
|
||||||
|
|
||||||
const time =
|
const time = start_date.getTime() + timeDiff * index;
|
||||||
index === 0
|
|
||||||
? getTime(maxTime)
|
|
||||||
: getTime(maxTime - index * timeDiff + randomTimeAdjustment(timeDiff));
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
attributes,
|
a: attributes,
|
||||||
entity_id: state.entity_id,
|
s: change.state || state.state,
|
||||||
state: change.state || state.state,
|
lc: time / 1000,
|
||||||
last_changed: time,
|
lu: time / 1000,
|
||||||
last_updated: time,
|
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
@@ -65,15 +41,29 @@ const generateHistory = (state, deltas) => {
|
|||||||
const incrementalUnits = ["clients", "queries", "ads"];
|
const incrementalUnits = ["clients", "queries", "ads"];
|
||||||
|
|
||||||
export const mockHistory = (mockHass: MockHomeAssistant) => {
|
export const mockHistory = (mockHass: MockHomeAssistant) => {
|
||||||
mockHass.mockAPI(
|
mockHass.mockWS(
|
||||||
/history\/period\/.+/,
|
"history/stream",
|
||||||
(hass, _method, path, _parameters) => {
|
(
|
||||||
const params = parseQuery<HistoryQueryParams>(path.split("?")[1]);
|
{
|
||||||
const entities = params.filter_entity_id.split(",");
|
entity_ids,
|
||||||
|
start_time,
|
||||||
|
end_time,
|
||||||
|
}: {
|
||||||
|
entity_ids: string[];
|
||||||
|
start_time: string;
|
||||||
|
end_time?: string;
|
||||||
|
},
|
||||||
|
hass,
|
||||||
|
onChange
|
||||||
|
) => {
|
||||||
|
const states: HistoryStates = {};
|
||||||
|
|
||||||
const results: HassEntity[][] = [];
|
const start = new Date(start_time);
|
||||||
|
const end = end_time ? new Date(end_time) : new Date();
|
||||||
|
|
||||||
|
for (const entityId of entity_ids) {
|
||||||
|
states[entityId] = [];
|
||||||
|
|
||||||
for (const entityId of entities) {
|
|
||||||
const state = hass.states[entityId];
|
const state = hass.states[entityId];
|
||||||
|
|
||||||
if (!state) {
|
if (!state) {
|
||||||
@@ -81,7 +71,12 @@ export const mockHistory = (mockHass: MockHomeAssistant) => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!state.attributes.unit_of_measurement) {
|
if (!state.attributes.unit_of_measurement) {
|
||||||
results.push(generateHistory(state, [state.state]));
|
states[entityId] = generateStateHistory(
|
||||||
|
state,
|
||||||
|
[state.state],
|
||||||
|
start,
|
||||||
|
end
|
||||||
|
);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -120,17 +115,23 @@ export const mockHistory = (mockHass: MockHomeAssistant) => {
|
|||||||
numberState - diff + Math.floor(Math.random() * 2 * diff);
|
numberState - diff + Math.floor(Math.random() * 2 * diff);
|
||||||
}
|
}
|
||||||
|
|
||||||
results.push(
|
states[entityId] = generateStateHistory(
|
||||||
generateHistory(
|
state,
|
||||||
{
|
Array.from({ length: statesToGenerate }, genFunc),
|
||||||
entity_id: state.entity_id,
|
start,
|
||||||
attributes: state.attributes,
|
end
|
||||||
},
|
|
||||||
Array.from({ length: statesToGenerate }, genFunc)
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return results;
|
|
||||||
|
setTimeout(() => {
|
||||||
|
onChange?.({
|
||||||
|
states,
|
||||||
|
start_time: start,
|
||||||
|
end_time: end,
|
||||||
|
});
|
||||||
|
}, 1);
|
||||||
|
|
||||||
|
return () => {};
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { html, css } from "lit";
|
import { css, html, nothing } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import { until } from "lit/directives/until";
|
import { until } from "lit/directives/until";
|
||||||
import { HaMarkdown } from "../../../src/components/ha-markdown";
|
import { HaMarkdown } from "../../../src/components/ha-markdown";
|
||||||
@@ -10,7 +10,7 @@ class PageDescription extends HaMarkdown {
|
|||||||
|
|
||||||
render() {
|
render() {
|
||||||
if (!PAGES[this.page].description) {
|
if (!PAGES[this.page].description) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import { dump } from "js-yaml";
|
import { dump } from "js-yaml";
|
||||||
import { html, css, LitElement, TemplateResult } from "lit";
|
import { css, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import "../../../../src/components/ha-card";
|
import "../../../../src/components/ha-card";
|
||||||
import "../../../../src/components/ha-yaml-editor";
|
import "../../../../src/components/ha-yaml-editor";
|
||||||
@@ -127,9 +127,9 @@ export class DemoAutomationDescribeAction extends LitElement {
|
|||||||
|
|
||||||
@state() _action = initialAction;
|
@state() _action = initialAction;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<ha-card header="Actions">
|
<ha-card header="Actions">
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import { dump } from "js-yaml";
|
import { dump } from "js-yaml";
|
||||||
import { css, html, LitElement, TemplateResult } from "lit";
|
import { css, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import "../../../../src/components/ha-card";
|
import "../../../../src/components/ha-card";
|
||||||
import "../../../../src/components/ha-yaml-editor";
|
import "../../../../src/components/ha-yaml-editor";
|
||||||
@@ -53,9 +53,9 @@ export class DemoAutomationDescribeCondition extends LitElement {
|
|||||||
|
|
||||||
@state() _condition = initialCondition;
|
@state() _condition = initialCondition;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import { dump } from "js-yaml";
|
import { dump } from "js-yaml";
|
||||||
import { css, html, LitElement, TemplateResult } from "lit";
|
import { css, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import "../../../../src/components/ha-card";
|
import "../../../../src/components/ha-card";
|
||||||
import "../../../../src/components/ha-yaml-editor";
|
import "../../../../src/components/ha-yaml-editor";
|
||||||
@@ -64,9 +64,9 @@ export class DemoAutomationDescribeTrigger extends LitElement {
|
|||||||
|
|
||||||
@state() _trigger = initialTrigger;
|
@state() _trigger = initialTrigger;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
|
@@ -1,5 +1,6 @@
|
|||||||
/* eslint-disable lit/no-template-arrow */
|
/* eslint-disable lit/no-template-arrow */
|
||||||
import { html, css, LitElement, TemplateResult } 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-script-graph";
|
||||||
@@ -29,9 +30,9 @@ const traces: DemoTrace[] = [
|
|||||||
export class DemoAutomationTraceTimeline extends LitElement {
|
export class DemoAutomationTraceTimeline extends LitElement {
|
||||||
@property({ attribute: false }) hass?: HomeAssistant;
|
@property({ attribute: false }) hass?: HomeAssistant;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
${traces.map(
|
${traces.map(
|
||||||
|
@@ -1,14 +1,15 @@
|
|||||||
/* eslint-disable lit/no-template-arrow */
|
/* eslint-disable lit/no-template-arrow */
|
||||||
import { html, css, LitElement, TemplateResult } from "lit";
|
|
||||||
|
import { css, html, LitElement, nothing } from "lit";
|
||||||
|
import { customElement, property, state } 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-script-graph";
|
||||||
import "../../../../src/components/trace/hat-trace-timeline";
|
import "../../../../src/components/trace/hat-trace-timeline";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
|
||||||
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";
|
||||||
import { DemoTrace } from "../../data/traces/types";
|
|
||||||
import { basicTrace } from "../../data/traces/basic_trace";
|
import { basicTrace } from "../../data/traces/basic_trace";
|
||||||
import { motionLightTrace } from "../../data/traces/motion-light-trace";
|
import { motionLightTrace } from "../../data/traces/motion-light-trace";
|
||||||
|
import { DemoTrace } from "../../data/traces/types";
|
||||||
|
|
||||||
const traces: DemoTrace[] = [basicTrace, motionLightTrace];
|
const traces: DemoTrace[] = [basicTrace, motionLightTrace];
|
||||||
|
|
||||||
@@ -18,9 +19,9 @@ export class DemoAutomationTrace extends LitElement {
|
|||||||
|
|
||||||
@state() private _selected = {};
|
@state() private _selected = {};
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
${traces.map(
|
${traces.map(
|
||||||
|
3
gallery/src/pages/components/ha-control-select.markdown
Normal file
3
gallery/src/pages/components/ha-control-select.markdown
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
---
|
||||||
|
title: Control Select
|
||||||
|
---
|
212
gallery/src/pages/components/ha-control-select.ts
Normal file
212
gallery/src/pages/components/ha-control-select.ts
Normal file
@@ -0,0 +1,212 @@
|
|||||||
|
import { mdiFanOff, mdiFanSpeed1, mdiFanSpeed2, mdiFanSpeed3 } from "@mdi/js";
|
||||||
|
import { css, html, LitElement, TemplateResult } from "lit";
|
||||||
|
import { customElement, state } from "lit/decorators";
|
||||||
|
import { ifDefined } from "lit/directives/if-defined";
|
||||||
|
import { repeat } from "lit/directives/repeat";
|
||||||
|
import "../../../../src/components/ha-card";
|
||||||
|
import "../../../../src/components/ha-control-select";
|
||||||
|
import type { ControlSelectOption } from "../../../../src/components/ha-control-select";
|
||||||
|
|
||||||
|
const fullOptions: ControlSelectOption[] = [
|
||||||
|
{
|
||||||
|
value: "off",
|
||||||
|
label: "Off",
|
||||||
|
path: mdiFanOff,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: "low",
|
||||||
|
label: "Low",
|
||||||
|
path: mdiFanSpeed1,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: "medium",
|
||||||
|
label: "Medium",
|
||||||
|
path: mdiFanSpeed2,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: "high",
|
||||||
|
label: "High",
|
||||||
|
path: mdiFanSpeed3,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const iconOptions: ControlSelectOption[] = [
|
||||||
|
{
|
||||||
|
value: "off",
|
||||||
|
path: mdiFanOff,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: "low",
|
||||||
|
path: mdiFanSpeed1,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: "medium",
|
||||||
|
path: mdiFanSpeed2,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: "high",
|
||||||
|
path: mdiFanSpeed3,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const labelOptions: ControlSelectOption[] = [
|
||||||
|
{
|
||||||
|
value: "off",
|
||||||
|
label: "Off",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: "low",
|
||||||
|
label: "Low",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: "medium",
|
||||||
|
label: "Medium",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
value: "high",
|
||||||
|
label: "High",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const selects: {
|
||||||
|
id: string;
|
||||||
|
label: string;
|
||||||
|
class?: string;
|
||||||
|
options: ControlSelectOption[];
|
||||||
|
disabled?: boolean;
|
||||||
|
}[] = [
|
||||||
|
{
|
||||||
|
id: "label",
|
||||||
|
label: "Select with labels",
|
||||||
|
options: labelOptions,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "icon",
|
||||||
|
label: "Select with icons",
|
||||||
|
options: iconOptions,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "icon",
|
||||||
|
label: "Disabled select",
|
||||||
|
options: iconOptions,
|
||||||
|
disabled: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
id: "custom",
|
||||||
|
label: "Select and custom style",
|
||||||
|
class: "custom",
|
||||||
|
options: fullOptions,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
@customElement("demo-components-ha-control-select")
|
||||||
|
export class DemoHaControlSelect extends LitElement {
|
||||||
|
@state() private value?: string = "off";
|
||||||
|
|
||||||
|
handleValueChanged(e: CustomEvent) {
|
||||||
|
this.value = e.detail.value as string;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
return html`
|
||||||
|
<ha-card>
|
||||||
|
<div class="card-content">
|
||||||
|
<p><b>Slider values</b></p>
|
||||||
|
<table>
|
||||||
|
<tbody>
|
||||||
|
<tr>
|
||||||
|
<td>value</td>
|
||||||
|
<td>${this.value ?? "-"}</td>
|
||||||
|
</tr>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</ha-card>
|
||||||
|
${repeat(selects, (select) => {
|
||||||
|
const { id, label, options, ...config } = select;
|
||||||
|
return html`
|
||||||
|
<ha-card>
|
||||||
|
<div class="card-content">
|
||||||
|
<label id=${id}>${label}</label>
|
||||||
|
<pre>Config: ${JSON.stringify(config)}</pre>
|
||||||
|
<ha-control-select
|
||||||
|
.value=${this.value}
|
||||||
|
.options=${options}
|
||||||
|
class=${ifDefined(config.class)}
|
||||||
|
@value-changed=${this.handleValueChanged}
|
||||||
|
aria-labelledby=${id}
|
||||||
|
disabled=${ifDefined(config.disabled)}
|
||||||
|
>
|
||||||
|
</ha-control-select>
|
||||||
|
</div>
|
||||||
|
</ha-card>
|
||||||
|
`;
|
||||||
|
})}
|
||||||
|
<ha-card>
|
||||||
|
<div class="card-content">
|
||||||
|
<p class="title"><b>Vertical</b></p>
|
||||||
|
<div class="vertical-selects">
|
||||||
|
${repeat(selects, (select) => {
|
||||||
|
const { id, label, options, ...config } = select;
|
||||||
|
return html`
|
||||||
|
<ha-control-select
|
||||||
|
.value=${this.value}
|
||||||
|
.options=${options}
|
||||||
|
vertical
|
||||||
|
class=${ifDefined(config.class)}
|
||||||
|
@value-changed=${this.handleValueChanged}
|
||||||
|
aria-labelledby=${id}
|
||||||
|
disabled=${ifDefined(config.disabled)}
|
||||||
|
>
|
||||||
|
</ha-control-select>
|
||||||
|
`;
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</ha-card>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles() {
|
||||||
|
return css`
|
||||||
|
ha-card {
|
||||||
|
max-width: 600px;
|
||||||
|
margin: 24px auto;
|
||||||
|
}
|
||||||
|
pre {
|
||||||
|
margin-top: 0;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
p {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
label {
|
||||||
|
font-weight: 600;
|
||||||
|
}
|
||||||
|
.custom {
|
||||||
|
--mdc-icon-size: 24px;
|
||||||
|
--control-select-color: var(--state-fan-active-color);
|
||||||
|
--control-select-thickness: 100px;
|
||||||
|
--control-select-border-radius: 24px;
|
||||||
|
}
|
||||||
|
.vertical-selects {
|
||||||
|
height: 300px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
p.title {
|
||||||
|
margin-bottom: 12px;
|
||||||
|
}
|
||||||
|
.vertical-selects > *:not(:last-child) {
|
||||||
|
margin-right: 4px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"demo-components-ha-control-select": DemoHaControlSelect;
|
||||||
|
}
|
||||||
|
}
|
@@ -2,7 +2,7 @@ import {
|
|||||||
HassEntity,
|
HassEntity,
|
||||||
HassEntityAttributeBase,
|
HassEntityAttributeBase,
|
||||||
} from "home-assistant-js-websocket";
|
} from "home-assistant-js-websocket";
|
||||||
import { css, html, LitElement, TemplateResult } from "lit";
|
import { css, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { computeDomain } from "../../../../src/common/entity/compute_domain";
|
import { computeDomain } from "../../../../src/common/entity/compute_domain";
|
||||||
@@ -387,9 +387,9 @@ export class DemoEntityState extends LitElement {
|
|||||||
hass.updateTranslations("config", "en");
|
hass.updateTranslations("config", "en");
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
|
@@ -1,22 +1,22 @@
|
|||||||
import { html, css, LitElement, TemplateResult } from "lit";
|
import { css, html, LitElement, nothing } from "lit";
|
||||||
import "../../../../src/components/ha-formfield";
|
import "../../../../src/components/ha-formfield";
|
||||||
import "../../../../src/components/ha-switch";
|
import "../../../../src/components/ha-switch";
|
||||||
|
|
||||||
import { classMap } from "lit/directives/class-map";
|
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
|
import { classMap } from "lit/directives/class-map";
|
||||||
import { IntegrationManifest } from "../../../../src/data/integration";
|
import { IntegrationManifest } from "../../../../src/data/integration";
|
||||||
|
|
||||||
|
import { DeviceRegistryEntry } from "../../../../src/data/device_registry";
|
||||||
|
import { EntityRegistryEntry } from "../../../../src/data/entity_registry";
|
||||||
import { provideHass } from "../../../../src/fake_data/provide_hass";
|
import { provideHass } from "../../../../src/fake_data/provide_hass";
|
||||||
import { HomeAssistant } from "../../../../src/types";
|
|
||||||
import "../../../../src/panels/config/integrations/ha-integration-card";
|
|
||||||
import "../../../../src/panels/config/integrations/ha-ignored-config-entry-card";
|
|
||||||
import "../../../../src/panels/config/integrations/ha-config-flow-card";
|
import "../../../../src/panels/config/integrations/ha-config-flow-card";
|
||||||
import type {
|
import type {
|
||||||
ConfigEntryExtended,
|
ConfigEntryExtended,
|
||||||
DataEntryFlowProgressExtended,
|
DataEntryFlowProgressExtended,
|
||||||
} from "../../../../src/panels/config/integrations/ha-config-integrations";
|
} from "../../../../src/panels/config/integrations/ha-config-integrations";
|
||||||
import { DeviceRegistryEntry } from "../../../../src/data/device_registry";
|
import "../../../../src/panels/config/integrations/ha-ignored-config-entry-card";
|
||||||
import { EntityRegistryEntry } from "../../../../src/data/entity_registry";
|
import "../../../../src/panels/config/integrations/ha-integration-card";
|
||||||
|
import { HomeAssistant } from "../../../../src/types";
|
||||||
|
|
||||||
const createConfigEntry = (
|
const createConfigEntry = (
|
||||||
title: string,
|
title: string,
|
||||||
@@ -231,9 +231,9 @@ export class DemoIntegrationCard extends LitElement {
|
|||||||
|
|
||||||
@state() isCloud = false;
|
@state() isCloud = false;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<div class="container">
|
<div class="container">
|
||||||
|
@@ -6,6 +6,7 @@ import {
|
|||||||
CSSResultGroup,
|
CSSResultGroup,
|
||||||
html,
|
html,
|
||||||
LitElement,
|
LitElement,
|
||||||
|
nothing,
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
TemplateResult,
|
TemplateResult,
|
||||||
} from "lit";
|
} from "lit";
|
||||||
@@ -73,8 +74,8 @@ export class HassioAddonStore extends LitElement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
let repos: TemplateResult[] = [];
|
let repos: (TemplateResult | typeof nothing)[] = [];
|
||||||
|
|
||||||
if (this.supervisor.store.repositories) {
|
if (this.supervisor.store.repositories) {
|
||||||
repos = this.addonRepositories(
|
repos = this.addonRepositories(
|
||||||
@@ -173,7 +174,7 @@ export class HassioAddonStore extends LitElement {
|
|||||||
.supervisor=${this.supervisor}
|
.supervisor=${this.supervisor}
|
||||||
></hassio-addon-repository>
|
></hassio-addon-repository>
|
||||||
`
|
`
|
||||||
: html``;
|
: nothing;
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@@ -114,9 +114,6 @@ class HassioAddonAudio extends LitElement {
|
|||||||
ha-card {
|
ha-card {
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
paper-item {
|
|
||||||
width: 450px;
|
|
||||||
}
|
|
||||||
.card-actions {
|
.card-actions {
|
||||||
text-align: right;
|
text-align: right;
|
||||||
}
|
}
|
||||||
|
@@ -4,7 +4,7 @@ import {
|
|||||||
html,
|
html,
|
||||||
LitElement,
|
LitElement,
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
TemplateResult,
|
nothing,
|
||||||
} from "lit";
|
} from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
@@ -47,9 +47,9 @@ class HassioAddonNetwork extends LitElement {
|
|||||||
this._setNetworkConfig();
|
this._setNetworkConfig();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this._config) {
|
if (!this._config) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
const hasHiddenOptions = Object.keys(this._config).find(
|
const hasHiddenOptions = Object.keys(this._config).find(
|
||||||
|
@@ -8,7 +8,7 @@ import {
|
|||||||
html,
|
html,
|
||||||
LitElement,
|
LitElement,
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
TemplateResult,
|
nothing,
|
||||||
} from "lit";
|
} from "lit";
|
||||||
import { customElement, property, query, state } from "lit/decorators";
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
import { classMap } from "lit/directives/class-map";
|
import { classMap } from "lit/directives/class-map";
|
||||||
@@ -160,9 +160,9 @@ export class HassioBackups extends LitElement {
|
|||||||
}))
|
}))
|
||||||
);
|
);
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.supervisor) {
|
if (!this.supervisor) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<hass-tabs-subpage-data-table
|
<hass-tabs-subpage-data-table
|
||||||
@@ -248,9 +248,9 @@ export class HassioBackups extends LitElement {
|
|||||||
class="warning"
|
class="warning"
|
||||||
@click=${this._deleteSelected}
|
@click=${this._deleteSelected}
|
||||||
></ha-icon-button>
|
></ha-icon-button>
|
||||||
<paper-tooltip animation-delay="0" for="delete-btn">
|
<simple-tooltip animation-delay="0" for="delete-btn">
|
||||||
${this.supervisor.localize("backup.delete_selected")}
|
${this.supervisor.localize("backup.delete_selected")}
|
||||||
</paper-tooltip>
|
</simple-tooltip>
|
||||||
`}
|
`}
|
||||||
</div>
|
</div>
|
||||||
</div> `
|
</div> `
|
||||||
|
@@ -1,6 +1,13 @@
|
|||||||
import { mdiFolder, mdiHomeAssistant, mdiPuzzle } from "@mdi/js";
|
import { mdiFolder, mdiHomeAssistant, mdiPuzzle } from "@mdi/js";
|
||||||
import { PaperInputElement } from "@polymer/paper-input/paper-input";
|
import { PaperInputElement } from "@polymer/paper-input/paper-input";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import {
|
||||||
|
css,
|
||||||
|
CSSResultGroup,
|
||||||
|
html,
|
||||||
|
LitElement,
|
||||||
|
TemplateResult,
|
||||||
|
nothing,
|
||||||
|
} from "lit";
|
||||||
import { customElement, property, query } from "lit/decorators";
|
import { customElement, property, query } from "lit/decorators";
|
||||||
import { atLeastVersion } from "../../../src/common/config/version";
|
import { atLeastVersion } from "../../../src/common/config/version";
|
||||||
import { formatDate } from "../../../src/common/datetime/format_date";
|
import { formatDate } from "../../../src/common/datetime/format_date";
|
||||||
@@ -11,9 +18,9 @@ import "../../../src/components/ha-formfield";
|
|||||||
import "../../../src/components/ha-radio";
|
import "../../../src/components/ha-radio";
|
||||||
import type { HaRadio } from "../../../src/components/ha-radio";
|
import type { HaRadio } from "../../../src/components/ha-radio";
|
||||||
import {
|
import {
|
||||||
|
HassioBackupDetail,
|
||||||
HassioFullBackupCreateParams,
|
HassioFullBackupCreateParams,
|
||||||
HassioPartialBackupCreateParams,
|
HassioPartialBackupCreateParams,
|
||||||
HassioBackupDetail,
|
|
||||||
} from "../../../src/data/hassio/backup";
|
} from "../../../src/data/hassio/backup";
|
||||||
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
import { PolymerChangedEvent } from "../../../src/polymer-types";
|
import { PolymerChangedEvent } from "../../../src/polymer-types";
|
||||||
@@ -115,9 +122,9 @@ export class SupervisorBackupContent extends LitElement {
|
|||||||
this.supervisor?.localize(`backup.${key}`) ||
|
this.supervisor?.localize(`backup.${key}`) ||
|
||||||
this.localize!(`ui.panel.page-onboarding.restore.${key}`);
|
this.localize!(`ui.panel.page-onboarding.restore.${key}`);
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.onboarding && !this.supervisor) {
|
if (!this.onboarding && !this.supervisor) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
const foldersSection =
|
const foldersSection =
|
||||||
this.backupType === "partial" ? this._getSection("folders") : undefined;
|
this.backupType === "partial" ? this._getSection("folders") : undefined;
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import "@material/mwc-button";
|
import "@material/mwc-button";
|
||||||
import { mdiHomeAssistant } from "@mdi/js";
|
import { mdiHomeAssistant } from "@mdi/js";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import "../../../src/components/buttons/ha-progress-button";
|
import "../../../src/components/buttons/ha-progress-button";
|
||||||
@@ -33,14 +33,14 @@ export class HassioUpdate extends LitElement {
|
|||||||
).length
|
).length
|
||||||
);
|
);
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.supervisor) {
|
if (!this.supervisor) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
const updatesAvailable = this._pendingUpdates(this.supervisor);
|
const updatesAvailable = this._pendingUpdates(this.supervisor);
|
||||||
if (!updatesAvailable) {
|
if (!updatesAvailable) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
@@ -80,9 +80,9 @@ export class HassioUpdate extends LitElement {
|
|||||||
name: string,
|
name: string,
|
||||||
key: string,
|
key: string,
|
||||||
object: HassioHomeAssistantInfo | HassioSupervisorInfo | HassioHassOSInfo
|
object: HassioHomeAssistantInfo | HassioSupervisorInfo | HassioHassOSInfo
|
||||||
): TemplateResult {
|
) {
|
||||||
if (!object.update_available) {
|
if (!object.update_available) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<ha-card outlined>
|
<ha-card outlined>
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import { mdiClose } from "@mdi/js";
|
import { mdiClose } from "@mdi/js";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
import "../../../../src/components/ha-header-bar";
|
import "../../../../src/components/ha-header-bar";
|
||||||
@@ -36,9 +36,9 @@ export class DialogHassioBackupUpload
|
|||||||
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this._dialogParams) {
|
if (!this._dialogParams) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
|
@@ -1,9 +1,11 @@
|
|||||||
import { ActionDetail } from "@material/mwc-list";
|
import { ActionDetail } from "@material/mwc-list";
|
||||||
import "@material/mwc-list/mwc-list-item";
|
import "@material/mwc-list/mwc-list-item";
|
||||||
import { mdiClose, mdiDotsVertical } from "@mdi/js";
|
import { mdiClose, mdiDotsVertical } from "@mdi/js";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, query, state } from "lit/decorators";
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
|
import { atLeastVersion } from "../../../../src/common/config/version";
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
|
import { stopPropagation } from "../../../../src/common/dom/stop_propagation";
|
||||||
import { slugify } from "../../../../src/common/string/slugify";
|
import { slugify } from "../../../../src/common/string/slugify";
|
||||||
import "../../../../src/components/buttons/ha-progress-button";
|
import "../../../../src/components/buttons/ha-progress-button";
|
||||||
import "../../../../src/components/ha-alert";
|
import "../../../../src/components/ha-alert";
|
||||||
@@ -11,11 +13,12 @@ import "../../../../src/components/ha-button-menu";
|
|||||||
import "../../../../src/components/ha-header-bar";
|
import "../../../../src/components/ha-header-bar";
|
||||||
import "../../../../src/components/ha-icon-button";
|
import "../../../../src/components/ha-icon-button";
|
||||||
import { getSignedPath } from "../../../../src/data/auth";
|
import { getSignedPath } from "../../../../src/data/auth";
|
||||||
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
|
||||||
import {
|
import {
|
||||||
fetchHassioBackupInfo,
|
fetchHassioBackupInfo,
|
||||||
HassioBackupDetail,
|
HassioBackupDetail,
|
||||||
|
removeBackup,
|
||||||
} from "../../../../src/data/hassio/backup";
|
} from "../../../../src/data/hassio/backup";
|
||||||
|
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
||||||
import {
|
import {
|
||||||
showAlertDialog,
|
showAlertDialog,
|
||||||
showConfirmationDialog,
|
showConfirmationDialog,
|
||||||
@@ -27,8 +30,6 @@ import { fileDownload } from "../../../../src/util/file_download";
|
|||||||
import "../../components/supervisor-backup-content";
|
import "../../components/supervisor-backup-content";
|
||||||
import type { SupervisorBackupContent } from "../../components/supervisor-backup-content";
|
import type { SupervisorBackupContent } from "../../components/supervisor-backup-content";
|
||||||
import { HassioBackupDialogParams } from "./show-dialog-hassio-backup";
|
import { HassioBackupDialogParams } from "./show-dialog-hassio-backup";
|
||||||
import { atLeastVersion } from "../../../../src/common/config/version";
|
|
||||||
import { stopPropagation } from "../../../../src/common/dom/stop_propagation";
|
|
||||||
|
|
||||||
@customElement("dialog-hassio-backup")
|
@customElement("dialog-hassio-backup")
|
||||||
class HassioBackupDialog
|
class HassioBackupDialog
|
||||||
@@ -62,9 +63,9 @@ class HassioBackupDialog
|
|||||||
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this._dialogParams || !this._backup) {
|
if (!this._dialogParams || !this._backup) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<ha-dialog
|
<ha-dialog
|
||||||
@@ -286,24 +287,15 @@ class HassioBackupDialog
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this.hass!.callApi(
|
try {
|
||||||
atLeastVersion(this.hass!.config.version, 2021, 9) ? "DELETE" : "POST",
|
await removeBackup(this.hass!, this._backup!.slug);
|
||||||
`hassio/${
|
if (this._dialogParams!.onDelete) {
|
||||||
atLeastVersion(this.hass!.config.version, 2021, 9)
|
this._dialogParams!.onDelete();
|
||||||
? `backups/${this._backup!.slug}`
|
|
||||||
: `snapshots/${this._backup!.slug}/remove`
|
|
||||||
}`
|
|
||||||
).then(
|
|
||||||
() => {
|
|
||||||
if (this._dialogParams!.onDelete) {
|
|
||||||
this._dialogParams!.onDelete();
|
|
||||||
}
|
|
||||||
this.closeDialog();
|
|
||||||
},
|
|
||||||
(error) => {
|
|
||||||
this._error = error.body.message;
|
|
||||||
}
|
}
|
||||||
);
|
this.closeDialog();
|
||||||
|
} catch (err: any) {
|
||||||
|
this._error = err.body.message;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _downloadClicked() {
|
private async _downloadClicked() {
|
||||||
|
@@ -1,15 +1,15 @@
|
|||||||
import "@material/mwc-button";
|
import "@material/mwc-button";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, query, state } from "lit/decorators";
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
import "../../../../src/components/ha-alert";
|
|
||||||
import "../../../../src/components/buttons/ha-progress-button";
|
import "../../../../src/components/buttons/ha-progress-button";
|
||||||
|
import "../../../../src/components/ha-alert";
|
||||||
import { createCloseHeading } from "../../../../src/components/ha-dialog";
|
import { createCloseHeading } from "../../../../src/components/ha-dialog";
|
||||||
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
|
||||||
import {
|
import {
|
||||||
createHassioFullBackup,
|
createHassioFullBackup,
|
||||||
createHassioPartialBackup,
|
createHassioPartialBackup,
|
||||||
} from "../../../../src/data/hassio/backup";
|
} from "../../../../src/data/hassio/backup";
|
||||||
|
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
||||||
import { showAlertDialog } from "../../../../src/dialogs/generic/show-dialog-box";
|
import { showAlertDialog } from "../../../../src/dialogs/generic/show-dialog-box";
|
||||||
import { haStyle, haStyleDialog } from "../../../../src/resources/styles";
|
import { haStyle, haStyleDialog } from "../../../../src/resources/styles";
|
||||||
import { HomeAssistant } from "../../../../src/types";
|
import { HomeAssistant } from "../../../../src/types";
|
||||||
@@ -42,9 +42,9 @@ class HassioCreateBackupDialog extends LitElement {
|
|||||||
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this._dialogParams) {
|
if (!this._dialogParams) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<ha-dialog
|
<ha-dialog
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import "@material/mwc-list/mwc-list-item";
|
import "@material/mwc-list/mwc-list-item";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
@@ -55,9 +55,9 @@ class HassioDatadiskDialog extends LitElement {
|
|||||||
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.dialogParams) {
|
if (!this.dialogParams) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<ha-dialog
|
<ha-dialog
|
||||||
|
8
hassio/src/dialogs/hardware/dialog-hassio-hardware.ts
Executable file → Normal file
8
hassio/src/dialogs/hardware/dialog-hassio-hardware.ts
Executable file → Normal file
@@ -1,13 +1,13 @@
|
|||||||
import { mdiClose } from "@mdi/js";
|
import { mdiClose } from "@mdi/js";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
import "../../../../src/components/search-input";
|
|
||||||
import { stringCompare } from "../../../../src/common/string/compare";
|
import { stringCompare } from "../../../../src/common/string/compare";
|
||||||
import "../../../../src/components/ha-dialog";
|
import "../../../../src/components/ha-dialog";
|
||||||
import "../../../../src/components/ha-expansion-panel";
|
import "../../../../src/components/ha-expansion-panel";
|
||||||
import "../../../../src/components/ha-icon-button";
|
import "../../../../src/components/ha-icon-button";
|
||||||
|
import "../../../../src/components/search-input";
|
||||||
import { HassioHardwareInfo } from "../../../../src/data/hassio/hardware";
|
import { HassioHardwareInfo } from "../../../../src/data/hassio/hardware";
|
||||||
import { dump } from "../../../../src/resources/js-yaml-dump";
|
import { dump } from "../../../../src/resources/js-yaml-dump";
|
||||||
import { haStyle, haStyleDialog } from "../../../../src/resources/styles";
|
import { haStyle, haStyleDialog } from "../../../../src/resources/styles";
|
||||||
@@ -53,9 +53,9 @@ class HassioHardwareDialog extends LitElement {
|
|||||||
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this._dialogParams) {
|
if (!this._dialogParams) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
const devices = _filterDevices(
|
const devices = _filterDevices(
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import { createCloseHeading } from "../../../../src/components/ha-dialog";
|
import { createCloseHeading } from "../../../../src/components/ha-dialog";
|
||||||
import "../../../../src/components/ha-markdown";
|
import "../../../../src/components/ha-markdown";
|
||||||
@@ -27,9 +27,9 @@ class HassioMarkdownDialog extends LitElement {
|
|||||||
this._opened = false;
|
this._opened = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this._opened) {
|
if (!this._opened) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<ha-dialog
|
<ha-dialog
|
||||||
@@ -50,20 +50,7 @@ class HassioMarkdownDialog extends LitElement {
|
|||||||
haStyleDialog,
|
haStyleDialog,
|
||||||
hassioStyle,
|
hassioStyle,
|
||||||
css`
|
css`
|
||||||
app-toolbar {
|
|
||||||
margin: 0;
|
|
||||||
padding: 0 16px;
|
|
||||||
color: var(--primary-text-color);
|
|
||||||
background-color: var(--secondary-background-color);
|
|
||||||
}
|
|
||||||
app-toolbar [main-title] {
|
|
||||||
margin-left: 16px;
|
|
||||||
}
|
|
||||||
@media all and (max-width: 450px), all and (max-height: 500px) {
|
@media all and (max-width: 450px), all and (max-height: 500px) {
|
||||||
app-toolbar {
|
|
||||||
color: var(--text-primary-color);
|
|
||||||
background-color: var(--primary-color);
|
|
||||||
}
|
|
||||||
ha-markdown {
|
ha-markdown {
|
||||||
padding: 16px;
|
padding: 16px;
|
||||||
}
|
}
|
||||||
|
@@ -5,7 +5,7 @@ import "@material/mwc-tab";
|
|||||||
import "@material/mwc-tab-bar";
|
import "@material/mwc-tab-bar";
|
||||||
import { mdiClose } from "@mdi/js";
|
import { mdiClose } from "@mdi/js";
|
||||||
import { PaperInputElement } from "@polymer/paper-input/paper-input";
|
import { PaperInputElement } from "@polymer/paper-input/paper-input";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import { cache } from "lit/directives/cache";
|
import { cache } from "lit/directives/cache";
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
@@ -83,9 +83,9 @@ export class DialogHassioNetwork
|
|||||||
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this._params || !this._interface) {
|
if (!this._params || !this._interface) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
@@ -597,10 +597,6 @@ export class DialogHassioNetwork
|
|||||||
margin-left: 8px;
|
margin-left: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
:host([rtl]) app-toolbar {
|
|
||||||
direction: rtl;
|
|
||||||
text-align: right;
|
|
||||||
}
|
|
||||||
.container {
|
.container {
|
||||||
padding: 0 8px 4px;
|
padding: 0 8px 4px;
|
||||||
}
|
}
|
||||||
|
@@ -1,11 +1,11 @@
|
|||||||
import "@polymer/paper-tooltip/paper-tooltip";
|
|
||||||
import "@material/mwc-button/mwc-button";
|
import "@material/mwc-button/mwc-button";
|
||||||
import { mdiDelete, mdiDeleteOff } from "@mdi/js";
|
import { mdiDelete, mdiDeleteOff } from "@mdi/js";
|
||||||
import "@polymer/paper-input/paper-input";
|
import "@polymer/paper-input/paper-input";
|
||||||
import type { PaperInputElement } from "@polymer/paper-input/paper-input";
|
import type { PaperInputElement } from "@polymer/paper-input/paper-input";
|
||||||
import "@polymer/paper-item/paper-item";
|
import "@polymer/paper-item/paper-item";
|
||||||
import "@polymer/paper-item/paper-item-body";
|
import "@polymer/paper-item/paper-item-body";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import "@lrnwebcomponents/simple-tooltip/simple-tooltip";
|
||||||
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, query, state } from "lit/decorators";
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
@@ -19,14 +19,14 @@ import {
|
|||||||
HassioAddonRepository,
|
HassioAddonRepository,
|
||||||
} from "../../../../src/data/hassio/addon";
|
} from "../../../../src/data/hassio/addon";
|
||||||
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
||||||
import { haStyle, haStyleDialog } from "../../../../src/resources/styles";
|
|
||||||
import type { HomeAssistant } from "../../../../src/types";
|
|
||||||
import { HassioRepositoryDialogParams } from "./show-dialog-repositories";
|
|
||||||
import {
|
import {
|
||||||
addStoreRepository,
|
addStoreRepository,
|
||||||
fetchStoreRepositories,
|
fetchStoreRepositories,
|
||||||
removeStoreRepository,
|
removeStoreRepository,
|
||||||
} from "../../../../src/data/supervisor/store";
|
} from "../../../../src/data/supervisor/store";
|
||||||
|
import { haStyle, haStyleDialog } from "../../../../src/resources/styles";
|
||||||
|
import type { HomeAssistant } from "../../../../src/types";
|
||||||
|
import { HassioRepositoryDialogParams } from "./show-dialog-repositories";
|
||||||
|
|
||||||
@customElement("dialog-hassio-repositories")
|
@customElement("dialog-hassio-repositories")
|
||||||
class HassioRepositoriesDialog extends LitElement {
|
class HassioRepositoriesDialog extends LitElement {
|
||||||
@@ -82,9 +82,9 @@ class HassioRepositoriesDialog extends LitElement {
|
|||||||
.map((repo) => repo.slug)
|
.map((repo) => repo.slug)
|
||||||
);
|
);
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this._dialogParams?.supervisor || this._repositories === undefined) {
|
if (!this._dialogParams?.supervisor || this._repositories === undefined) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
const repositories = this._filteredRepositories(this._repositories);
|
const repositories = this._filteredRepositories(this._repositories);
|
||||||
const usedRepositories = this._filteredUsedRepositories(
|
const usedRepositories = this._filteredUsedRepositories(
|
||||||
@@ -128,7 +128,7 @@ class HassioRepositoriesDialog extends LitElement {
|
|||||||
@click=${this._removeRepository}
|
@click=${this._removeRepository}
|
||||||
>
|
>
|
||||||
</ha-icon-button>
|
</ha-icon-button>
|
||||||
<paper-tooltip
|
<simple-tooltip
|
||||||
animation-delay="0"
|
animation-delay="0"
|
||||||
position="bottom"
|
position="bottom"
|
||||||
offset="1"
|
offset="1"
|
||||||
@@ -138,7 +138,7 @@ class HassioRepositoriesDialog extends LitElement {
|
|||||||
? "dialog.repositories.used"
|
? "dialog.repositories.used"
|
||||||
: "dialog.repositories.remove"
|
: "dialog.repositories.remove"
|
||||||
)}
|
)}
|
||||||
</paper-tooltip>
|
</simple-tooltip>
|
||||||
</div>
|
</div>
|
||||||
</paper-item>
|
</paper-item>
|
||||||
`
|
`
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import { sanitizeUrl } from "@braintree/sanitize-url";
|
import { sanitizeUrl } from "@braintree/sanitize-url";
|
||||||
import { html, LitElement, TemplateResult } from "lit";
|
import { html, LitElement, TemplateResult, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import { navigate } from "../../src/common/navigate";
|
import { navigate } from "../../src/common/navigate";
|
||||||
import {
|
import {
|
||||||
@@ -101,13 +101,13 @@ class HassioMyRedirect extends LitElement {
|
|||||||
navigate(url, { replace: true });
|
navigate(url, { replace: true });
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (this._error) {
|
if (this._error) {
|
||||||
return html`<hass-error-screen
|
return html`<hass-error-screen
|
||||||
.error=${this._error}
|
.error=${this._error}
|
||||||
></hass-error-screen>`;
|
></hass-error-screen>`;
|
||||||
}
|
}
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
private _createRedirectUrl(redirect: Redirect): string {
|
private _createRedirectUrl(redirect: Redirect): string {
|
||||||
|
@@ -5,7 +5,7 @@ import {
|
|||||||
html,
|
html,
|
||||||
LitElement,
|
LitElement,
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
TemplateResult,
|
nothing,
|
||||||
} from "lit";
|
} from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
@@ -116,12 +116,12 @@ class UpdateAvailableCard extends LitElement {
|
|||||||
storeAddons.find((addon) => addon.slug === slug)
|
storeAddons.find((addon) => addon.slug === slug)
|
||||||
);
|
);
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (
|
if (
|
||||||
!this._updateType ||
|
!this._updateType ||
|
||||||
(this._updateType === "addon" && !this._addonInfo)
|
(this._updateType === "addon" && !this._addonInfo)
|
||||||
) {
|
) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
const changelog = changelogUrl(this._updateType, this._version_latest);
|
const changelog = changelogUrl(this._updateType, this._version_latest);
|
||||||
|
432
package.json
432
package.json
@@ -24,236 +24,240 @@
|
|||||||
"author": "Paulus Schoutsen <Paulus@PaulusSchoutsen.nl> (http://paulusschoutsen.nl)",
|
"author": "Paulus Schoutsen <Paulus@PaulusSchoutsen.nl> (http://paulusschoutsen.nl)",
|
||||||
"license": "Apache-2.0",
|
"license": "Apache-2.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@braintree/sanitize-url": "^6.0.2",
|
"@braintree/sanitize-url": "6.0.2",
|
||||||
"@codemirror/autocomplete": "^6.4.2",
|
"@codemirror/autocomplete": "6.4.2",
|
||||||
"@codemirror/commands": "^6.2.1",
|
"@codemirror/commands": "6.2.2",
|
||||||
"@codemirror/language": "^6.6.0",
|
"@codemirror/language": "6.6.0",
|
||||||
"@codemirror/legacy-modes": "^6.3.1",
|
"@codemirror/legacy-modes": "6.3.2",
|
||||||
"@codemirror/search": "^6.2.3",
|
"@codemirror/search": "6.3.0",
|
||||||
"@codemirror/state": "^6.2.0",
|
"@codemirror/state": "6.2.0",
|
||||||
"@codemirror/view": "^6.9.1",
|
"@codemirror/view": "6.9.3",
|
||||||
"@egjs/hammerjs": "^2.0.17",
|
"@egjs/hammerjs": "2.0.17",
|
||||||
"@formatjs/intl-datetimeformat": "^6.5.1",
|
"@formatjs/intl-datetimeformat": "6.5.1",
|
||||||
"@formatjs/intl-getcanonicallocales": "^2.1.0",
|
"@formatjs/intl-getcanonicallocales": "2.1.0",
|
||||||
"@formatjs/intl-locale": "^3.1.1",
|
"@formatjs/intl-locale": "3.1.1",
|
||||||
"@formatjs/intl-numberformat": "^8.3.5",
|
"@formatjs/intl-numberformat": "8.3.5",
|
||||||
"@formatjs/intl-pluralrules": "^5.1.10",
|
"@formatjs/intl-pluralrules": "5.1.10",
|
||||||
"@formatjs/intl-relativetimeformat": "^11.1.10",
|
"@formatjs/intl-relativetimeformat": "11.1.10",
|
||||||
"@fullcalendar/core": "^6.1.4",
|
"@fullcalendar/core": "6.1.5",
|
||||||
"@fullcalendar/daygrid": "^6.1.4",
|
"@fullcalendar/daygrid": "6.1.5",
|
||||||
"@fullcalendar/interaction": "^6.1.4",
|
"@fullcalendar/interaction": "6.1.5",
|
||||||
"@fullcalendar/list": "^6.1.4",
|
"@fullcalendar/list": "6.1.5",
|
||||||
"@fullcalendar/timegrid": "^6.1.4",
|
"@fullcalendar/timegrid": "6.1.5",
|
||||||
"@lezer/highlight": "^1.1.3",
|
"@lezer/highlight": "1.1.4",
|
||||||
"@lit-labs/motion": "^1.0.3",
|
"@lit-labs/context": "0.3.0",
|
||||||
"@lit-labs/virtualizer": "^1.0.1",
|
"@lit-labs/motion": "1.0.3",
|
||||||
|
"@lit-labs/virtualizer": "1.0.1",
|
||||||
|
"@lrnwebcomponents/simple-tooltip": "4.1.0",
|
||||||
"@material/chips": "=14.0.0-canary.53b3cad2f.0",
|
"@material/chips": "=14.0.0-canary.53b3cad2f.0",
|
||||||
"@material/data-table": "=14.0.0-canary.53b3cad2f.0",
|
"@material/data-table": "=14.0.0-canary.53b3cad2f.0",
|
||||||
"@material/mwc-button": "^0.27.0",
|
"@material/mwc-button": "0.27.0",
|
||||||
"@material/mwc-checkbox": "^0.27.0",
|
"@material/mwc-checkbox": "0.27.0",
|
||||||
"@material/mwc-circular-progress": "^0.27.0",
|
"@material/mwc-circular-progress": "0.27.0",
|
||||||
"@material/mwc-dialog": "^0.27.0",
|
"@material/mwc-dialog": "0.27.0",
|
||||||
"@material/mwc-drawer": "^0.27.0",
|
"@material/mwc-drawer": "0.27.0",
|
||||||
"@material/mwc-fab": "^0.27.0",
|
"@material/mwc-fab": "0.27.0",
|
||||||
"@material/mwc-formfield": "^0.27.0",
|
"@material/mwc-formfield": "0.27.0",
|
||||||
"@material/mwc-icon-button": "^0.27.0",
|
"@material/mwc-icon-button": "0.27.0",
|
||||||
"@material/mwc-linear-progress": "^0.27.0",
|
"@material/mwc-linear-progress": "0.27.0",
|
||||||
"@material/mwc-list": "^0.27.0",
|
"@material/mwc-list": "0.27.0",
|
||||||
"@material/mwc-menu": "^0.27.0",
|
"@material/mwc-menu": "0.27.0",
|
||||||
"@material/mwc-radio": "^0.27.0",
|
"@material/mwc-radio": "0.27.0",
|
||||||
"@material/mwc-ripple": "^0.27.0",
|
"@material/mwc-ripple": "0.27.0",
|
||||||
"@material/mwc-select": "^0.27.0",
|
"@material/mwc-select": "0.27.0",
|
||||||
"@material/mwc-slider": "^0.27.0",
|
"@material/mwc-slider": "0.27.0",
|
||||||
"@material/mwc-switch": "^0.27.0",
|
"@material/mwc-switch": "0.27.0",
|
||||||
"@material/mwc-tab": "^0.27.0",
|
"@material/mwc-tab": "0.27.0",
|
||||||
"@material/mwc-tab-bar": "^0.27.0",
|
"@material/mwc-tab-bar": "0.27.0",
|
||||||
"@material/mwc-textarea": "^0.27.0",
|
"@material/mwc-textarea": "0.27.0",
|
||||||
"@material/mwc-textfield": "^0.27.0",
|
"@material/mwc-textfield": "0.27.0",
|
||||||
"@material/mwc-top-app-bar-fixed": "^0.27.0",
|
"@material/mwc-top-app-bar": "0.27.0",
|
||||||
|
"@material/mwc-top-app-bar-fixed": "0.27.0",
|
||||||
"@material/top-app-bar": "=14.0.0-canary.53b3cad2f.0",
|
"@material/top-app-bar": "=14.0.0-canary.53b3cad2f.0",
|
||||||
"@material/web": "=1.0.0-pre.3",
|
"@material/web": "=1.0.0-pre.4",
|
||||||
"@mdi/js": "7.1.96",
|
"@mdi/js": "7.2.96",
|
||||||
"@mdi/svg": "7.1.96",
|
"@mdi/svg": "7.2.96",
|
||||||
"@polymer/app-layout": "^3.1.0",
|
"@polymer/app-layout": "3.1.0",
|
||||||
"@polymer/iron-flex-layout": "^3.0.1",
|
"@polymer/iron-flex-layout": "3.0.1",
|
||||||
"@polymer/iron-icon": "^3.0.1",
|
"@polymer/iron-input": "3.0.1",
|
||||||
"@polymer/iron-input": "^3.0.1",
|
"@polymer/iron-resizable-behavior": "3.0.1",
|
||||||
"@polymer/iron-resizable-behavior": "^3.0.1",
|
"@polymer/paper-input": "3.2.1",
|
||||||
"@polymer/paper-input": "^3.2.1",
|
"@polymer/paper-item": "3.0.1",
|
||||||
"@polymer/paper-item": "^3.0.1",
|
"@polymer/paper-listbox": "3.0.1",
|
||||||
"@polymer/paper-listbox": "^3.0.1",
|
"@polymer/paper-slider": "3.0.1",
|
||||||
"@polymer/paper-slider": "^3.0.1",
|
"@polymer/paper-tabs": "3.1.0",
|
||||||
"@polymer/paper-styles": "^3.0.1",
|
"@polymer/paper-toast": "3.0.1",
|
||||||
"@polymer/paper-tabs": "^3.1.0",
|
"@polymer/polymer": "3.5.1",
|
||||||
"@polymer/paper-toast": "^3.0.1",
|
|
||||||
"@polymer/paper-tooltip": "^3.0.1",
|
|
||||||
"@polymer/polymer": "3.4.1",
|
|
||||||
"@thomasloven/round-slider": "0.6.0",
|
"@thomasloven/round-slider": "0.6.0",
|
||||||
"@vaadin/combo-box": "^23.3.7",
|
"@vaadin/combo-box": "23.3.9",
|
||||||
"@vaadin/vaadin-themable-mixin": "^23.3.7",
|
"@vaadin/vaadin-themable-mixin": "23.3.9",
|
||||||
"@vibrant/color": "^3.2.1-alpha.1",
|
"@vibrant/color": "3.2.1-alpha.1",
|
||||||
"@vibrant/core": "^3.2.1-alpha.1",
|
"@vibrant/core": "3.2.1-alpha.1",
|
||||||
"@vibrant/quantizer-mmcq": "^3.2.1-alpha.1",
|
"@vibrant/quantizer-mmcq": "3.2.1-alpha.1",
|
||||||
"@vue/web-component-wrapper": "^1.3.0",
|
"@vue/web-component-wrapper": "1.3.0",
|
||||||
"@webcomponents/scoped-custom-element-registry": "^0.0.8",
|
"@webcomponents/scoped-custom-element-registry": "0.0.8",
|
||||||
"@webcomponents/webcomponentsjs": "^2.7.0",
|
"@webcomponents/webcomponentsjs": "2.7.0",
|
||||||
"app-datepicker": "^5.1.0",
|
"app-datepicker": "5.1.1",
|
||||||
"chart.js": "^3.3.2",
|
"chart.js": "3.3.2",
|
||||||
"comlink": "^4.4.1",
|
"comlink": "4.4.1",
|
||||||
"core-js": "^3.28.0",
|
"core-js": "3.29.1",
|
||||||
"cropperjs": "^1.5.13",
|
"cropperjs": "1.5.13",
|
||||||
"date-fns": "^2.29.3",
|
"date-fns": "2.29.3",
|
||||||
"date-fns-tz": "^2.0.0",
|
"date-fns-tz": "2.0.0",
|
||||||
"deep-clone-simple": "^1.1.1",
|
"deep-clone-simple": "1.1.1",
|
||||||
"deep-freeze": "^0.0.1",
|
"deep-freeze": "0.0.1",
|
||||||
"fuse.js": "^6.6.2",
|
"fuse.js": "6.6.2",
|
||||||
"google-timezones-json": "^1.0.2",
|
"google-timezones-json": "1.0.2",
|
||||||
"hls.js": "^1.3.3",
|
"hls.js": "1.3.5",
|
||||||
"home-assistant-js-websocket": "^8.0.1",
|
"home-assistant-js-websocket": "8.0.1",
|
||||||
"idb-keyval": "^6.2.0",
|
"idb-keyval": "6.2.0",
|
||||||
"intl-messageformat": "^10.3.1",
|
"intl-messageformat": "10.3.3",
|
||||||
"js-yaml": "^4.1.0",
|
"js-yaml": "4.1.0",
|
||||||
"leaflet": "^1.9.3",
|
"leaflet": "1.9.3",
|
||||||
"leaflet-draw": "^1.0.4",
|
"leaflet-draw": "1.0.4",
|
||||||
"lit": "^2.6.1",
|
"lit": "2.7.0",
|
||||||
"marked": "^4.2.12",
|
"marked": "4.3.0",
|
||||||
"memoize-one": "^6.0.0",
|
"memoize-one": "6.0.0",
|
||||||
"node-vibrant": "3.2.1-alpha.1",
|
"node-vibrant": "3.2.1-alpha.1",
|
||||||
"proxy-polyfill": "^0.3.2",
|
"proxy-polyfill": "0.3.2",
|
||||||
"punycode": "^2.3.0",
|
"punycode": "2.3.0",
|
||||||
"qr-scanner": "^1.4.2",
|
"qr-scanner": "1.4.2",
|
||||||
"qrcode": "^1.5.1",
|
"qrcode": "1.5.1",
|
||||||
"regenerator-runtime": "^0.13.11",
|
"regenerator-runtime": "0.13.11",
|
||||||
"resize-observer-polyfill": "^1.5.1",
|
"resize-observer-polyfill": "1.5.1",
|
||||||
"roboto-fontface": "^0.10.0",
|
"roboto-fontface": "0.10.0",
|
||||||
"rrule": "^2.7.2",
|
"rrule": "2.7.2",
|
||||||
"sortablejs": "^1.15.0",
|
"sortablejs": "1.15.0",
|
||||||
"superstruct": "^1.0.3",
|
"superstruct": "1.0.3",
|
||||||
"tinykeys": "^1.4.0",
|
"tinykeys": "1.4.0",
|
||||||
"tsparticles-engine": "^2.9.3",
|
"tsparticles-engine": "2.9.3",
|
||||||
"tsparticles-preset-links": "^2.9.3",
|
"tsparticles-preset-links": "2.9.3",
|
||||||
"unfetch": "^5.0.0",
|
"unfetch": "5.0.0",
|
||||||
"vis-data": "^7.1.4",
|
"vis-data": "7.1.6",
|
||||||
"vis-network": "^9.1.4",
|
"vis-network": "9.1.6",
|
||||||
"vue": "^2.7.14",
|
"vue": "2.7.14",
|
||||||
"vue2-daterange-picker": "^0.6.8",
|
"vue2-daterange-picker": "0.6.8",
|
||||||
"weekstart": "^2.0.0",
|
"weekstart": "2.0.0",
|
||||||
"workbox-cacheable-response": "^6.5.4",
|
"workbox-cacheable-response": "6.5.4",
|
||||||
"workbox-core": "^6.5.4",
|
"workbox-core": "6.5.4",
|
||||||
"workbox-expiration": "^6.5.4",
|
"workbox-expiration": "6.5.4",
|
||||||
"workbox-precaching": "^6.5.4",
|
"workbox-precaching": "6.5.4",
|
||||||
"workbox-routing": "^6.5.4",
|
"workbox-routing": "6.5.4",
|
||||||
"workbox-strategies": "^6.5.4",
|
"workbox-strategies": "6.5.4",
|
||||||
"xss": "^1.0.14"
|
"xss": "1.0.14"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "^7.21.0",
|
"@babel/core": "7.21.3",
|
||||||
"@babel/plugin-external-helpers": "^7.18.6",
|
"@babel/plugin-external-helpers": "7.18.6",
|
||||||
"@babel/plugin-proposal-class-properties": "^7.18.6",
|
"@babel/plugin-proposal-class-properties": "7.18.6",
|
||||||
"@babel/plugin-proposal-decorators": "^7.21.0",
|
"@babel/plugin-proposal-decorators": "7.21.0",
|
||||||
"@babel/plugin-proposal-nullish-coalescing-operator": "^7.18.6",
|
"@babel/plugin-proposal-nullish-coalescing-operator": "7.18.6",
|
||||||
"@babel/plugin-proposal-object-rest-spread": "^7.20.7",
|
"@babel/plugin-proposal-object-rest-spread": "7.20.7",
|
||||||
"@babel/plugin-proposal-optional-chaining": "^7.21.0",
|
"@babel/plugin-proposal-optional-chaining": "7.21.0",
|
||||||
"@babel/plugin-syntax-dynamic-import": "^7.8.3",
|
"@babel/plugin-syntax-dynamic-import": "7.8.3",
|
||||||
"@babel/plugin-syntax-import-meta": "^7.10.4",
|
"@babel/plugin-syntax-import-meta": "7.10.4",
|
||||||
"@babel/plugin-syntax-top-level-await": "^7.14.5",
|
"@babel/plugin-syntax-top-level-await": "7.14.5",
|
||||||
"@babel/preset-env": "^7.20.2",
|
"@babel/preset-env": "7.20.2",
|
||||||
"@babel/preset-typescript": "^7.21.0",
|
"@babel/preset-typescript": "7.21.0",
|
||||||
"@koa/cors": "^4.0.0",
|
"@koa/cors": "4.0.0",
|
||||||
"@octokit/auth-oauth-device": "^4.0.4",
|
"@octokit/auth-oauth-device": "4.0.4",
|
||||||
"@octokit/rest": "^19.0.7",
|
"@octokit/rest": "19.0.7",
|
||||||
"@open-wc/dev-server-hmr": "^0.1.3",
|
"@open-wc/dev-server-hmr": "0.1.4",
|
||||||
"@rollup/plugin-babel": "^5.2.1",
|
"@rollup/plugin-babel": "6.0.3",
|
||||||
"@rollup/plugin-commonjs": "^11.1.0",
|
"@rollup/plugin-commonjs": "24.0.1",
|
||||||
"@rollup/plugin-json": "^4.0.3",
|
"@rollup/plugin-json": "6.0.0",
|
||||||
"@rollup/plugin-node-resolve": "^7.1.3",
|
"@rollup/plugin-node-resolve": "15.0.1",
|
||||||
"@rollup/plugin-replace": "^2.3.2",
|
"@rollup/plugin-replace": "5.0.2",
|
||||||
"@types/chromecast-caf-receiver": "5.0.12",
|
"@types/chromecast-caf-receiver": "5.0.12",
|
||||||
"@types/chromecast-caf-sender": "^1.0.5",
|
"@types/chromecast-caf-sender": "1.0.5",
|
||||||
"@types/esprima": "^4",
|
"@types/esprima": "4.0.3",
|
||||||
"@types/glob": "^8",
|
"@types/glob": "8.1.0",
|
||||||
"@types/js-yaml": "^4",
|
"@types/html-minifier-terser": "7.0.0",
|
||||||
"@types/leaflet": "^1",
|
"@types/js-yaml": "4.0.5",
|
||||||
"@types/leaflet-draw": "^1",
|
"@types/leaflet": "1.9.3",
|
||||||
"@types/marked": "^4",
|
"@types/leaflet-draw": "1.0.6",
|
||||||
"@types/mocha": "^10",
|
"@types/marked": "4.0.8",
|
||||||
"@types/qrcode": "^1.5.0",
|
"@types/mocha": "10.0.1",
|
||||||
"@types/sortablejs": "^1",
|
"@types/qrcode": "1.5.0",
|
||||||
"@types/tar": "^6",
|
"@types/serve-handler": "6.1.1",
|
||||||
"@types/webspeechapi": "^0.0.29",
|
"@types/sortablejs": "1.15.1",
|
||||||
"@typescript-eslint/eslint-plugin": "^5.53.0",
|
"@types/tar": "6.1.4",
|
||||||
"@typescript-eslint/parser": "^5.53.0",
|
"@types/webspeechapi": "0.0.29",
|
||||||
"@web/dev-server": "^0.1.35",
|
"@typescript-eslint/eslint-plugin": "5.56.0",
|
||||||
"@web/dev-server-rollup": "^0.2.11",
|
"@typescript-eslint/parser": "5.56.0",
|
||||||
"babel-loader": "^9.1.2",
|
"@web/dev-server": "0.1.36",
|
||||||
"chai": "^4.3.7",
|
"@web/dev-server-rollup": "0.4.0",
|
||||||
"del": "^7.0.0",
|
"babel-loader": "9.1.2",
|
||||||
"eslint": "^8.35.0",
|
"babel-plugin-template-html-minifier": "4.1.0",
|
||||||
"eslint-config-airbnb-base": "^15.0.0",
|
"chai": "4.3.7",
|
||||||
"eslint-config-airbnb-typescript": "^17.0.0",
|
"del": "7.0.0",
|
||||||
"eslint-config-prettier": "^8.6.0",
|
"eslint": "8.36.0",
|
||||||
"eslint-import-resolver-webpack": "^0.13.2",
|
"eslint-config-airbnb-base": "15.0.0",
|
||||||
"eslint-plugin-disable": "^2.0.3",
|
"eslint-config-airbnb-typescript": "17.0.0",
|
||||||
"eslint-plugin-import": "^2.27.5",
|
"eslint-config-prettier": "8.8.0",
|
||||||
"eslint-plugin-lit": "^1.8.2",
|
"eslint-import-resolver-webpack": "0.13.2",
|
||||||
"eslint-plugin-lit-a11y": "^2.3.0",
|
"eslint-plugin-disable": "2.0.3",
|
||||||
"eslint-plugin-unused-imports": "^2.0.0",
|
"eslint-plugin-import": "2.27.5",
|
||||||
"eslint-plugin-wc": "^1.4.0",
|
"eslint-plugin-lit": "1.8.2",
|
||||||
"esprima": "^4.0.1",
|
"eslint-plugin-lit-a11y": "2.4.0",
|
||||||
"fancy-log": "^2.0.0",
|
"eslint-plugin-unused-imports": "2.0.0",
|
||||||
"fs-extra": "^11.1.0",
|
"eslint-plugin-wc": "1.4.0",
|
||||||
"glob": "^8.1.0",
|
"esprima": "4.0.1",
|
||||||
"gulp": "^4.0.2",
|
"fancy-log": "2.0.0",
|
||||||
"gulp-flatmap": "^1.0.2",
|
"fs-extra": "11.1.1",
|
||||||
"gulp-json-transform": "^0.4.8",
|
"glob": "9.3.2",
|
||||||
"gulp-merge-json": "^2.1.2",
|
"gulp": "4.0.2",
|
||||||
"gulp-rename": "^2.0.0",
|
"gulp-flatmap": "1.0.2",
|
||||||
"gulp-zopfli-green": "^6.0.1",
|
"gulp-json-transform": "0.4.8",
|
||||||
"html-minifier": "^4.0.0",
|
"gulp-merge-json": "2.1.2",
|
||||||
"husky": "^8.0.3",
|
"gulp-rename": "2.0.0",
|
||||||
"instant-mocha": "^1.5.0",
|
"gulp-zopfli-green": "6.0.1",
|
||||||
"jszip": "^3.10.1",
|
"html-minifier-terser": "7.1.0",
|
||||||
"lint-staged": "^13.1.2",
|
"husky": "8.0.3",
|
||||||
"lit-analyzer": "^1.2.1",
|
"instant-mocha": "1.5.0",
|
||||||
"lodash.template": "^4.5.0",
|
"jszip": "3.10.1",
|
||||||
"magic-string": "^0.30.0",
|
"lint-staged": "13.2.0",
|
||||||
"map-stream": "^0.0.7",
|
"lit-analyzer": "1.2.1",
|
||||||
"merge-stream": "^2.0.0",
|
"lodash.template": "4.5.0",
|
||||||
"mocha": "^10.2.0",
|
"magic-string": "0.30.0",
|
||||||
"object-hash": "^3.0.0",
|
"map-stream": "0.0.7",
|
||||||
"open": "^8.4.1",
|
"merge-stream": "2.0.0",
|
||||||
"pinst": "^3.0.0",
|
"mocha": "10.2.0",
|
||||||
"prettier": "^2.8.4",
|
"object-hash": "3.0.0",
|
||||||
"require-dir": "^1.2.0",
|
"open": "8.4.2",
|
||||||
"rollup": "^2.8.2",
|
"pinst": "3.0.0",
|
||||||
"rollup-plugin-string": "^3.0.0",
|
"prettier": "2.8.7",
|
||||||
"rollup-plugin-terser": "^5.3.0",
|
"require-dir": "1.2.0",
|
||||||
"rollup-plugin-visualizer": "^5.9.0",
|
"rollup": "2.79.1",
|
||||||
"serve": "^11.3.2",
|
"rollup-plugin-string": "3.0.0",
|
||||||
"sinon": "^15.0.1",
|
"rollup-plugin-terser": "7.0.2",
|
||||||
"source-map-url": "^0.4.1",
|
"rollup-plugin-visualizer": "5.9.0",
|
||||||
"systemjs": "^6.14.0",
|
"serve-handler": "6.1.5",
|
||||||
"tar": "^6.1.13",
|
"sinon": "15.0.2",
|
||||||
"terser-webpack-plugin": "^5.3.6",
|
"source-map-url": "0.4.1",
|
||||||
"ts-lit-plugin": "^1.2.1",
|
"systemjs": "6.14.1",
|
||||||
"typescript": "^4.9.5",
|
"tar": "6.1.13",
|
||||||
"vinyl-buffer": "^1.0.1",
|
"terser-webpack-plugin": "5.3.7",
|
||||||
"vinyl-source-stream": "^2.0.0",
|
"ts-lit-plugin": "1.2.1",
|
||||||
|
"typescript": "4.9.5",
|
||||||
|
"vinyl-buffer": "1.0.1",
|
||||||
|
"vinyl-source-stream": "2.0.0",
|
||||||
"webpack": "=5.72.1",
|
"webpack": "=5.72.1",
|
||||||
"webpack-cli": "^5.0.1",
|
"webpack-cli": "5.0.1",
|
||||||
"webpack-dev-server": "^4.11.1",
|
"webpack-dev-server": "4.13.1",
|
||||||
"webpack-manifest-plugin": "^5.0.0",
|
"webpack-manifest-plugin": "5.0.0",
|
||||||
"webpackbar": "^5.0.2",
|
"webpackbar": "5.0.2",
|
||||||
"workbox-build": "^6.5.4"
|
"workbox-build": "6.5.4"
|
||||||
},
|
},
|
||||||
"_comment": "Polymer 3.2 contained a bug, fixed in https://github.com/Polymer/polymer/pull/5569, add as patch",
|
"_comment": "Polymer 3.2 contained a bug, fixed in https://github.com/Polymer/polymer/pull/5569, add as patch",
|
||||||
"resolutions": {
|
"resolutions": {
|
||||||
"@polymer/polymer": "patch:@polymer/polymer@3.4.1#./.yarn/patches/@polymer/polymer/pr-5569.patch"
|
"@polymer/polymer": "patch:@polymer/polymer@3.5.1#./.yarn/patches/@polymer/polymer/pr-5569.patch",
|
||||||
|
"@material/mwc-button@^0.25.3": "^0.27.0"
|
||||||
},
|
},
|
||||||
"main": "src/home-assistant.js",
|
"main": "src/home-assistant.js",
|
||||||
"prettier": {
|
"prettier": {
|
||||||
"trailingComma": "es5",
|
"trailingComma": "es5",
|
||||||
"arrowParens": "always"
|
"arrowParens": "always"
|
||||||
},
|
},
|
||||||
"packageManager": "yarn@3.3.1"
|
"packageManager": "yarn@3.5.0"
|
||||||
}
|
}
|
||||||
|
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
|
|||||||
|
|
||||||
[project]
|
[project]
|
||||||
name = "home-assistant-frontend"
|
name = "home-assistant-frontend"
|
||||||
version = "20230224.0"
|
version = "20230329.0"
|
||||||
license = {text = "Apache-2.0"}
|
license = {text = "Apache-2.0"}
|
||||||
description = "The Home Assistant frontend"
|
description = "The Home Assistant frontend"
|
||||||
readme = "README.md"
|
readme = "README.md"
|
||||||
|
32
renovate.json
Normal file
32
renovate.json
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
{
|
||||||
|
"$schema": "https://docs.renovatebot.com/renovate-schema.json",
|
||||||
|
"extends": [
|
||||||
|
":ignoreModulesAndTests",
|
||||||
|
":label(dependencies)",
|
||||||
|
":pinVersions",
|
||||||
|
":prConcurrentLimit10",
|
||||||
|
":semanticCommitsDisabled",
|
||||||
|
"group:monorepos",
|
||||||
|
"group:recommended",
|
||||||
|
"npm:unpublishSafe"
|
||||||
|
],
|
||||||
|
"enabledManagers": ["npm"],
|
||||||
|
"postUpdateOptions": ["yarnDedupeHighest"],
|
||||||
|
"lockFileMaintenance": {
|
||||||
|
"description": ["Run after patch releases but before next beta"],
|
||||||
|
"enabled": true,
|
||||||
|
"schedule": ["on the 19th day of the month"]
|
||||||
|
},
|
||||||
|
"packageRules": [
|
||||||
|
{
|
||||||
|
"description": ["MDC packages are pinned to the same version as MWC"],
|
||||||
|
"extends": ["monorepo:material-components-web"],
|
||||||
|
"enabled": false
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"description": ["Vue is only used by date range which is only v2"],
|
||||||
|
"matchPackageNames": ["vue"],
|
||||||
|
"allowedVersions": "< 3"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
@@ -5,8 +5,8 @@ import {
|
|||||||
CSSResultGroup,
|
CSSResultGroup,
|
||||||
html,
|
html,
|
||||||
LitElement,
|
LitElement,
|
||||||
|
nothing,
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
TemplateResult,
|
|
||||||
} from "lit";
|
} from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import "../components/ha-alert";
|
import "../components/ha-alert";
|
||||||
@@ -134,11 +134,11 @@ export class HaAuthFlow extends litLocalizeLiteMixin(LitElement) {
|
|||||||
}, 500);
|
}, 500);
|
||||||
}
|
}
|
||||||
|
|
||||||
private _renderForm(): TemplateResult {
|
private _renderForm() {
|
||||||
switch (this._state) {
|
switch (this._state) {
|
||||||
case "step":
|
case "step":
|
||||||
if (this._step == null) {
|
if (this._step == null) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
${this._renderStep(this._step)}
|
${this._renderStep(this._step)}
|
||||||
@@ -176,11 +176,11 @@ export class HaAuthFlow extends litLocalizeLiteMixin(LitElement) {
|
|||||||
</ha-alert>
|
</ha-alert>
|
||||||
`;
|
`;
|
||||||
default:
|
default:
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private _renderStep(step: DataEntryFlowStep): TemplateResult {
|
private _renderStep(step: DataEntryFlowStep) {
|
||||||
switch (step.type) {
|
switch (step.type) {
|
||||||
case "abort":
|
case "abort":
|
||||||
return html`
|
return html`
|
||||||
@@ -202,7 +202,7 @@ export class HaAuthFlow extends litLocalizeLiteMixin(LitElement) {
|
|||||||
.content=${this._computeStepDescription(step)}
|
.content=${this._computeStepDescription(step)}
|
||||||
></ha-markdown>
|
></ha-markdown>
|
||||||
`
|
`
|
||||||
: html``}
|
: nothing}
|
||||||
<ha-form
|
<ha-form
|
||||||
.data=${this._stepData}
|
.data=${this._stepData}
|
||||||
.schema=${autocompleteLoginFields(step.data_schema)}
|
.schema=${autocompleteLoginFields(step.data_schema)}
|
||||||
@@ -228,7 +228,7 @@ export class HaAuthFlow extends litLocalizeLiteMixin(LitElement) {
|
|||||||
: ""}
|
: ""}
|
||||||
`;
|
`;
|
||||||
default:
|
default:
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1,9 +1,9 @@
|
|||||||
import "@polymer/paper-item/paper-item";
|
import "@material/mwc-list";
|
||||||
import "@polymer/paper-item/paper-item-body";
|
|
||||||
import { css, html, LitElement } from "lit";
|
import { css, html, LitElement } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
import "../components/ha-icon-next";
|
import "../components/ha-icon-next";
|
||||||
|
import "../components/ha-list-item";
|
||||||
import { AuthProvider } from "../data/auth";
|
import { AuthProvider } from "../data/auth";
|
||||||
import { litLocalizeLiteMixin } from "../mixins/lit-localize-lite-mixin";
|
import { litLocalizeLiteMixin } from "../mixins/lit-localize-lite-mixin";
|
||||||
|
|
||||||
@@ -20,18 +20,21 @@ export class HaPickAuthProvider extends litLocalizeLiteMixin(LitElement) {
|
|||||||
protected render() {
|
protected render() {
|
||||||
return html`
|
return html`
|
||||||
<p>${this.localize("ui.panel.page-authorize.pick_auth_provider")}:</p>
|
<p>${this.localize("ui.panel.page-authorize.pick_auth_provider")}:</p>
|
||||||
${this.authProviders.map(
|
<mwc-list>
|
||||||
(provider) => html`
|
${this.authProviders.map(
|
||||||
<paper-item
|
(provider) => html`
|
||||||
role="button"
|
<ha-list-item
|
||||||
.auth_provider=${provider}
|
hasMeta
|
||||||
@click=${this._handlePick}
|
role="button"
|
||||||
>
|
.auth_provider=${provider}
|
||||||
<paper-item-body>${provider.name}</paper-item-body>
|
@click=${this._handlePick}
|
||||||
<ha-icon-next></ha-icon-next>
|
>
|
||||||
</paper-item>
|
${provider.name}
|
||||||
`
|
<ha-icon-next slot="meta"></ha-icon-next>
|
||||||
)}
|
</ha-list-item>
|
||||||
|
`
|
||||||
|
)}</mwc-list
|
||||||
|
>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -40,11 +43,12 @@ export class HaPickAuthProvider extends litLocalizeLiteMixin(LitElement) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
static styles = css`
|
static styles = css`
|
||||||
paper-item {
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
p {
|
p {
|
||||||
margin-top: 0;
|
margin-top: 0;
|
||||||
}
|
}
|
||||||
|
mwc-list {
|
||||||
|
margin: 0 -16px;
|
||||||
|
--mdc-list-side-padding: 16px;
|
||||||
|
}
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
@@ -1,7 +1,6 @@
|
|||||||
/** Constants to be used in the frontend. */
|
/** Constants to be used in the frontend. */
|
||||||
|
|
||||||
import {
|
import {
|
||||||
mdiAccount,
|
|
||||||
mdiAirFilter,
|
mdiAirFilter,
|
||||||
mdiAlert,
|
mdiAlert,
|
||||||
mdiAngleAcute,
|
mdiAngleAcute,
|
||||||
@@ -48,7 +47,6 @@ import {
|
|||||||
mdiProgressClock,
|
mdiProgressClock,
|
||||||
mdiRayVertex,
|
mdiRayVertex,
|
||||||
mdiRemote,
|
mdiRemote,
|
||||||
mdiRobot,
|
|
||||||
mdiRobotVacuum,
|
mdiRobotVacuum,
|
||||||
mdiScriptText,
|
mdiScriptText,
|
||||||
mdiSineWave,
|
mdiSineWave,
|
||||||
@@ -59,15 +57,12 @@ import {
|
|||||||
mdiThermostat,
|
mdiThermostat,
|
||||||
mdiTimerOutline,
|
mdiTimerOutline,
|
||||||
mdiTransmissionTower,
|
mdiTransmissionTower,
|
||||||
mdiVideo,
|
|
||||||
mdiWater,
|
mdiWater,
|
||||||
mdiWaterPercent,
|
mdiWaterPercent,
|
||||||
mdiWeatherCloudy,
|
|
||||||
mdiWeatherPouring,
|
mdiWeatherPouring,
|
||||||
mdiWeatherRainy,
|
mdiWeatherRainy,
|
||||||
mdiWeatherWindy,
|
mdiWeatherWindy,
|
||||||
mdiWeight,
|
mdiWeight,
|
||||||
mdiWhiteBalanceSunny,
|
|
||||||
mdiWifi,
|
mdiWifi,
|
||||||
} from "@mdi/js";
|
} from "@mdi/js";
|
||||||
|
|
||||||
@@ -82,9 +77,7 @@ export const DEFAULT_DOMAIN_ICON = mdiBookmark;
|
|||||||
export const FIXED_DOMAIN_ICONS = {
|
export const FIXED_DOMAIN_ICONS = {
|
||||||
alert: mdiAlert,
|
alert: mdiAlert,
|
||||||
air_quality: mdiAirFilter,
|
air_quality: mdiAirFilter,
|
||||||
automation: mdiRobot,
|
|
||||||
calendar: mdiCalendar,
|
calendar: mdiCalendar,
|
||||||
camera: mdiVideo,
|
|
||||||
climate: mdiThermostat,
|
climate: mdiThermostat,
|
||||||
configurator: mdiCog,
|
configurator: mdiCog,
|
||||||
conversation: mdiMicrophoneMessage,
|
conversation: mdiMicrophoneMessage,
|
||||||
@@ -105,7 +98,6 @@ export const FIXED_DOMAIN_ICONS = {
|
|||||||
notify: mdiCommentAlert,
|
notify: mdiCommentAlert,
|
||||||
number: mdiRayVertex,
|
number: mdiRayVertex,
|
||||||
persistent_notification: mdiBell,
|
persistent_notification: mdiBell,
|
||||||
person: mdiAccount,
|
|
||||||
plant: mdiFlower,
|
plant: mdiFlower,
|
||||||
proximity: mdiAppleSafari,
|
proximity: mdiAppleSafari,
|
||||||
remote: mdiRemote,
|
remote: mdiRemote,
|
||||||
@@ -116,13 +108,10 @@ export const FIXED_DOMAIN_ICONS = {
|
|||||||
sensor: mdiEye,
|
sensor: mdiEye,
|
||||||
siren: mdiBullhorn,
|
siren: mdiBullhorn,
|
||||||
simple_alarm: mdiBell,
|
simple_alarm: mdiBell,
|
||||||
sun: mdiWhiteBalanceSunny,
|
|
||||||
text: mdiFormTextbox,
|
text: mdiFormTextbox,
|
||||||
timer: mdiTimerOutline,
|
timer: mdiTimerOutline,
|
||||||
updater: mdiCloudUpload,
|
updater: mdiCloudUpload,
|
||||||
vacuum: mdiRobotVacuum,
|
vacuum: mdiRobotVacuum,
|
||||||
water_heater: mdiThermometer,
|
|
||||||
weather: mdiWeatherCloudy,
|
|
||||||
zone: mdiMapMarkerRadius,
|
zone: mdiMapMarkerRadius,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@@ -1,16 +1,19 @@
|
|||||||
import secondsToDuration from "./seconds_to_duration";
|
import millisecondsToDuration from "./milliseconds_to_duration";
|
||||||
|
|
||||||
const DAY_IN_SECONDS = 86400;
|
const DAY_IN_MILLISECONDS = 86400000;
|
||||||
const HOUR_IN_SECONDS = 3600;
|
const HOUR_IN_MILLISECONDS = 3600000;
|
||||||
const MINUTE_IN_SECONDS = 60;
|
const MINUTE_IN_MILLISECONDS = 60000;
|
||||||
|
const SECOND_IN_MILLISECONDS = 1000;
|
||||||
|
|
||||||
export const UNIT_TO_SECOND_CONVERT = {
|
export const UNIT_TO_MILLISECOND_CONVERT = {
|
||||||
s: 1,
|
ms: 1,
|
||||||
min: MINUTE_IN_SECONDS,
|
s: SECOND_IN_MILLISECONDS,
|
||||||
h: HOUR_IN_SECONDS,
|
min: MINUTE_IN_MILLISECONDS,
|
||||||
d: DAY_IN_SECONDS,
|
h: HOUR_IN_MILLISECONDS,
|
||||||
|
d: DAY_IN_MILLISECONDS,
|
||||||
};
|
};
|
||||||
|
|
||||||
export const formatDuration = (duration: string, units: string): string =>
|
export const formatDuration = (duration: string, units: string): string =>
|
||||||
secondsToDuration(parseFloat(duration) * UNIT_TO_SECOND_CONVERT[units]) ||
|
millisecondsToDuration(
|
||||||
"0";
|
parseFloat(duration) * UNIT_TO_MILLISECOND_CONVERT[units]
|
||||||
|
) || "0";
|
||||||
|
25
src/common/datetime/milliseconds_to_duration.ts
Normal file
25
src/common/datetime/milliseconds_to_duration.ts
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
const leftPad = (num: number, digits = 2) => {
|
||||||
|
let paddedNum = "" + num;
|
||||||
|
for (let i = 1; i < digits; i++) {
|
||||||
|
paddedNum = parseInt(paddedNum) < 10 ** i ? `0${paddedNum}` : paddedNum;
|
||||||
|
}
|
||||||
|
return paddedNum;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default function millisecondsToDuration(d: number) {
|
||||||
|
const h = Math.floor(d / 1000 / 3600);
|
||||||
|
const m = Math.floor(((d / 1000) % 3600) / 60);
|
||||||
|
const s = Math.floor(((d / 1000) % 3600) % 60);
|
||||||
|
const ms = Math.floor(d % 1000);
|
||||||
|
|
||||||
|
if (h > 0) {
|
||||||
|
return `${h}:${leftPad(m)}:${leftPad(s)}`;
|
||||||
|
}
|
||||||
|
if (m > 0) {
|
||||||
|
return `${m}:${leftPad(s)}`;
|
||||||
|
}
|
||||||
|
if (s > 0 || ms > 0) {
|
||||||
|
return `${s}${ms > 0 ? `.${leftPad(ms, 3)}` : ``}`;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
111
src/common/decorators/transform.ts
Normal file
111
src/common/decorators/transform.ts
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
import { PropertyDeclaration, PropertyValues, ReactiveElement } from "lit";
|
||||||
|
import { ClassElement } from "../../types";
|
||||||
|
import { shallowEqual } from "../util/shallow-equal";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Transform function type.
|
||||||
|
*/
|
||||||
|
export interface Transformer<T = any, V = any> {
|
||||||
|
(value: V): T;
|
||||||
|
}
|
||||||
|
|
||||||
|
type ReactiveTransformElement = ReactiveElement & {
|
||||||
|
_transformers: Map<PropertyKey, Transformer>;
|
||||||
|
_watching: Map<PropertyKey, Set<PropertyKey>>;
|
||||||
|
};
|
||||||
|
|
||||||
|
type ReactiveElementClassWithTransformers = typeof ReactiveElement & {
|
||||||
|
prototype: ReactiveTransformElement;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies an tranformer callback that is run when the value of the decorated property, or any of the properties in the watching array, changes.
|
||||||
|
* The result of the tranformer is assigned to the decorated property.
|
||||||
|
* The tranformer receives the current as arguments.
|
||||||
|
*/
|
||||||
|
export const transform =
|
||||||
|
<T, V>(config: {
|
||||||
|
transformer: Transformer<T, V>;
|
||||||
|
watch?: PropertyKey[];
|
||||||
|
propertyOptions?: PropertyDeclaration;
|
||||||
|
}): any =>
|
||||||
|
(clsElement: ClassElement) => {
|
||||||
|
const key = String(clsElement.key);
|
||||||
|
return {
|
||||||
|
...clsElement,
|
||||||
|
kind: "method",
|
||||||
|
descriptor: {
|
||||||
|
set(this: ReactiveTransformElement, value: V) {
|
||||||
|
const oldValue = this[`__transform_${key}`];
|
||||||
|
const trnsformr: Transformer<T, V> | undefined =
|
||||||
|
this._transformers.get(key);
|
||||||
|
if (trnsformr) {
|
||||||
|
this[`__transform_${key}`] = trnsformr.call(this, value);
|
||||||
|
} else {
|
||||||
|
this[`__transform_${key}`] = value;
|
||||||
|
}
|
||||||
|
this[`__original_${key}`] = value;
|
||||||
|
this.requestUpdate(key, oldValue);
|
||||||
|
},
|
||||||
|
get(): T {
|
||||||
|
return this[`__transform_${key}`];
|
||||||
|
},
|
||||||
|
enumerable: true,
|
||||||
|
configurable: true,
|
||||||
|
},
|
||||||
|
finisher(cls: ReactiveElementClassWithTransformers) {
|
||||||
|
// if we haven't wrapped `willUpdate` in this class, do so
|
||||||
|
if (!cls.prototype._transformers) {
|
||||||
|
cls.prototype._transformers = new Map<PropertyKey, Transformer>();
|
||||||
|
cls.prototype._watching = new Map<PropertyKey, Set<PropertyKey>>();
|
||||||
|
// @ts-ignore
|
||||||
|
const userWillUpdate = cls.prototype.willUpdate;
|
||||||
|
// @ts-ignore
|
||||||
|
cls.prototype.willUpdate = function (
|
||||||
|
this: ReactiveTransformElement,
|
||||||
|
changedProperties: PropertyValues
|
||||||
|
) {
|
||||||
|
userWillUpdate.call(this, changedProperties);
|
||||||
|
const keys = new Set<PropertyKey>();
|
||||||
|
changedProperties.forEach((_v, k) => {
|
||||||
|
const watchers = this._watching;
|
||||||
|
const ks: Set<PropertyKey> | undefined = watchers.get(k);
|
||||||
|
if (ks !== undefined) {
|
||||||
|
ks.forEach((wk) => keys.add(wk));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
keys.forEach((k) => {
|
||||||
|
// trigger setter
|
||||||
|
this[k] = this[`__original_${String(k)}`];
|
||||||
|
});
|
||||||
|
};
|
||||||
|
// clone any existing observers (superclasses)
|
||||||
|
// eslint-disable-next-line no-prototype-builtins
|
||||||
|
} else if (!cls.prototype.hasOwnProperty("_transformers")) {
|
||||||
|
const tranformers = cls.prototype._transformers;
|
||||||
|
cls.prototype._transformers = new Map();
|
||||||
|
tranformers.forEach((v: any, k: PropertyKey) =>
|
||||||
|
cls.prototype._transformers.set(k, v)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
// set this method
|
||||||
|
cls.prototype._transformers.set(clsElement.key, config.transformer);
|
||||||
|
if (config.watch) {
|
||||||
|
// store watchers
|
||||||
|
config.watch.forEach((k) => {
|
||||||
|
let curWatch = cls.prototype._watching.get(k);
|
||||||
|
if (!curWatch) {
|
||||||
|
curWatch = new Set();
|
||||||
|
cls.prototype._watching.set(k, curWatch);
|
||||||
|
}
|
||||||
|
curWatch.add(clsElement.key);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
cls.createProperty(clsElement.key, {
|
||||||
|
noAccessor: true,
|
||||||
|
hasChanged: (v: any, o: any) => !shallowEqual(v, o),
|
||||||
|
...config.propertyOptions,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
};
|
||||||
|
};
|
@@ -93,7 +93,7 @@ export const applyThemesOnElement = (
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Nothing was changed
|
// Nothing was changed
|
||||||
if (element._themes?.cacheKey === cacheKey) {
|
if (element.__themes?.cacheKey === cacheKey) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -119,7 +119,7 @@ export const applyThemesOnElement = (
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!element._themes?.keys && !Object.keys(themeRules).length) {
|
if (!element.__themes?.keys && !Object.keys(themeRules).length) {
|
||||||
// No styles to reset, and no styles to set
|
// No styles to reset, and no styles to set
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -130,8 +130,8 @@ export const applyThemesOnElement = (
|
|||||||
: undefined;
|
: undefined;
|
||||||
|
|
||||||
// Add previous set keys to reset them, and new theme
|
// Add previous set keys to reset them, and new theme
|
||||||
const styles = { ...element._themes?.keys, ...newTheme?.styles };
|
const styles = { ...element.__themes?.keys, ...newTheme?.styles };
|
||||||
element._themes = { cacheKey, keys: newTheme?.keys };
|
element.__themes = { cacheKey, keys: newTheme?.keys };
|
||||||
|
|
||||||
// Set and/or reset styles
|
// Set and/or reset styles
|
||||||
if (element.updateStyles) {
|
if (element.updateStyles) {
|
||||||
|
@@ -1,30 +1,126 @@
|
|||||||
import { HassEntity } from "home-assistant-js-websocket";
|
import { HassEntity } from "home-assistant-js-websocket";
|
||||||
|
import { html, TemplateResult } from "lit";
|
||||||
|
import { until } from "lit/directives/until";
|
||||||
import { EntityRegistryDisplayEntry } from "../../data/entity_registry";
|
import { EntityRegistryDisplayEntry } from "../../data/entity_registry";
|
||||||
import { HomeAssistant } from "../../types";
|
import { HomeAssistant } from "../../types";
|
||||||
|
import checkValidDate from "../datetime/check_valid_date";
|
||||||
|
import { formatDate } from "../datetime/format_date";
|
||||||
|
import { formatDateTimeWithSeconds } from "../datetime/format_date_time";
|
||||||
|
import { formatNumber } from "../number/format_number";
|
||||||
|
import { capitalizeFirstLetter } from "../string/capitalize-first-letter";
|
||||||
|
import { isDate } from "../string/is_date";
|
||||||
|
import { isTimestamp } from "../string/is_timestamp";
|
||||||
import { LocalizeFunc } from "../translations/localize";
|
import { LocalizeFunc } from "../translations/localize";
|
||||||
import { computeDomain } from "./compute_domain";
|
import { computeDomain } from "./compute_domain";
|
||||||
|
import { FrontendLocaleData } from "../../data/translation";
|
||||||
|
|
||||||
|
let jsYamlPromise: Promise<typeof import("../../resources/js-yaml-dump")>;
|
||||||
|
|
||||||
export const computeAttributeValueDisplay = (
|
export const computeAttributeValueDisplay = (
|
||||||
localize: LocalizeFunc,
|
localize: LocalizeFunc,
|
||||||
stateObj: HassEntity,
|
stateObj: HassEntity,
|
||||||
|
locale: FrontendLocaleData,
|
||||||
entities: HomeAssistant["entities"],
|
entities: HomeAssistant["entities"],
|
||||||
attribute: string,
|
attribute: string,
|
||||||
value?: any
|
value?: any
|
||||||
): string => {
|
): string | TemplateResult => {
|
||||||
const entityId = stateObj.entity_id;
|
|
||||||
const attributeValue =
|
const attributeValue =
|
||||||
value !== undefined ? value : stateObj.attributes[attribute];
|
value !== undefined ? value : stateObj.attributes[attribute];
|
||||||
|
|
||||||
|
// Null value, the state is unknown
|
||||||
|
if (attributeValue === null) {
|
||||||
|
return localize("state.default.unknown");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Number value, return formatted number
|
||||||
|
if (typeof attributeValue === "number") {
|
||||||
|
return formatNumber(attributeValue, locale);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Special handling in case this is a string with an known format
|
||||||
|
if (typeof attributeValue === "string") {
|
||||||
|
// URL handling
|
||||||
|
if (attributeValue.startsWith("http")) {
|
||||||
|
try {
|
||||||
|
// If invalid URL, exception will be raised
|
||||||
|
const url = new URL(attributeValue);
|
||||||
|
if (url.protocol === "http:" || url.protocol === "https:")
|
||||||
|
return html`<a target="_blank" rel="noreferrer" href=${value}
|
||||||
|
>${attributeValue}</a
|
||||||
|
>`;
|
||||||
|
} catch (_) {
|
||||||
|
// Nothing to do here
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Date handling
|
||||||
|
if (isDate(attributeValue, true)) {
|
||||||
|
// Timestamp handling
|
||||||
|
if (isTimestamp(attributeValue)) {
|
||||||
|
const date = new Date(attributeValue);
|
||||||
|
if (checkValidDate(date)) {
|
||||||
|
return formatDateTimeWithSeconds(date, locale);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Value was not a timestamp, so only do date formatting
|
||||||
|
const date = new Date(attributeValue);
|
||||||
|
if (checkValidDate(date)) {
|
||||||
|
return formatDate(date, locale);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Values are objects, render object
|
||||||
|
if (
|
||||||
|
(Array.isArray(attributeValue) &&
|
||||||
|
attributeValue.some((val) => val instanceof Object)) ||
|
||||||
|
(!Array.isArray(attributeValue) && attributeValue instanceof Object)
|
||||||
|
) {
|
||||||
|
if (!jsYamlPromise) {
|
||||||
|
jsYamlPromise = import("../../resources/js-yaml-dump");
|
||||||
|
}
|
||||||
|
const yaml = jsYamlPromise.then((jsYaml) => jsYaml.dump(attributeValue));
|
||||||
|
return html`<pre>${until(yaml, "")}</pre>`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// If this is an array, try to determine the display value for each item
|
||||||
|
if (Array.isArray(attributeValue)) {
|
||||||
|
return attributeValue
|
||||||
|
.map((item) =>
|
||||||
|
computeAttributeValueDisplay(
|
||||||
|
localize,
|
||||||
|
stateObj,
|
||||||
|
locale,
|
||||||
|
entities,
|
||||||
|
attribute,
|
||||||
|
item
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.join(", ");
|
||||||
|
}
|
||||||
|
|
||||||
|
// We've explored all known value handling, so now we'll try to find a
|
||||||
|
// translation for the value.
|
||||||
|
const entityId = stateObj.entity_id;
|
||||||
const domain = computeDomain(entityId);
|
const domain = computeDomain(entityId);
|
||||||
const entity = entities[entityId] as EntityRegistryDisplayEntry | undefined;
|
const deviceClass = stateObj.attributes.device_class;
|
||||||
const translationKey = entity?.translation_key;
|
const registryEntry = entities[entityId] as
|
||||||
|
| EntityRegistryDisplayEntry
|
||||||
|
| undefined;
|
||||||
|
const translationKey = registryEntry?.translation_key;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
(translationKey &&
|
(translationKey &&
|
||||||
localize(
|
localize(
|
||||||
`component.${entity.platform}.entity.${domain}.${translationKey}.state_attributes.${attribute}.state.${attributeValue}`
|
`component.${registryEntry.platform}.entity.${domain}.${translationKey}.state_attributes.${attribute}.state.${attributeValue}`
|
||||||
|
)) ||
|
||||||
|
(deviceClass &&
|
||||||
|
localize(
|
||||||
|
`component.${domain}.entity_component.${deviceClass}.state_attributes.${attribute}.state.${attributeValue}`
|
||||||
)) ||
|
)) ||
|
||||||
localize(
|
localize(
|
||||||
`component.${domain}.state_attributes._.${attribute}.state.${attributeValue}`
|
`component.${domain}.entity_component._.state_attributes.${attribute}.state.${attributeValue}`
|
||||||
) ||
|
) ||
|
||||||
attributeValue
|
attributeValue
|
||||||
);
|
);
|
||||||
@@ -37,6 +133,7 @@ export const computeAttributeNameDisplay = (
|
|||||||
attribute: string
|
attribute: string
|
||||||
): string => {
|
): string => {
|
||||||
const entityId = stateObj.entity_id;
|
const entityId = stateObj.entity_id;
|
||||||
|
const deviceClass = stateObj.attributes.device_class;
|
||||||
const domain = computeDomain(entityId);
|
const domain = computeDomain(entityId);
|
||||||
const entity = entities[entityId] as EntityRegistryDisplayEntry | undefined;
|
const entity = entities[entityId] as EntityRegistryDisplayEntry | undefined;
|
||||||
const translationKey = entity?.translation_key;
|
const translationKey = entity?.translation_key;
|
||||||
@@ -46,7 +143,20 @@ export const computeAttributeNameDisplay = (
|
|||||||
localize(
|
localize(
|
||||||
`component.${entity.platform}.entity.${domain}.${translationKey}.state_attributes.${attribute}.name`
|
`component.${entity.platform}.entity.${domain}.${translationKey}.state_attributes.${attribute}.name`
|
||||||
)) ||
|
)) ||
|
||||||
localize(`component.${domain}.state_attributes._.${attribute}.name`) ||
|
(deviceClass &&
|
||||||
attribute
|
localize(
|
||||||
|
`component.${domain}.entity_component.${deviceClass}.state_attributes.${attribute}.name`
|
||||||
|
)) ||
|
||||||
|
localize(
|
||||||
|
`component.${domain}.entity_component._.state_attributes.${attribute}.name`
|
||||||
|
) ||
|
||||||
|
capitalizeFirstLetter(
|
||||||
|
attribute
|
||||||
|
.replace(/_/g, " ")
|
||||||
|
.replace(/\bid\b/g, "ID")
|
||||||
|
.replace(/\bip\b/g, "IP")
|
||||||
|
.replace(/\bmac\b/g, "MAC")
|
||||||
|
.replace(/\bgps\b/g, "GPS")
|
||||||
|
)
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@@ -7,7 +7,10 @@ import {
|
|||||||
UPDATE_SUPPORT_PROGRESS,
|
UPDATE_SUPPORT_PROGRESS,
|
||||||
} from "../../data/update";
|
} from "../../data/update";
|
||||||
import { HomeAssistant } from "../../types";
|
import { HomeAssistant } from "../../types";
|
||||||
import { formatDuration, UNIT_TO_SECOND_CONVERT } from "../datetime/duration";
|
import {
|
||||||
|
formatDuration,
|
||||||
|
UNIT_TO_MILLISECOND_CONVERT,
|
||||||
|
} from "../datetime/duration";
|
||||||
import { formatDate } from "../datetime/format_date";
|
import { formatDate } from "../datetime/format_date";
|
||||||
import { formatDateTime } from "../datetime/format_date_time";
|
import { formatDateTime } from "../datetime/format_date_time";
|
||||||
import { formatTime } from "../datetime/format_time";
|
import { formatTime } from "../datetime/format_time";
|
||||||
@@ -57,7 +60,7 @@ export const computeStateDisplayFromEntityAttributes = (
|
|||||||
if (
|
if (
|
||||||
attributes.device_class === "duration" &&
|
attributes.device_class === "duration" &&
|
||||||
attributes.unit_of_measurement &&
|
attributes.unit_of_measurement &&
|
||||||
UNIT_TO_SECOND_CONVERT[attributes.unit_of_measurement]
|
UNIT_TO_MILLISECOND_CONVERT[attributes.unit_of_measurement]
|
||||||
) {
|
) {
|
||||||
try {
|
try {
|
||||||
return formatDuration(state, attributes.unit_of_measurement);
|
return formatDuration(state, attributes.unit_of_measurement);
|
||||||
@@ -71,6 +74,11 @@ export const computeStateDisplayFromEntityAttributes = (
|
|||||||
style: "currency",
|
style: "currency",
|
||||||
currency: attributes.unit_of_measurement,
|
currency: attributes.unit_of_measurement,
|
||||||
minimumFractionDigits: 2,
|
minimumFractionDigits: 2,
|
||||||
|
// Override monetary options with number format
|
||||||
|
...getNumberFormatOptions(
|
||||||
|
{ state, attributes } as HassEntity,
|
||||||
|
entity
|
||||||
|
),
|
||||||
});
|
});
|
||||||
} catch (_err) {
|
} catch (_err) {
|
||||||
// fallback to default
|
// fallback to default
|
||||||
@@ -209,10 +217,10 @@ export const computeStateDisplayFromEntityAttributes = (
|
|||||||
// Return device class translation
|
// Return device class translation
|
||||||
(attributes.device_class &&
|
(attributes.device_class &&
|
||||||
localize(
|
localize(
|
||||||
`component.${domain}.state.${attributes.device_class}.${state}`
|
`component.${domain}.entity_component.${attributes.device_class}.state.${state}`
|
||||||
)) ||
|
)) ||
|
||||||
// Return default translation
|
// Return default translation
|
||||||
localize(`component.${domain}.state._.${state}`) ||
|
localize(`component.${domain}.entity_component._.state.${state}`) ||
|
||||||
// We don't know! Return the raw state.
|
// We don't know! Return the raw state.
|
||||||
state
|
state
|
||||||
);
|
);
|
||||||
|
@@ -30,6 +30,8 @@ import {
|
|||||||
mdiPowerPlug,
|
mdiPowerPlug,
|
||||||
mdiPowerPlugOff,
|
mdiPowerPlugOff,
|
||||||
mdiRestart,
|
mdiRestart,
|
||||||
|
mdiRobot,
|
||||||
|
mdiRobotOff,
|
||||||
mdiSpeaker,
|
mdiSpeaker,
|
||||||
mdiSpeakerOff,
|
mdiSpeakerOff,
|
||||||
mdiSpeakerPause,
|
mdiSpeakerPause,
|
||||||
@@ -41,7 +43,12 @@ import {
|
|||||||
mdiTelevisionPlay,
|
mdiTelevisionPlay,
|
||||||
mdiToggleSwitchVariant,
|
mdiToggleSwitchVariant,
|
||||||
mdiToggleSwitchVariantOff,
|
mdiToggleSwitchVariantOff,
|
||||||
|
mdiVideo,
|
||||||
|
mdiVideoOff,
|
||||||
|
mdiWaterBoiler,
|
||||||
|
mdiWaterBoilerOff,
|
||||||
mdiWeatherNight,
|
mdiWeatherNight,
|
||||||
|
mdiWhiteBalanceSunny,
|
||||||
} from "@mdi/js";
|
} from "@mdi/js";
|
||||||
import { HassEntity } from "home-assistant-js-websocket";
|
import { HassEntity } from "home-assistant-js-websocket";
|
||||||
import { UpdateEntity, updateIsInstalling } from "../../data/update";
|
import { UpdateEntity, updateIsInstalling } from "../../data/update";
|
||||||
@@ -83,6 +90,9 @@ export const domainIconWithoutDefault = (
|
|||||||
case "alarm_control_panel":
|
case "alarm_control_panel":
|
||||||
return alarmPanelIcon(compareState);
|
return alarmPanelIcon(compareState);
|
||||||
|
|
||||||
|
case "automation":
|
||||||
|
return compareState === "off" ? mdiRobotOff : mdiRobot;
|
||||||
|
|
||||||
case "binary_sensor":
|
case "binary_sensor":
|
||||||
return binarySensorIcon(compareState, stateObj);
|
return binarySensorIcon(compareState, stateObj);
|
||||||
|
|
||||||
@@ -96,6 +106,9 @@ export const domainIconWithoutDefault = (
|
|||||||
return mdiGestureTapButton;
|
return mdiGestureTapButton;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
case "camera":
|
||||||
|
return compareState === "off" ? mdiVideoOff : mdiVideo;
|
||||||
|
|
||||||
case "cover":
|
case "cover":
|
||||||
return coverIcon(compareState, stateObj);
|
return coverIcon(compareState, stateObj);
|
||||||
|
|
||||||
@@ -221,7 +234,7 @@ export const domainIconWithoutDefault = (
|
|||||||
|
|
||||||
case "sun":
|
case "sun":
|
||||||
return stateObj?.state === "above_horizon"
|
return stateObj?.state === "above_horizon"
|
||||||
? FIXED_DOMAIN_ICONS[domain]
|
? mdiWhiteBalanceSunny
|
||||||
: mdiWeatherNight;
|
: mdiWeatherNight;
|
||||||
|
|
||||||
case "switch_as_x":
|
case "switch_as_x":
|
||||||
@@ -237,6 +250,9 @@ export const domainIconWithoutDefault = (
|
|||||||
: mdiPackageUp
|
: mdiPackageUp
|
||||||
: mdiPackage;
|
: mdiPackage;
|
||||||
|
|
||||||
|
case "water_heater":
|
||||||
|
return compareState === "off" ? mdiWaterBoilerOff : mdiWaterBoiler;
|
||||||
|
|
||||||
case "weather":
|
case "weather":
|
||||||
return weatherIcon(stateObj?.state);
|
return weatherIcon(stateObj?.state);
|
||||||
}
|
}
|
||||||
|
@@ -1,6 +1,7 @@
|
|||||||
/** Return an color representing a state. */
|
/** Return an color representing a state. */
|
||||||
import { HassEntity } from "home-assistant-js-websocket";
|
import { HassEntity } from "home-assistant-js-websocket";
|
||||||
import { UNAVAILABLE } from "../../data/entity";
|
import { UNAVAILABLE } from "../../data/entity";
|
||||||
|
import { computeGroupDomain, GroupEntity } from "../../data/group";
|
||||||
import { computeCssVariable } from "../../resources/css-variables";
|
import { computeCssVariable } from "../../resources/css-variables";
|
||||||
import { slugify } from "../string/slugify";
|
import { slugify } from "../string/slugify";
|
||||||
import { batteryStateColorProperty } from "./color/battery_color";
|
import { batteryStateColorProperty } from "./color/battery_color";
|
||||||
@@ -52,11 +53,11 @@ export const stateColorCss = (stateObj: HassEntity, state?: string) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const domainStateColorProperties = (
|
export const domainStateColorProperties = (
|
||||||
|
domain: string,
|
||||||
stateObj: HassEntity,
|
stateObj: HassEntity,
|
||||||
state?: string
|
state?: string
|
||||||
): string[] => {
|
): string[] => {
|
||||||
const compareState = state !== undefined ? state : stateObj.state;
|
const compareState = state !== undefined ? state : stateObj.state;
|
||||||
const domain = computeDomain(stateObj.entity_id);
|
|
||||||
const active = stateActive(stateObj, state);
|
const active = stateActive(stateObj, state);
|
||||||
|
|
||||||
const properties: string[] = [];
|
const properties: string[] = [];
|
||||||
@@ -95,8 +96,16 @@ export const stateColorProperties = (
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Special rules for group coloring
|
||||||
|
if (domain === "group") {
|
||||||
|
const groupDomain = computeGroupDomain(stateObj as GroupEntity);
|
||||||
|
if (groupDomain && STATE_COLORED_DOMAIN.has(groupDomain)) {
|
||||||
|
return domainStateColorProperties(groupDomain, stateObj, state);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (STATE_COLORED_DOMAIN.has(domain)) {
|
if (STATE_COLORED_DOMAIN.has(domain)) {
|
||||||
return domainStateColorProperties(stateObj, state);
|
return domainStateColorProperties(domain, stateObj, state);
|
||||||
}
|
}
|
||||||
|
|
||||||
return undefined;
|
return undefined;
|
||||||
|
@@ -77,6 +77,23 @@ export const formatNumber = (
|
|||||||
).format(Number(num));
|
).format(Number(num));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
!Number.isNaN(Number(num)) &&
|
||||||
|
num !== "" &&
|
||||||
|
localeOptions?.number_format === NumberFormat.none &&
|
||||||
|
Intl
|
||||||
|
) {
|
||||||
|
// If NumberFormat is none, use en-US format without grouping.
|
||||||
|
return new Intl.NumberFormat(
|
||||||
|
"en-US",
|
||||||
|
getDefaultFormatOptions(num, {
|
||||||
|
...options,
|
||||||
|
useGrouping: false,
|
||||||
|
})
|
||||||
|
).format(Number(num));
|
||||||
|
}
|
||||||
|
|
||||||
if (typeof num === "string") {
|
if (typeof num === "string") {
|
||||||
return num;
|
return num;
|
||||||
}
|
}
|
||||||
|
108
src/common/util/shallow-equal.ts
Normal file
108
src/common/util/shallow-equal.ts
Normal file
@@ -0,0 +1,108 @@
|
|||||||
|
/**
|
||||||
|
* Compares two values for shallow equality, only 1 level deep.
|
||||||
|
*/
|
||||||
|
export const shallowEqual = (a: any, b: any): boolean => {
|
||||||
|
if (a === b) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (a && b && typeof a === "object" && typeof b === "object") {
|
||||||
|
if (a.constructor !== b.constructor) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
let i: number | [any, any];
|
||||||
|
let length: number;
|
||||||
|
if (Array.isArray(a)) {
|
||||||
|
length = a.length;
|
||||||
|
if (length !== b.length) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
for (i = length; i-- !== 0; ) {
|
||||||
|
if (a[i] !== b[i]) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (a instanceof Map && b instanceof Map) {
|
||||||
|
if (a.size !== b.size) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
for (i of a.entries()) {
|
||||||
|
if (!b.has(i[0])) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (i of a.entries()) {
|
||||||
|
if (i[1] !== b.get(i[0])) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (a instanceof Set && b instanceof Set) {
|
||||||
|
if (a.size !== b.size) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
for (i of a.entries()) {
|
||||||
|
if (!b.has(i[0])) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (ArrayBuffer.isView(a) && ArrayBuffer.isView(b)) {
|
||||||
|
// @ts-ignore
|
||||||
|
length = a.length;
|
||||||
|
// @ts-ignore
|
||||||
|
if (length !== b.length) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
for (i = length; i-- !== 0; ) {
|
||||||
|
if (a[i] !== b[i]) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (a.constructor === RegExp) {
|
||||||
|
return a.source === b.source && a.flags === b.flags;
|
||||||
|
}
|
||||||
|
if (a.valueOf !== Object.prototype.valueOf) {
|
||||||
|
return a.valueOf() === b.valueOf();
|
||||||
|
}
|
||||||
|
if (a.toString !== Object.prototype.toString) {
|
||||||
|
return a.toString() === b.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
const keys = Object.keys(a);
|
||||||
|
length = keys.length;
|
||||||
|
if (length !== Object.keys(b).length) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
for (i = length; i-- !== 0; ) {
|
||||||
|
if (!Object.prototype.hasOwnProperty.call(b, keys[i])) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for (i = length; i-- !== 0; ) {
|
||||||
|
const key = keys[i];
|
||||||
|
|
||||||
|
if (a[key] !== b[key]) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// true if both NaN, false otherwise
|
||||||
|
// eslint-disable-next-line no-self-compare
|
||||||
|
return a !== a && b !== b;
|
||||||
|
};
|
@@ -143,8 +143,16 @@ export class StateHistoryChartTimeline extends LitElement {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
afterUpdate: (y) => {
|
afterUpdate: (y) => {
|
||||||
if (this._yWidth !== Math.floor(y.width)) {
|
const yWidth = this.showNames
|
||||||
this._yWidth = Math.floor(y.width);
|
? y.width ?? 0
|
||||||
|
: computeRTL(this.hass)
|
||||||
|
? 0
|
||||||
|
: y.left ?? 0;
|
||||||
|
if (
|
||||||
|
this._yWidth !== Math.floor(yWidth) &&
|
||||||
|
y.ticks.length === this.data.length
|
||||||
|
) {
|
||||||
|
this._yWidth = Math.floor(yWidth);
|
||||||
fireEvent(this, "y-width-changed", {
|
fireEvent(this, "y-width-changed", {
|
||||||
value: this._yWidth,
|
value: this._yWidth,
|
||||||
chartIndex: this.chartIndex,
|
chartIndex: this.chartIndex,
|
||||||
|
@@ -4,11 +4,12 @@ import {
|
|||||||
CSSResultGroup,
|
CSSResultGroup,
|
||||||
html,
|
html,
|
||||||
LitElement,
|
LitElement,
|
||||||
|
nothing,
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
TemplateResult,
|
|
||||||
} from "lit";
|
} from "lit";
|
||||||
import { customElement, property, state, eventOptions } from "lit/decorators";
|
import { customElement, eventOptions, property, state } from "lit/decorators";
|
||||||
import { isComponentLoaded } from "../../common/config/is_component_loaded";
|
import { isComponentLoaded } from "../../common/config/is_component_loaded";
|
||||||
|
import { restoreScroll } from "../../common/decorators/restore-scroll";
|
||||||
import {
|
import {
|
||||||
HistoryResult,
|
HistoryResult,
|
||||||
LineChartUnit,
|
LineChartUnit,
|
||||||
@@ -17,7 +18,6 @@ import {
|
|||||||
import type { HomeAssistant } from "../../types";
|
import type { HomeAssistant } from "../../types";
|
||||||
import "./state-history-chart-line";
|
import "./state-history-chart-line";
|
||||||
import "./state-history-chart-timeline";
|
import "./state-history-chart-timeline";
|
||||||
import { restoreScroll } from "../../common/decorators/restore-scroll";
|
|
||||||
|
|
||||||
const CANVAS_TIMELINE_ROWS_CHUNK = 10; // Split up the canvases to avoid hitting the render limit
|
const CANVAS_TIMELINE_ROWS_CHUNK = 10; // Split up the canvases to avoid hitting the render limit
|
||||||
|
|
||||||
@@ -71,7 +71,7 @@ export class StateHistoryCharts extends LitElement {
|
|||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
@restoreScroll(".container") private _savedScrollPos?: number;
|
@restoreScroll(".container") private _savedScrollPos?: number;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!isComponentLoaded(this.hass, "history")) {
|
if (!isComponentLoaded(this.hass, "history")) {
|
||||||
return html`<div class="info">
|
return html`<div class="info">
|
||||||
${this.hass.localize("ui.components.history_charts.history_disabled")}
|
${this.hass.localize("ui.components.history_charts.history_disabled")}
|
||||||
@@ -130,9 +130,9 @@ export class StateHistoryCharts extends LitElement {
|
|||||||
private _renderHistoryItem = (
|
private _renderHistoryItem = (
|
||||||
item: TimelineEntity[] | LineChartUnit,
|
item: TimelineEntity[] | LineChartUnit,
|
||||||
index: number
|
index: number
|
||||||
): TemplateResult => {
|
) => {
|
||||||
if (!item || index === undefined) {
|
if (!item || index === undefined) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
if (!Array.isArray(item)) {
|
if (!Array.isArray(item)) {
|
||||||
return html`<div class="entry-container">
|
return html`<div class="entry-container">
|
||||||
@@ -175,15 +175,14 @@ export class StateHistoryCharts extends LitElement {
|
|||||||
if (changedProps.has("_chartCount")) {
|
if (changedProps.has("_chartCount")) {
|
||||||
if (this._chartCount < this._childYWidths.length) {
|
if (this._chartCount < this._childYWidths.length) {
|
||||||
this._childYWidths.length = this._chartCount;
|
this._childYWidths.length = this._chartCount;
|
||||||
this._maxYWidth =
|
this._maxYWidth = Math.max(...Object.values(this._childYWidths), 0);
|
||||||
this._childYWidths.length === 0 ? 0 : Math.max(...this._childYWidths);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private _yWidthChanged(e: CustomEvent<HASSDomEvents["y-width-changed"]>) {
|
private _yWidthChanged(e: CustomEvent<HASSDomEvents["y-width-changed"]>) {
|
||||||
this._childYWidths[e.detail.chartIndex] = e.detail.value;
|
this._childYWidths[e.detail.chartIndex] = e.detail.value;
|
||||||
this._maxYWidth = Math.max(...this._childYWidths);
|
this._maxYWidth = Math.max(...Object.values(this._childYWidths), 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
private _isHistoryEmpty(): boolean {
|
private _isHistoryEmpty(): boolean {
|
||||||
|
@@ -48,8 +48,8 @@ class HaDataTableIcon extends LitElement {
|
|||||||
outline: none;
|
outline: none;
|
||||||
font-size: 10px;
|
font-size: 10px;
|
||||||
line-height: 1;
|
line-height: 1;
|
||||||
background-color: var(--paper-tooltip-background, #616161);
|
background-color: var(--simple-tooltip-background, #616161);
|
||||||
color: var(--paper-tooltip-text-color, white);
|
color: var(--simple-tooltip-text-color, white);
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
border-radius: 2px;
|
border-radius: 2px;
|
||||||
}
|
}
|
||||||
|
@@ -6,6 +6,7 @@ import {
|
|||||||
CSSResultGroup,
|
CSSResultGroup,
|
||||||
html,
|
html,
|
||||||
LitElement,
|
LitElement,
|
||||||
|
nothing,
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
TemplateResult,
|
TemplateResult,
|
||||||
} from "lit";
|
} from "lit";
|
||||||
@@ -73,7 +74,7 @@ export interface DataTableColumnData<T = any> extends DataTableSortColumnData {
|
|||||||
title: TemplateResult | string;
|
title: TemplateResult | string;
|
||||||
label?: TemplateResult | string;
|
label?: TemplateResult | string;
|
||||||
type?: "numeric" | "icon" | "icon-button" | "overflow-menu";
|
type?: "numeric" | "icon" | "icon-button" | "overflow-menu";
|
||||||
template?: (data: any, row: T) => TemplateResult | string;
|
template?: (data: any, row: T) => TemplateResult | string | typeof nothing;
|
||||||
width?: string;
|
width?: string;
|
||||||
maxWidth?: string;
|
maxWidth?: string;
|
||||||
grows?: boolean;
|
grows?: boolean;
|
||||||
@@ -352,13 +353,10 @@ export class HaDataTable extends LitElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
private _renderRow = (
|
private _renderRow = (row: DataTableRowData, index: number) => {
|
||||||
row: DataTableRowData,
|
|
||||||
index: number
|
|
||||||
): TemplateResult => {
|
|
||||||
// not sure how this happens...
|
// not sure how this happens...
|
||||||
if (!row) {
|
if (!row) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
if (row.append) {
|
if (row.append) {
|
||||||
return html` <div class="mdc-data-table__row">${row.content}</div> `;
|
return html` <div class="mdc-data-table__row">${row.content}</div> `;
|
||||||
|
@@ -67,11 +67,11 @@ const sortData = (
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure "undefined" is always sorted to the bottom
|
// Ensure "undefined" and "null" are always sorted to the bottom
|
||||||
if (valA === undefined && valB !== undefined) {
|
if (valA == null && valB != null) {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
if (valB === undefined && valA !== undefined) {
|
if (valB == null && valA != null) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -6,6 +6,21 @@ import DateRangePicker from "vue2-daterange-picker";
|
|||||||
import dateRangePickerStyles from "vue2-daterange-picker/dist/vue2-daterange-picker.css";
|
import dateRangePickerStyles from "vue2-daterange-picker/dist/vue2-daterange-picker.css";
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
|
|
||||||
|
// Set the current date to the left picker instead of the right picker because the right is hidden
|
||||||
|
const CustomDateRangePicker = Vue.extend({
|
||||||
|
mixins: [DateRangePicker],
|
||||||
|
methods: {
|
||||||
|
selectMonthDate() {
|
||||||
|
const dt: Date = this.end || new Date();
|
||||||
|
// @ts-ignore
|
||||||
|
this.changeLeftMonth({
|
||||||
|
year: dt.getFullYear(),
|
||||||
|
month: dt.getMonth() + 1,
|
||||||
|
});
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
const Component = Vue.extend({
|
const Component = Vue.extend({
|
||||||
props: {
|
props: {
|
||||||
timePicker: {
|
timePicker: {
|
||||||
@@ -47,7 +62,7 @@ const Component = Vue.extend({
|
|||||||
},
|
},
|
||||||
render(createElement) {
|
render(createElement) {
|
||||||
// @ts-expect-error
|
// @ts-expect-error
|
||||||
return createElement(DateRangePicker, {
|
return createElement(CustomDateRangePicker, {
|
||||||
props: {
|
props: {
|
||||||
"time-picker": this.timePicker,
|
"time-picker": this.timePicker,
|
||||||
"auto-apply": this.autoApply,
|
"auto-apply": this.autoApply,
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
import "@material/mwc-button/mwc-button";
|
import "@material/mwc-button/mwc-button";
|
||||||
import { UnsubscribeFunc } from "home-assistant-js-websocket";
|
|
||||||
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
|
||||||
import { ComboBoxLitRenderer } from "@vaadin/combo-box/lit";
|
import { ComboBoxLitRenderer } from "@vaadin/combo-box/lit";
|
||||||
|
import { UnsubscribeFunc } from "home-assistant-js-websocket";
|
||||||
|
import { html, LitElement, PropertyValues, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { fireEvent } from "../../common/dom/fire_event";
|
import { fireEvent } from "../../common/dom/fire_event";
|
||||||
@@ -230,9 +230,9 @@ export class HaAreaDevicesPicker extends SubscribeMixin(LitElement) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this._devices || !this._areas || !this._entities) {
|
if (!this._devices || !this._areas || !this._entities) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
const areas = this._getAreasWithDevices(
|
const areas = this._getAreasWithDevices(
|
||||||
this._devices,
|
this._devices,
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import "@material/mwc-list/mwc-list-item";
|
import "@material/mwc-list/mwc-list-item";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { property, state } from "lit/decorators";
|
import { property, state } from "lit/decorators";
|
||||||
import { fireEvent } from "../../common/dom/fire_event";
|
import { fireEvent } from "../../common/dom/fire_event";
|
||||||
import {
|
import {
|
||||||
@@ -85,9 +85,9 @@ export abstract class HaDeviceAutomationPicker<
|
|||||||
return `${this._automations[idx].device_id}_${idx}`;
|
return `${this._automations[idx].device_id}_${idx}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (this._renderEmpty) {
|
if (this._renderEmpty) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
const value = this._value;
|
const value = this._value;
|
||||||
return html`
|
return html`
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { css, html, LitElement, TemplateResult } from "lit";
|
import { css, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import { fireEvent } from "../../common/dom/fire_event";
|
import { fireEvent } from "../../common/dom/fire_event";
|
||||||
import { PolymerChangedEvent } from "../../polymer-types";
|
import { PolymerChangedEvent } from "../../polymer-types";
|
||||||
@@ -49,9 +49,9 @@ class HaDevicesPicker extends LitElement {
|
|||||||
|
|
||||||
@property() public entityFilter?: HaDevicePickerEntityFilterFunc;
|
@property() public entityFilter?: HaDevicePickerEntityFilterFunc;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
const currentDevices = this._currentDevices;
|
const currentDevices = this._currentDevices;
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import type { HassEntity } from "home-assistant-js-websocket";
|
import type { HassEntity } from "home-assistant-js-websocket";
|
||||||
import { css, html, LitElement, TemplateResult } from "lit";
|
import { css, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { fireEvent } from "../../common/dom/fire_event";
|
import { fireEvent } from "../../common/dom/fire_event";
|
||||||
@@ -76,9 +76,9 @@ class HaEntitiesPickerLight extends LitElement {
|
|||||||
|
|
||||||
@property() public entityFilter?: HaEntityPickerEntityFilterFunc;
|
@property() public entityFilter?: HaEntityPickerEntityFilterFunc;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
const currentEntities = this._currentEntities;
|
const currentEntities = this._currentEntities;
|
||||||
|
@@ -1,7 +1,7 @@
|
|||||||
import { HassEntity } from "home-assistant-js-websocket";
|
import { HassEntity } from "home-assistant-js-websocket";
|
||||||
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
import { html, LitElement, PropertyValues, nothing } from "lit";
|
||||||
import { customElement, property, query } from "lit/decorators";
|
import { customElement, property, query } from "lit/decorators";
|
||||||
import { formatAttributeName } from "../../data/entity_attributes";
|
import { computeAttributeNameDisplay } from "../../common/entity/compute_attribute_display";
|
||||||
import { PolymerChangedEvent } from "../../polymer-types";
|
import { PolymerChangedEvent } from "../../polymer-types";
|
||||||
import { HomeAssistant } from "../../types";
|
import { HomeAssistant } from "../../types";
|
||||||
import "../ha-combo-box";
|
import "../ha-combo-box";
|
||||||
@@ -54,21 +54,33 @@ class HaEntityAttributePicker extends LitElement {
|
|||||||
.filter((key) => !this.hideAttributes?.includes(key))
|
.filter((key) => !this.hideAttributes?.includes(key))
|
||||||
.map((key) => ({
|
.map((key) => ({
|
||||||
value: key,
|
value: key,
|
||||||
label: formatAttributeName(key),
|
label: computeAttributeNameDisplay(
|
||||||
|
this.hass.localize,
|
||||||
|
state,
|
||||||
|
this.hass.entities,
|
||||||
|
key
|
||||||
|
),
|
||||||
}))
|
}))
|
||||||
: [];
|
: [];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
<ha-combo-box
|
<ha-combo-box
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.value=${this.value ? formatAttributeName(this.value) : ""}
|
.value=${this.value
|
||||||
|
? computeAttributeNameDisplay(
|
||||||
|
this.hass.localize,
|
||||||
|
this.hass.states[this.entityId!],
|
||||||
|
this.hass.entities,
|
||||||
|
this.value
|
||||||
|
)
|
||||||
|
: ""}
|
||||||
.autofocus=${this.autofocus}
|
.autofocus=${this.autofocus}
|
||||||
.label=${this.label ??
|
.label=${this.label ??
|
||||||
this.hass.localize(
|
this.hass.localize(
|
||||||
|
@@ -1,14 +1,14 @@
|
|||||||
import { HassEntity } from "home-assistant-js-websocket";
|
import { HassEntity } from "home-assistant-js-websocket";
|
||||||
import { html, LitElement, PropertyValues, TemplateResult } from "lit";
|
import { html, LitElement, PropertyValues, nothing } from "lit";
|
||||||
import { customElement, property, query } from "lit/decorators";
|
import { customElement, property, query } from "lit/decorators";
|
||||||
|
import { fireEvent } from "../../common/dom/fire_event";
|
||||||
import { computeStateDisplay } from "../../common/entity/compute_state_display";
|
import { computeStateDisplay } from "../../common/entity/compute_state_display";
|
||||||
import { PolymerChangedEvent } from "../../polymer-types";
|
|
||||||
import { getStates } from "../../common/entity/get_states";
|
import { getStates } from "../../common/entity/get_states";
|
||||||
|
import { computeAttributeValueDisplay } from "../../common/entity/compute_attribute_display";
|
||||||
|
import { PolymerChangedEvent } from "../../polymer-types";
|
||||||
import { HomeAssistant } from "../../types";
|
import { HomeAssistant } from "../../types";
|
||||||
import "../ha-combo-box";
|
import "../ha-combo-box";
|
||||||
import type { HaComboBox } from "../ha-combo-box";
|
import type { HaComboBox } from "../ha-combo-box";
|
||||||
import { formatAttributeValue } from "../../data/entity_attributes";
|
|
||||||
import { fireEvent } from "../../common/dom/fire_event";
|
|
||||||
|
|
||||||
export type HaEntityPickerEntityFilterFunc = (entityId: HassEntity) => boolean;
|
export type HaEntityPickerEntityFilterFunc = (entityId: HassEntity) => boolean;
|
||||||
|
|
||||||
@@ -58,15 +58,22 @@ class HaEntityStatePicker extends LitElement {
|
|||||||
this.hass.entities,
|
this.hass.entities,
|
||||||
key
|
key
|
||||||
)
|
)
|
||||||
: formatAttributeValue(this.hass, key),
|
: computeAttributeValueDisplay(
|
||||||
|
this.hass.localize,
|
||||||
|
state,
|
||||||
|
this.hass.locale,
|
||||||
|
this.hass.entities,
|
||||||
|
this.attribute,
|
||||||
|
key
|
||||||
|
),
|
||||||
}))
|
}))
|
||||||
: [];
|
: [];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
|
@@ -30,6 +30,9 @@ export class HaStatisticPicker extends LitElement {
|
|||||||
@property({ attribute: "statistic-types" })
|
@property({ attribute: "statistic-types" })
|
||||||
public statisticTypes?: "mean" | "sum";
|
public statisticTypes?: "mean" | "sum";
|
||||||
|
|
||||||
|
@property({ type: Boolean, attribute: "allow-custom-entity" })
|
||||||
|
public allowCustomEntity;
|
||||||
|
|
||||||
@property({ type: Array }) public statisticIds?: StatisticsMetaData[];
|
@property({ type: Array }) public statisticIds?: StatisticsMetaData[];
|
||||||
|
|
||||||
@property({ type: Boolean }) public disabled?: boolean;
|
@property({ type: Boolean }) public disabled?: boolean;
|
||||||
@@ -245,6 +248,7 @@ export class HaStatisticPicker extends LitElement {
|
|||||||
.value=${this._value}
|
.value=${this._value}
|
||||||
.renderer=${this._rowRenderer}
|
.renderer=${this._rowRenderer}
|
||||||
.disabled=${this.disabled}
|
.disabled=${this.disabled}
|
||||||
|
.allowCustomValue=${this.allowCustomEntity}
|
||||||
item-value-path="id"
|
item-value-path="id"
|
||||||
item-id-path="id"
|
item-id-path="id"
|
||||||
item-label-path="name"
|
item-label-path="name"
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import { fireEvent } from "../../common/dom/fire_event";
|
import { fireEvent } from "../../common/dom/fire_event";
|
||||||
import type { PolymerChangedEvent } from "../../polymer-types";
|
import type { PolymerChangedEvent } from "../../polymer-types";
|
||||||
@@ -22,6 +22,9 @@ class HaStatisticsPicker extends LitElement {
|
|||||||
@property({ attribute: "pick-statistic-label" })
|
@property({ attribute: "pick-statistic-label" })
|
||||||
public pickStatisticLabel?: string;
|
public pickStatisticLabel?: string;
|
||||||
|
|
||||||
|
@property({ type: Boolean, attribute: "allow-custom-entity" })
|
||||||
|
public allowCustomEntity;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Show only statistics natively stored with these units of measurements.
|
* Show only statistics natively stored with these units of measurements.
|
||||||
* @attr include-statistics-unit-of-measurement
|
* @attr include-statistics-unit-of-measurement
|
||||||
@@ -56,9 +59,9 @@ class HaStatisticsPicker extends LitElement {
|
|||||||
})
|
})
|
||||||
public ignoreRestrictionsOnFirstStatistic = false;
|
public ignoreRestrictionsOnFirstStatistic = false;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
const ignoreRestriction =
|
const ignoreRestriction =
|
||||||
@@ -71,6 +74,9 @@ class HaStatisticsPicker extends LitElement {
|
|||||||
const includeUnitClassCurrent = ignoreRestriction
|
const includeUnitClassCurrent = ignoreRestriction
|
||||||
? undefined
|
? undefined
|
||||||
: this.includeUnitClass;
|
: this.includeUnitClass;
|
||||||
|
const includeDeviceClassCurrent = ignoreRestriction
|
||||||
|
? undefined
|
||||||
|
: this.includeDeviceClass;
|
||||||
const includeStatisticTypesCurrent = ignoreRestriction
|
const includeStatisticTypesCurrent = ignoreRestriction
|
||||||
? undefined
|
? undefined
|
||||||
: this.statisticTypes;
|
: this.statisticTypes;
|
||||||
@@ -84,10 +90,12 @@ class HaStatisticsPicker extends LitElement {
|
|||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.includeStatisticsUnitOfMeasurement=${includeStatisticsUnitCurrent}
|
.includeStatisticsUnitOfMeasurement=${includeStatisticsUnitCurrent}
|
||||||
.includeUnitClass=${includeUnitClassCurrent}
|
.includeUnitClass=${includeUnitClassCurrent}
|
||||||
|
.includeDeviceClass=${includeDeviceClassCurrent}
|
||||||
.value=${statisticId}
|
.value=${statisticId}
|
||||||
.statisticTypes=${includeStatisticTypesCurrent}
|
.statisticTypes=${includeStatisticTypesCurrent}
|
||||||
.statisticIds=${this.statisticIds}
|
.statisticIds=${this.statisticIds}
|
||||||
.label=${this.pickedStatisticLabel}
|
.label=${this.pickedStatisticLabel}
|
||||||
|
.allowCustomEntity=${this.allowCustomEntity}
|
||||||
@value-changed=${this._statisticChanged}
|
@value-changed=${this._statisticChanged}
|
||||||
></ha-statistic-picker>
|
></ha-statistic-picker>
|
||||||
</div>
|
</div>
|
||||||
@@ -103,6 +111,7 @@ class HaStatisticsPicker extends LitElement {
|
|||||||
.statisticTypes=${this.statisticTypes}
|
.statisticTypes=${this.statisticTypes}
|
||||||
.statisticIds=${this.statisticIds}
|
.statisticIds=${this.statisticIds}
|
||||||
.label=${this.pickStatisticLabel}
|
.label=${this.pickStatisticLabel}
|
||||||
|
.allowCustomEntity=${this.allowCustomEntity}
|
||||||
@value-changed=${this._addStatistic}
|
@value-changed=${this._addStatistic}
|
||||||
></ha-statistic-picker>
|
></ha-statistic-picker>
|
||||||
</div>
|
</div>
|
||||||
|
@@ -6,7 +6,7 @@ import {
|
|||||||
html,
|
html,
|
||||||
LitElement,
|
LitElement,
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
TemplateResult,
|
nothing,
|
||||||
} from "lit";
|
} from "lit";
|
||||||
import { property, state } from "lit/decorators";
|
import { property, state } from "lit/decorators";
|
||||||
import { ifDefined } from "lit/directives/if-defined";
|
import { ifDefined } from "lit/directives/if-defined";
|
||||||
@@ -45,7 +45,7 @@ export class StateBadge extends LitElement {
|
|||||||
return this.stateColor || (domain === "light" && this.stateColor !== false);
|
return this.stateColor || (domain === "light" && this.stateColor !== false);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
const stateObj = this.stateObj;
|
const stateObj = this.stateObj;
|
||||||
|
|
||||||
// We either need a `stateObj` or one override
|
// We either need a `stateObj` or one override
|
||||||
@@ -56,7 +56,7 @@ export class StateBadge extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!this._showIcon) {
|
if (!this._showIcon) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
const domain = stateObj ? computeStateDomain(stateObj) : undefined;
|
const domain = stateObj ? computeStateDomain(stateObj) : undefined;
|
||||||
|
@@ -1,6 +1,6 @@
|
|||||||
import "@polymer/paper-tooltip/paper-tooltip";
|
import "@lrnwebcomponents/simple-tooltip/simple-tooltip";
|
||||||
import type { HassEntity } from "home-assistant-js-websocket";
|
import type { HassEntity } from "home-assistant-js-websocket";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import { computeStateName } from "../../common/entity/compute_state_name";
|
import { computeStateName } from "../../common/entity/compute_state_name";
|
||||||
import { computeRTL } from "../../common/util/compute_rtl";
|
import { computeRTL } from "../../common/util/compute_rtl";
|
||||||
@@ -21,9 +21,9 @@ class StateInfo extends LitElement {
|
|||||||
|
|
||||||
@property() public color?: string;
|
@property() public color?: string;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass || !this.stateObj) {
|
if (!this.hass || !this.stateObj) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
const name = computeStateName(this.stateObj);
|
const name = computeStateName(this.stateObj);
|
||||||
@@ -45,7 +45,7 @@ class StateInfo extends LitElement {
|
|||||||
.datetime=${this.stateObj.last_changed}
|
.datetime=${this.stateObj.last_changed}
|
||||||
capitalize
|
capitalize
|
||||||
></ha-relative-time>
|
></ha-relative-time>
|
||||||
<paper-tooltip animation-delay="0" for="last_changed">
|
<simple-tooltip animation-delay="0" for="last_changed">
|
||||||
<div>
|
<div>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<span class="column-name">
|
<span class="column-name">
|
||||||
@@ -72,7 +72,7 @@ class StateInfo extends LitElement {
|
|||||||
></ha-relative-time>
|
></ha-relative-time>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</paper-tooltip>
|
</simple-tooltip>
|
||||||
</div>`
|
</div>`
|
||||||
: html`<div class="extra-info"><slot></slot></div>`}
|
: html`<div class="extra-info"><slot></slot></div>`}
|
||||||
</div>`;
|
</div>`;
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import { html, LitElement, TemplateResult } from "lit";
|
|
||||||
import { ComboBoxLitRenderer } from "@vaadin/combo-box/lit";
|
import { ComboBoxLitRenderer } from "@vaadin/combo-box/lit";
|
||||||
|
import { html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, query, state } from "lit/decorators";
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
import { isComponentLoaded } from "../common/config/is_component_loaded";
|
import { isComponentLoaded } from "../common/config/is_component_loaded";
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
@@ -54,9 +54,9 @@ class HaAddonPicker extends LitElement {
|
|||||||
this._getAddons();
|
this._getAddons();
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this._addons) {
|
if (!this._addons) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<ha-combo-box
|
<ha-combo-box
|
||||||
|
@@ -1,11 +0,0 @@
|
|||||||
import { html } from "lit";
|
|
||||||
import { HomeAssistant } from "../types";
|
|
||||||
import { documentationUrl } from "../util/documentation-url";
|
|
||||||
|
|
||||||
export const analyticsLearnMore = (hass: HomeAssistant) => html`<a
|
|
||||||
.href=${documentationUrl(hass, "/integrations/analytics/")}
|
|
||||||
target="_blank"
|
|
||||||
rel="noreferrer"
|
|
||||||
>
|
|
||||||
How we process your data
|
|
||||||
</a>`;
|
|
@@ -1,4 +1,4 @@
|
|||||||
import "@polymer/paper-tooltip/paper-tooltip";
|
import "@lrnwebcomponents/simple-tooltip/simple-tooltip";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
@@ -9,18 +9,7 @@ import "./ha-settings-row";
|
|||||||
import "./ha-switch";
|
import "./ha-switch";
|
||||||
import type { HaSwitch } from "./ha-switch";
|
import type { HaSwitch } from "./ha-switch";
|
||||||
|
|
||||||
const ADDITIONAL_PREFERENCES = [
|
const ADDITIONAL_PREFERENCES = ["usage", "statistics"] as const;
|
||||||
{
|
|
||||||
key: "usage",
|
|
||||||
title: "Usage",
|
|
||||||
description: "Details of what you use with Home Assistant",
|
|
||||||
},
|
|
||||||
{
|
|
||||||
key: "statistics",
|
|
||||||
title: "Statistical data",
|
|
||||||
description: "Counts containing total number of datapoints",
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
interface HASSDomEvents {
|
interface HASSDomEvents {
|
||||||
@@ -34,15 +23,25 @@ export class HaAnalytics extends LitElement {
|
|||||||
|
|
||||||
@property({ attribute: false }) public analytics?: Analytics;
|
@property({ attribute: false }) public analytics?: Analytics;
|
||||||
|
|
||||||
|
@property({ attribute: "translation_key_panel" }) public translationKeyPanel:
|
||||||
|
| "page-onboarding"
|
||||||
|
| "config" = "config";
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
const loading = this.analytics === undefined;
|
const loading = this.analytics === undefined;
|
||||||
const baseEnabled = !loading && this.analytics!.preferences.base;
|
const baseEnabled = !loading && this.analytics!.preferences.base;
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
<ha-settings-row>
|
<ha-settings-row>
|
||||||
<span slot="heading" data-for="base"> Basic analytics </span>
|
<span slot="heading" data-for="base">
|
||||||
|
${this.hass.localize(
|
||||||
|
`ui.panel.${this.translationKeyPanel}.analytics.preferences.base.title`
|
||||||
|
)}
|
||||||
|
</span>
|
||||||
<span slot="description" data-for="base">
|
<span slot="description" data-for="base">
|
||||||
This includes information about your system.
|
${this.hass.localize(
|
||||||
|
`ui.panel.${this.translationKeyPanel}.analytics.preferences.base.description`
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<ha-switch
|
<ha-switch
|
||||||
@change=${this._handleRowClick}
|
@change=${this._handleRowClick}
|
||||||
@@ -57,26 +56,31 @@ export class HaAnalytics extends LitElement {
|
|||||||
(preference) =>
|
(preference) =>
|
||||||
html`
|
html`
|
||||||
<ha-settings-row>
|
<ha-settings-row>
|
||||||
<span slot="heading" data-for=${preference.key}>
|
<span slot="heading" data-for=${preference}>
|
||||||
${preference.title}
|
${this.hass.localize(
|
||||||
|
`ui.panel.${this.translationKeyPanel}.analytics.preferences.${preference}.title`
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<span slot="description" data-for=${preference.key}>
|
<span slot="description" data-for=${preference}>
|
||||||
${preference.description}
|
${this.hass.localize(
|
||||||
|
`ui.panel.${this.translationKeyPanel}.analytics.preferences.${preference}.description`
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<span>
|
<span>
|
||||||
<ha-switch
|
<ha-switch
|
||||||
@change=${this._handleRowClick}
|
@change=${this._handleRowClick}
|
||||||
.checked=${this.analytics?.preferences[preference.key]}
|
.checked=${this.analytics?.preferences[preference]}
|
||||||
.preference=${preference.key}
|
.preference=${preference}
|
||||||
name=${preference.key}
|
name=${preference}
|
||||||
>
|
>
|
||||||
</ha-switch>
|
</ha-switch>
|
||||||
${!baseEnabled
|
${!baseEnabled
|
||||||
? html`
|
? html`
|
||||||
<paper-tooltip animation-delay="0" position="right">
|
<simple-tooltip animation-delay="0" position="right">
|
||||||
You need to enable basic analytics for this option to be
|
${this.hass.localize(
|
||||||
available
|
`ui.panel.${this.translationKeyPanel}.analytics.need_base_enabled`
|
||||||
</paper-tooltip>
|
)}
|
||||||
|
</simple-tooltip>
|
||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
</span>
|
</span>
|
||||||
@@ -84,9 +88,15 @@ export class HaAnalytics extends LitElement {
|
|||||||
`
|
`
|
||||||
)}
|
)}
|
||||||
<ha-settings-row>
|
<ha-settings-row>
|
||||||
<span slot="heading" data-for="diagnostics"> Diagnostics </span>
|
<span slot="heading" data-for="diagnostics">
|
||||||
|
${this.hass.localize(
|
||||||
|
`ui.panel.${this.translationKeyPanel}.analytics.preferences.diagnostics.title`
|
||||||
|
)}
|
||||||
|
</span>
|
||||||
<span slot="description" data-for="diagnostics">
|
<span slot="description" data-for="diagnostics">
|
||||||
Share crash reports when unexpected errors occur.
|
${this.hass.localize(
|
||||||
|
`ui.panel.${this.translationKeyPanel}.analytics.preferences.diagnostics.description`
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
<ha-switch
|
<ha-switch
|
||||||
@change=${this._handleRowClick}
|
@change=${this._handleRowClick}
|
||||||
@@ -132,7 +142,7 @@ export class HaAnalytics extends LitElement {
|
|||||||
preferences[preference] = target.checked;
|
preferences[preference] = target.checked;
|
||||||
|
|
||||||
if (
|
if (
|
||||||
ADDITIONAL_PREFERENCES.some((entry) => entry.key === preference) &&
|
ADDITIONAL_PREFERENCES.some((entry) => entry === preference) &&
|
||||||
target.checked
|
target.checked
|
||||||
) {
|
) {
|
||||||
preferences.base = true;
|
preferences.base = true;
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import { HassEntity } from "home-assistant-js-websocket";
|
import { HassEntity } from "home-assistant-js-websocket";
|
||||||
import { css, html, LitElement, TemplateResult } from "lit";
|
import { css, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
import { SubscribeMixin } from "../mixins/subscribe-mixin";
|
import { SubscribeMixin } from "../mixins/subscribe-mixin";
|
||||||
@@ -60,9 +60,9 @@ export class HaAreasPicker extends SubscribeMixin(LitElement) {
|
|||||||
|
|
||||||
@property({ type: Boolean }) public required?: boolean;
|
@property({ type: Boolean }) public required?: boolean;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
const currentAreas = this._currentAreas;
|
const currentAreas = this._currentAreas;
|
||||||
|
@@ -1,11 +1,11 @@
|
|||||||
import { HassEntity } from "home-assistant-js-websocket";
|
import { HassEntity } from "home-assistant-js-websocket";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import {
|
import {
|
||||||
formatAttributeName,
|
computeAttributeNameDisplay,
|
||||||
formatAttributeValue,
|
computeAttributeValueDisplay,
|
||||||
STATE_ATTRIBUTES,
|
} from "../common/entity/compute_attribute_display";
|
||||||
} from "../data/entity_attributes";
|
import { STATE_ATTRIBUTES } from "../data/entity_attributes";
|
||||||
import { haStyle } from "../resources/styles";
|
import { haStyle } from "../resources/styles";
|
||||||
import { HomeAssistant } from "../types";
|
import { HomeAssistant } from "../types";
|
||||||
|
|
||||||
@@ -21,9 +21,9 @@ class HaAttributes extends LitElement {
|
|||||||
|
|
||||||
@state() private _expanded = false;
|
@state() private _expanded = false;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.stateObj) {
|
if (!this.stateObj) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
const attributes = this.computeDisplayAttributes(
|
const attributes = this.computeDisplayAttributes(
|
||||||
@@ -32,7 +32,7 @@ class HaAttributes extends LitElement {
|
|||||||
)
|
)
|
||||||
);
|
);
|
||||||
if (attributes.length === 0) {
|
if (attributes.length === 0) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
return html`
|
return html`
|
||||||
@@ -49,9 +49,22 @@ class HaAttributes extends LitElement {
|
|||||||
${attributes.map(
|
${attributes.map(
|
||||||
(attribute) => html`
|
(attribute) => html`
|
||||||
<div class="data-entry">
|
<div class="data-entry">
|
||||||
<div class="key">${formatAttributeName(attribute)}</div>
|
<div class="key">
|
||||||
|
${computeAttributeNameDisplay(
|
||||||
|
this.hass.localize,
|
||||||
|
this.stateObj!,
|
||||||
|
this.hass.entities,
|
||||||
|
attribute
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
<div class="value">
|
<div class="value">
|
||||||
${this.formatAttribute(attribute)}
|
${computeAttributeValueDisplay(
|
||||||
|
this.hass.localize,
|
||||||
|
this.stateObj!,
|
||||||
|
this.hass.locale,
|
||||||
|
this.hass.entities,
|
||||||
|
attribute
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
`
|
`
|
||||||
@@ -121,14 +134,6 @@ class HaAttributes extends LitElement {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
private formatAttribute(attribute: string): string | TemplateResult {
|
|
||||||
if (!this.stateObj) {
|
|
||||||
return "—";
|
|
||||||
}
|
|
||||||
const value = this.stateObj.attributes[attribute];
|
|
||||||
return formatAttributeValue(this.hass, value);
|
|
||||||
}
|
|
||||||
|
|
||||||
private expandedChanged(ev) {
|
private expandedChanged(ev) {
|
||||||
this._expanded = ev.detail.expanded;
|
this._expanded = ev.detail.expanded;
|
||||||
}
|
}
|
||||||
|
@@ -1,5 +1,5 @@
|
|||||||
import "@material/mwc-list/mwc-list-item";
|
import "@material/mwc-list/mwc-list-item";
|
||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
@@ -51,9 +51,9 @@ class HaBluePrintPicker extends LitElement {
|
|||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.hass) {
|
if (!this.hass) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<ha-select
|
<ha-select
|
||||||
|
@@ -4,7 +4,7 @@ import {
|
|||||||
html,
|
html,
|
||||||
LitElement,
|
LitElement,
|
||||||
PropertyValues,
|
PropertyValues,
|
||||||
TemplateResult,
|
nothing,
|
||||||
} from "lit";
|
} from "lit";
|
||||||
import { customElement, property, state } from "lit/decorators";
|
import { customElement, property, state } from "lit/decorators";
|
||||||
import { isComponentLoaded } from "../common/config/is_component_loaded";
|
import { isComponentLoaded } from "../common/config/is_component_loaded";
|
||||||
@@ -76,9 +76,9 @@ export class HaCameraStream extends LitElement {
|
|||||||
this._connected = false;
|
this._connected = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this.stateObj) {
|
if (!this.stateObj) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
if (__DEMO__ || this._shouldRenderMJPEG) {
|
if (__DEMO__ || this._shouldRenderMJPEG) {
|
||||||
return html`<img
|
return html`<img
|
||||||
@@ -102,7 +102,7 @@ export class HaCameraStream extends LitElement {
|
|||||||
.url=${this._url}
|
.url=${this._url}
|
||||||
.posterUrl=${this._posterUrl}
|
.posterUrl=${this._posterUrl}
|
||||||
></ha-hls-player>`
|
></ha-hls-player>`
|
||||||
: html``;
|
: nothing;
|
||||||
}
|
}
|
||||||
if (this.stateObj.attributes.frontend_stream_type === STREAM_TYPE_WEB_RTC) {
|
if (this.stateObj.attributes.frontend_stream_type === STREAM_TYPE_WEB_RTC) {
|
||||||
return html`<ha-web-rtc-player
|
return html`<ha-web-rtc-player
|
||||||
@@ -115,7 +115,7 @@ export class HaCameraStream extends LitElement {
|
|||||||
.posterUrl=${this._posterUrl}
|
.posterUrl=${this._posterUrl}
|
||||||
></ha-web-rtc-player>`;
|
></ha-web-rtc-player>`;
|
||||||
}
|
}
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
|
|
||||||
private get _shouldRenderMJPEG() {
|
private get _shouldRenderMJPEG() {
|
||||||
|
@@ -1,4 +1,4 @@
|
|||||||
import { css, CSSResultGroup, html, LitElement, TemplateResult } from "lit";
|
import { css, CSSResultGroup, html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
|
|
||||||
@customElement("ha-card")
|
@customElement("ha-card")
|
||||||
@@ -70,11 +70,11 @@ export class HaCard extends LitElement {
|
|||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
return html`
|
return html`
|
||||||
${this.header
|
${this.header
|
||||||
? html`<h1 class="card-header">${this.header}</h1>`
|
? html`<h1 class="card-header">${this.header}</h1>`
|
||||||
: html``}
|
: nothing}
|
||||||
<slot></slot>
|
<slot></slot>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
@@ -1,13 +1,6 @@
|
|||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
import chipStyles from "@material/chips/dist/mdc.chips.min.css";
|
import chipStyles from "@material/chips/dist/mdc.chips.min.css";
|
||||||
import {
|
import { css, CSSResultGroup, html, LitElement, nothing, unsafeCSS } from "lit";
|
||||||
css,
|
|
||||||
CSSResultGroup,
|
|
||||||
html,
|
|
||||||
LitElement,
|
|
||||||
TemplateResult,
|
|
||||||
unsafeCSS,
|
|
||||||
} from "lit";
|
|
||||||
import { customElement, property } from "lit/decorators";
|
import { customElement, property } from "lit/decorators";
|
||||||
|
|
||||||
@customElement("ha-chip")
|
@customElement("ha-chip")
|
||||||
@@ -18,14 +11,14 @@ export class HaChip extends LitElement {
|
|||||||
|
|
||||||
@property({ type: Boolean }) public noText = false;
|
@property({ type: Boolean }) public noText = false;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
return html`
|
return html`
|
||||||
<div class="mdc-chip ${this.noText ? "no-text" : ""}">
|
<div class="mdc-chip ${this.noText ? "no-text" : ""}">
|
||||||
${this.hasIcon
|
${this.hasIcon
|
||||||
? html`<div class="mdc-chip__icon mdc-chip__icon--leading">
|
? html`<div class="mdc-chip__icon mdc-chip__icon--leading">
|
||||||
<slot name="icon"></slot>
|
<slot name="icon"></slot>
|
||||||
</div>`
|
</div>`
|
||||||
: null}
|
: nothing}
|
||||||
<div class="mdc-chip__ripple"></div>
|
<div class="mdc-chip__ripple"></div>
|
||||||
<span role="gridcell">
|
<span role="gridcell">
|
||||||
<span role="button" tabindex="0" class="mdc-chip__primary-action">
|
<span role="button" tabindex="0" class="mdc-chip__primary-action">
|
||||||
@@ -36,7 +29,7 @@ export class HaChip extends LitElement {
|
|||||||
? html`<div class="mdc-chip__icon mdc-chip__icon--trailing">
|
? html`<div class="mdc-chip__icon mdc-chip__icon--trailing">
|
||||||
<slot name="trailing-icon"></slot>
|
<slot name="trailing-icon"></slot>
|
||||||
</div>`
|
</div>`
|
||||||
: null}
|
: nothing}
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
@@ -27,6 +27,7 @@ class HaClimateState extends LitElement {
|
|||||||
${computeAttributeValueDisplay(
|
${computeAttributeValueDisplay(
|
||||||
this.hass.localize,
|
this.hass.localize,
|
||||||
this.stateObj,
|
this.stateObj,
|
||||||
|
this.hass.locale,
|
||||||
this.hass.entities,
|
this.hass.entities,
|
||||||
"preset_mode"
|
"preset_mode"
|
||||||
)}`
|
)}`
|
||||||
@@ -142,6 +143,7 @@ class HaClimateState extends LitElement {
|
|||||||
? `${computeAttributeValueDisplay(
|
? `${computeAttributeValueDisplay(
|
||||||
this.hass.localize,
|
this.hass.localize,
|
||||||
this.stateObj,
|
this.stateObj,
|
||||||
|
this.hass.locale,
|
||||||
this.hass.entities,
|
this.hass.entities,
|
||||||
"hvac_action"
|
"hvac_action"
|
||||||
)} (${stateString})`
|
)} (${stateString})`
|
||||||
|
@@ -1,16 +1,16 @@
|
|||||||
import "@material/mwc-list/mwc-list-item";
|
import "@material/mwc-list/mwc-list-item";
|
||||||
import { html, LitElement, TemplateResult } from "lit";
|
|
||||||
import { ComboBoxLitRenderer } from "@vaadin/combo-box/lit";
|
import { ComboBoxLitRenderer } from "@vaadin/combo-box/lit";
|
||||||
|
import { html, LitElement, nothing } from "lit";
|
||||||
import { customElement, property, query, state } from "lit/decorators";
|
import { customElement, property, query, state } from "lit/decorators";
|
||||||
import { fireEvent } from "../common/dom/fire_event";
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
import { PolymerChangedEvent } from "../polymer-types";
|
import { caseInsensitiveStringCompare } from "../common/string/compare";
|
||||||
import { HomeAssistant } from "../types";
|
|
||||||
import type { HaComboBox } from "./ha-combo-box";
|
|
||||||
import { ConfigEntry, getConfigEntries } from "../data/config_entries";
|
import { ConfigEntry, getConfigEntries } from "../data/config_entries";
|
||||||
import { domainToName } from "../data/integration";
|
import { domainToName } from "../data/integration";
|
||||||
import { caseInsensitiveStringCompare } from "../common/string/compare";
|
import { PolymerChangedEvent } from "../polymer-types";
|
||||||
|
import { HomeAssistant } from "../types";
|
||||||
import { brandsUrl } from "../util/brands-url";
|
import { brandsUrl } from "../util/brands-url";
|
||||||
import "./ha-combo-box";
|
import "./ha-combo-box";
|
||||||
|
import type { HaComboBox } from "./ha-combo-box";
|
||||||
|
|
||||||
export interface ConfigEntryExtended extends ConfigEntry {
|
export interface ConfigEntryExtended extends ConfigEntry {
|
||||||
localized_domain_name?: string;
|
localized_domain_name?: string;
|
||||||
@@ -72,9 +72,9 @@ class HaConfigEntryPicker extends LitElement {
|
|||||||
/>
|
/>
|
||||||
</mwc-list-item>`;
|
</mwc-list-item>`;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render() {
|
||||||
if (!this._configEntries) {
|
if (!this._configEntries) {
|
||||||
return html``;
|
return nothing;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<ha-combo-box
|
<ha-combo-box
|
||||||
|
@@ -85,6 +85,7 @@ export class HaControlButton extends LitElement {
|
|||||||
--control-button-background-opacity: 0.2;
|
--control-button-background-opacity: 0.2;
|
||||||
--control-button-border-radius: 10px;
|
--control-button-border-radius: 10px;
|
||||||
--mdc-icon-size: 20px;
|
--mdc-icon-size: 20px;
|
||||||
|
color: var(--primary-text-color);
|
||||||
width: 40px;
|
width: 40px;
|
||||||
height: 40px;
|
height: 40px;
|
||||||
-webkit-tap-highlight-color: transparent;
|
-webkit-tap-highlight-color: transparent;
|
||||||
@@ -107,8 +108,11 @@ export class HaControlButton extends LitElement {
|
|||||||
outline: none;
|
outline: none;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
background: none;
|
background: none;
|
||||||
z-index: 1;
|
|
||||||
--mdc-ripple-color: var(--control-button-background-color);
|
--mdc-ripple-color: var(--control-button-background-color);
|
||||||
|
/* For safari border-radius overflow */
|
||||||
|
z-index: 0;
|
||||||
|
font-size: inherit;
|
||||||
|
color: inherit;
|
||||||
}
|
}
|
||||||
.button::before {
|
.button::before {
|
||||||
content: "";
|
content: "";
|
||||||
|
336
src/components/ha-control-select.ts
Normal file
336
src/components/ha-control-select.ts
Normal file
@@ -0,0 +1,336 @@
|
|||||||
|
import {
|
||||||
|
css,
|
||||||
|
CSSResultGroup,
|
||||||
|
html,
|
||||||
|
LitElement,
|
||||||
|
nothing,
|
||||||
|
PropertyValues,
|
||||||
|
} from "lit";
|
||||||
|
import { customElement, property, state } from "lit/decorators";
|
||||||
|
import { classMap } from "lit/directives/class-map";
|
||||||
|
import { ifDefined } from "lit/directives/if-defined";
|
||||||
|
import { repeat } from "lit/directives/repeat";
|
||||||
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
|
import "./ha-icon";
|
||||||
|
import "./ha-svg-icon";
|
||||||
|
|
||||||
|
export type ControlSelectOption = {
|
||||||
|
value: string;
|
||||||
|
label?: string;
|
||||||
|
icon?: string;
|
||||||
|
path?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
@customElement("ha-control-select")
|
||||||
|
export class HaControlSelect extends LitElement {
|
||||||
|
@property({ type: Boolean, reflect: true }) disabled = false;
|
||||||
|
|
||||||
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public options?: ControlSelectOption[];
|
||||||
|
|
||||||
|
@property() public value?: string;
|
||||||
|
|
||||||
|
@property({ type: Boolean, reflect: true })
|
||||||
|
public vertical = false;
|
||||||
|
|
||||||
|
@property({ type: Boolean, attribute: "hide-label" })
|
||||||
|
public hideLabel = false;
|
||||||
|
|
||||||
|
@state() private _activeIndex?: number;
|
||||||
|
|
||||||
|
protected firstUpdated(changedProperties: PropertyValues): void {
|
||||||
|
super.firstUpdated(changedProperties);
|
||||||
|
this.setAttribute("role", "listbox");
|
||||||
|
if (!this.hasAttribute("tabindex")) {
|
||||||
|
this.setAttribute("tabindex", "0");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected updated(changedProps: PropertyValues) {
|
||||||
|
super.updated(changedProps);
|
||||||
|
if (changedProps.has("_activeIndex")) {
|
||||||
|
const activeValue =
|
||||||
|
this._activeIndex != null
|
||||||
|
? this.options?.[this._activeIndex]?.value
|
||||||
|
: undefined;
|
||||||
|
const activedescendant =
|
||||||
|
activeValue != null ? `option-${activeValue}` : undefined;
|
||||||
|
this.setAttribute("aria-activedescendant", activedescendant ?? "");
|
||||||
|
}
|
||||||
|
if (changedProps.has("vertical")) {
|
||||||
|
const orientation = this.vertical ? "vertical" : "horizontal";
|
||||||
|
this.setAttribute("aria-orientation", orientation);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public connectedCallback(): void {
|
||||||
|
super.connectedCallback();
|
||||||
|
this._setupListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
public disconnectedCallback(): void {
|
||||||
|
super.disconnectedCallback();
|
||||||
|
this._destroyListeners();
|
||||||
|
}
|
||||||
|
|
||||||
|
private _setupListeners() {
|
||||||
|
this.addEventListener("focus", this._handleFocus);
|
||||||
|
this.addEventListener("blur", this._handleBlur);
|
||||||
|
this.addEventListener("keydown", this._handleKeydown);
|
||||||
|
}
|
||||||
|
|
||||||
|
private _destroyListeners() {
|
||||||
|
this.removeEventListener("focus", this._handleFocus);
|
||||||
|
this.removeEventListener("blur", this._handleBlur);
|
||||||
|
this.removeEventListener("keydown", this._handleKeydown);
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleFocus() {
|
||||||
|
if (this.disabled) return;
|
||||||
|
this._activeIndex =
|
||||||
|
(this.value != null
|
||||||
|
? this.options?.findIndex((option) => option.value === this.value)
|
||||||
|
: undefined) ?? 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleBlur() {
|
||||||
|
this._activeIndex = undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleKeydown(ev: KeyboardEvent) {
|
||||||
|
if (!this.options || this._activeIndex == null || this.disabled) return;
|
||||||
|
const value = this.options[this._activeIndex].value;
|
||||||
|
switch (ev.key) {
|
||||||
|
case " ":
|
||||||
|
this.value = value;
|
||||||
|
fireEvent(this, "value-changed", { value });
|
||||||
|
break;
|
||||||
|
case "ArrowUp":
|
||||||
|
case "ArrowLeft":
|
||||||
|
this._activeIndex =
|
||||||
|
this._activeIndex <= 0
|
||||||
|
? this.options.length - 1
|
||||||
|
: this._activeIndex - 1;
|
||||||
|
break;
|
||||||
|
case "ArrowDown":
|
||||||
|
case "ArrowRight":
|
||||||
|
this._activeIndex = (this._activeIndex + 1) % this.options.length;
|
||||||
|
break;
|
||||||
|
case "Home":
|
||||||
|
this._activeIndex = 0;
|
||||||
|
break;
|
||||||
|
case "End":
|
||||||
|
this._activeIndex = this.options.length - 1;
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
ev.preventDefault();
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleOptionClick(ev: MouseEvent) {
|
||||||
|
if (this.disabled) return;
|
||||||
|
const value = (ev.target as any).value;
|
||||||
|
this.value = value;
|
||||||
|
fireEvent(this, "value-changed", { value });
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleOptionMouseDown(ev: MouseEvent) {
|
||||||
|
if (this.disabled) return;
|
||||||
|
ev.preventDefault();
|
||||||
|
const value = (ev.target as any).value;
|
||||||
|
this._activeIndex = this.options?.findIndex(
|
||||||
|
(option) => option.value === value
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleOptionMouseUp(ev: MouseEvent) {
|
||||||
|
ev.preventDefault();
|
||||||
|
this._activeIndex = undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected render() {
|
||||||
|
return html`
|
||||||
|
<div class="container">
|
||||||
|
${this.options
|
||||||
|
? repeat(
|
||||||
|
this.options,
|
||||||
|
(option) => option.value,
|
||||||
|
(option, idx) => this._renderOption(option, idx)
|
||||||
|
)
|
||||||
|
: nothing}
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _renderOption(option: ControlSelectOption, index: number) {
|
||||||
|
return html`
|
||||||
|
<div
|
||||||
|
id=${`option-${option.value}`}
|
||||||
|
class=${classMap({
|
||||||
|
option: true,
|
||||||
|
selected: this.value === option.value,
|
||||||
|
focused: this._activeIndex === index,
|
||||||
|
})}
|
||||||
|
role="option"
|
||||||
|
.value=${option.value}
|
||||||
|
aria-selected=${this.value === option.value}
|
||||||
|
aria-label=${ifDefined(option.label)}
|
||||||
|
title=${ifDefined(option.label)}
|
||||||
|
@click=${this._handleOptionClick}
|
||||||
|
@mousedown=${this._handleOptionMouseDown}
|
||||||
|
@mouseup=${this._handleOptionMouseUp}
|
||||||
|
>
|
||||||
|
<div class="content">
|
||||||
|
${option.path
|
||||||
|
? html`<ha-svg-icon .path=${option.path}></ha-svg-icon>`
|
||||||
|
: option.icon
|
||||||
|
? html`<ha-icon .icon=${option.icon}></ha-icon> `
|
||||||
|
: nothing}
|
||||||
|
${option.label && !this.hideLabel
|
||||||
|
? html`<span>${option.label}</span>`
|
||||||
|
: nothing}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles(): CSSResultGroup {
|
||||||
|
return css`
|
||||||
|
:host {
|
||||||
|
display: block;
|
||||||
|
--control-select-color: var(--primary-color);
|
||||||
|
--control-select-focused-opacity: 0.2;
|
||||||
|
--control-select-selected-opacity: 1;
|
||||||
|
--control-select-background: var(--disabled-color);
|
||||||
|
--control-select-background-opacity: 0.2;
|
||||||
|
--control-select-thickness: 40px;
|
||||||
|
--control-select-border-radius: 10px;
|
||||||
|
--control-select-padding: 4px;
|
||||||
|
--control-select-button-border-radius: calc(
|
||||||
|
var(--control-select-border-radius) - var(--control-select-padding)
|
||||||
|
);
|
||||||
|
--mdc-icon-size: 20px;
|
||||||
|
height: var(--control-select-thickness);
|
||||||
|
width: 100%;
|
||||||
|
border-radius: var(--control-select-border-radius);
|
||||||
|
outline: none;
|
||||||
|
transition: box-shadow 180ms ease-in-out;
|
||||||
|
font-style: normal;
|
||||||
|
font-weight: 500;
|
||||||
|
user-select: none;
|
||||||
|
-webkit-tap-highlight-color: transparent;
|
||||||
|
}
|
||||||
|
:host(:focus-visible) {
|
||||||
|
box-shadow: 0 0 0 2px var(--control-select-color);
|
||||||
|
}
|
||||||
|
:host([vertical]) {
|
||||||
|
width: var(--control-select-thickness);
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
.container {
|
||||||
|
position: relative;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
border-radius: var(--control-select-border-radius);
|
||||||
|
transform: translateZ(0);
|
||||||
|
overflow: hidden;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
padding: var(--control-select-padding);
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
.container::before {
|
||||||
|
position: absolute;
|
||||||
|
content: "";
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
background: var(--control-select-background);
|
||||||
|
opacity: var(--control-select-background-opacity);
|
||||||
|
}
|
||||||
|
|
||||||
|
.container > *:not(:last-child) {
|
||||||
|
margin-right: var(--control-select-padding);
|
||||||
|
margin-inline-end: var(--control-select-padding);
|
||||||
|
margin-inline-start: initial;
|
||||||
|
direction: var(--direction);
|
||||||
|
}
|
||||||
|
.option {
|
||||||
|
cursor: pointer;
|
||||||
|
position: relative;
|
||||||
|
flex: 1;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
border-radius: var(--control-select-button-border-radius);
|
||||||
|
overflow: hidden;
|
||||||
|
color: var(--primary-text-color);
|
||||||
|
/* For safari border-radius overflow */
|
||||||
|
z-index: 0;
|
||||||
|
}
|
||||||
|
.content > *:not(:last-child) {
|
||||||
|
margin-bottom: 4px;
|
||||||
|
}
|
||||||
|
.option::before {
|
||||||
|
position: absolute;
|
||||||
|
content: "";
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
background-color: var(--control-select-color);
|
||||||
|
opacity: 0;
|
||||||
|
transition: background-color ease-in-out 180ms, opacity ease-in-out 80ms;
|
||||||
|
}
|
||||||
|
.option.focused::before,
|
||||||
|
.option:hover::before {
|
||||||
|
opacity: var(--control-select-focused-opacity);
|
||||||
|
}
|
||||||
|
.option.selected {
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
.option.selected::before {
|
||||||
|
opacity: var(--control-select-selected-opacity);
|
||||||
|
}
|
||||||
|
.option .content {
|
||||||
|
position: relative;
|
||||||
|
pointer-events: none;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
flex-direction: column;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
:host([vertical]) {
|
||||||
|
width: var(--control-select-thickness);
|
||||||
|
height: auto;
|
||||||
|
}
|
||||||
|
:host([vertical]) .container {
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
:host([vertical]) .container > *:not(:last-child) {
|
||||||
|
margin-right: initial;
|
||||||
|
margin-inline-end: initial;
|
||||||
|
margin-bottom: var(--control-select-padding);
|
||||||
|
}
|
||||||
|
:host([disabled]) {
|
||||||
|
--control-select-color: var(--disabled-color);
|
||||||
|
--control-select-focused-opacity: 0;
|
||||||
|
}
|
||||||
|
:host([disabled]) .option {
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"ha-control-select": HaControlSelect;
|
||||||
|
}
|
||||||
|
}
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user