mirror of
https://github.com/home-assistant/frontend.git
synced 2025-07-31 13:07:49 +00:00
Merge pull request #7833 from home-assistant/dev
This commit is contained in:
commit
50e7410002
13
.devcontainer/Dockerfile
Normal file
13
.devcontainer/Dockerfile
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
# See here for image contents: https://github.com/microsoft/vscode-dev-containers/tree/v0.148.1/containers/python-3/.devcontainer/base.Dockerfile
|
||||||
|
FROM mcr.microsoft.com/vscode/devcontainers/python:0-3.9
|
||||||
|
|
||||||
|
ENV \
|
||||||
|
DEBIAN_FRONTEND=noninteractive \
|
||||||
|
DEVCONTAINER=true \
|
||||||
|
PATH=$PATH:./node_modules/.bin
|
||||||
|
|
||||||
|
# Install nvm
|
||||||
|
COPY .nvmrc /tmp/.nvmrc
|
||||||
|
RUN \
|
||||||
|
su vscode -c \
|
||||||
|
"source /usr/local/share/nvm/nvm.sh && nvm install $(cat /tmp/.nvmrc) 2>&1"
|
31
.devcontainer/devcontainer.json
Normal file
31
.devcontainer/devcontainer.json
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
{
|
||||||
|
"name": "Home Assistant Frontend",
|
||||||
|
"build": {
|
||||||
|
"dockerfile": "Dockerfile",
|
||||||
|
"context": ".."
|
||||||
|
},
|
||||||
|
"appPort": 8123,
|
||||||
|
"context": "..",
|
||||||
|
"postCreateCommand": "script/bootstrap",
|
||||||
|
"extensions": [
|
||||||
|
"github.vscode-pull-request-github",
|
||||||
|
"dbaeumer.vscode-eslint",
|
||||||
|
"ms-vscode.vscode-typescript-tslint-plugin",
|
||||||
|
"esbenp.prettier-vscode",
|
||||||
|
"bierner.lit-html",
|
||||||
|
"runem.lit-plugin",
|
||||||
|
"ms-python.vscode-pylance"
|
||||||
|
],
|
||||||
|
"settings": {
|
||||||
|
"terminal.integrated.shell.linux": "/bin/bash",
|
||||||
|
"files.eol": "\n",
|
||||||
|
"editor.tabSize": 2,
|
||||||
|
"editor.formatOnPaste": false,
|
||||||
|
"editor.formatOnSave": true,
|
||||||
|
"editor.formatOnType": true,
|
||||||
|
"[typescript]": {
|
||||||
|
"editor.defaultFormatter": "esbenp.prettier-vscode"
|
||||||
|
},
|
||||||
|
"files.trimTrailingWhitespace": true
|
||||||
|
}
|
||||||
|
}
|
2
.gitignore
vendored
2
.gitignore
vendored
@ -23,6 +23,8 @@ dist
|
|||||||
# vscode
|
# vscode
|
||||||
.vscode/*
|
.vscode/*
|
||||||
!.vscode/extensions.json
|
!.vscode/extensions.json
|
||||||
|
!.vscode/launch.json
|
||||||
|
!.vscode/tasks.json
|
||||||
|
|
||||||
# Cast dev settings
|
# Cast dev settings
|
||||||
src/cast/dev_const.ts
|
src/cast/dev_const.ts
|
||||||
|
@ -1,6 +0,0 @@
|
|||||||
jshint:
|
|
||||||
enabled: false
|
|
||||||
|
|
||||||
eslint:
|
|
||||||
enabled: true
|
|
||||||
config_file: .eslintrc-hound.json
|
|
44
.vscode/launch.json
vendored
Normal file
44
.vscode/launch.json
vendored
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
{
|
||||||
|
// https://github.com/microsoft/vscode-js-debug/blob/master/OPTIONS.md
|
||||||
|
"configurations": [
|
||||||
|
{
|
||||||
|
"name": "Debug Frontend",
|
||||||
|
"request": "launch",
|
||||||
|
"type": "pwa-chrome",
|
||||||
|
"url": "http://localhost:8123/",
|
||||||
|
"webRoot": "${workspaceFolder}/hass_frontend",
|
||||||
|
"disableNetworkCache": true,
|
||||||
|
"preLaunchTask": "Develop Frontend",
|
||||||
|
"outFiles": [
|
||||||
|
"${workspaceFolder}/hass_frontend/frontend_latest/*.js"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Debug Gallery",
|
||||||
|
"request": "launch",
|
||||||
|
"type": "pwa-chrome",
|
||||||
|
"url": "http://localhost:8100/",
|
||||||
|
"webRoot": "${workspaceFolder}/gallery/dist",
|
||||||
|
"disableNetworkCache": true,
|
||||||
|
"preLaunchTask": "Develop Gallery"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Debug Demo",
|
||||||
|
"request": "launch",
|
||||||
|
"type": "pwa-chrome",
|
||||||
|
"url": "http://localhost:8090/",
|
||||||
|
"webRoot": "${workspaceFolder}/demo/dist",
|
||||||
|
"disableNetworkCache": true,
|
||||||
|
"preLaunchTask": "Develop Demo"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"name": "Debug Cast",
|
||||||
|
"request": "launch",
|
||||||
|
"type": "pwa-chrome",
|
||||||
|
"url": "http://localhost:8080/",
|
||||||
|
"webRoot": "${workspaceFolder}/cast/dist",
|
||||||
|
"disableNetworkCache": true,
|
||||||
|
"preLaunchTask": "Develop Cast"
|
||||||
|
},
|
||||||
|
]
|
||||||
|
}
|
208
.vscode/tasks.json
vendored
Normal file
208
.vscode/tasks.json
vendored
Normal file
@ -0,0 +1,208 @@
|
|||||||
|
{
|
||||||
|
"version": "2.0.0",
|
||||||
|
"tasks": [
|
||||||
|
{
|
||||||
|
"label": "Develop Frontend",
|
||||||
|
"type": "gulp",
|
||||||
|
"task": "develop-app",
|
||||||
|
// Sync changes here to other tasks until issue resolved
|
||||||
|
// https://github.com/Microsoft/vscode/issues/61497
|
||||||
|
"problemMatcher": {
|
||||||
|
"owner": "ha-build",
|
||||||
|
"source": "ha-build",
|
||||||
|
"fileLocation": "absolute",
|
||||||
|
"severity": "error",
|
||||||
|
"pattern": [
|
||||||
|
{
|
||||||
|
"regexp": "(SyntaxError): (.+): (.+) \\((\\d+):(\\d+)\\)",
|
||||||
|
"severity": 1,
|
||||||
|
"file": 2,
|
||||||
|
"message": 3,
|
||||||
|
"line": 4,
|
||||||
|
"column": 5
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"background": {
|
||||||
|
"activeOnStart": true,
|
||||||
|
"beginsPattern": "Changes detected. Starting compilation",
|
||||||
|
"endsPattern": "Build done @"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"isBackground": true,
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"runOptions": {
|
||||||
|
"instanceLimit": 1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Develop Supervisor panel",
|
||||||
|
"type": "gulp",
|
||||||
|
"task": "develop-hassio",
|
||||||
|
"problemMatcher": {
|
||||||
|
"owner": "ha-build",
|
||||||
|
"source": "ha-build",
|
||||||
|
"fileLocation": "absolute",
|
||||||
|
"severity": "error",
|
||||||
|
"pattern": [
|
||||||
|
{
|
||||||
|
"regexp": "(SyntaxError): (.+): (.+) \\((\\d+):(\\d+)\\)",
|
||||||
|
"severity": 1,
|
||||||
|
"file": 2,
|
||||||
|
"message": 3,
|
||||||
|
"line": 4,
|
||||||
|
"column": 5
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"background": {
|
||||||
|
"activeOnStart": true,
|
||||||
|
"beginsPattern": "Changes detected. Starting compilation",
|
||||||
|
"endsPattern": "Build done @"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"isBackground": true,
|
||||||
|
"group": "build",
|
||||||
|
"runOptions": {
|
||||||
|
"instanceLimit": 1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Develop Gallery",
|
||||||
|
"type": "gulp",
|
||||||
|
"task": "develop-gallery",
|
||||||
|
"problemMatcher": {
|
||||||
|
"owner": "ha-build",
|
||||||
|
"source": "ha-build",
|
||||||
|
"fileLocation": "absolute",
|
||||||
|
"severity": "error",
|
||||||
|
"pattern": [
|
||||||
|
{
|
||||||
|
"regexp": "(SyntaxError): (.+): (.+) \\((\\d+):(\\d+)\\)",
|
||||||
|
"severity": 1,
|
||||||
|
"file": 2,
|
||||||
|
"message": 3,
|
||||||
|
"line": 4,
|
||||||
|
"column": 5
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"background": {
|
||||||
|
"activeOnStart": true,
|
||||||
|
"beginsPattern": "Changes detected. Starting compilation",
|
||||||
|
"endsPattern": "Build done @"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
"isBackground": true,
|
||||||
|
"group": "build",
|
||||||
|
"runOptions": {
|
||||||
|
"instanceLimit": 1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Develop Demo",
|
||||||
|
"type": "gulp",
|
||||||
|
"task": "develop-demo",
|
||||||
|
"problemMatcher": {
|
||||||
|
"owner": "ha-build",
|
||||||
|
"source": "ha-build",
|
||||||
|
"fileLocation": "absolute",
|
||||||
|
"severity": "error",
|
||||||
|
"pattern": [
|
||||||
|
{
|
||||||
|
"regexp": "(SyntaxError): (.+): (.+) \\((\\d+):(\\d+)\\)",
|
||||||
|
"severity": 1,
|
||||||
|
"file": 2,
|
||||||
|
"message": 3,
|
||||||
|
"line": 4,
|
||||||
|
"column": 5
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"background": {
|
||||||
|
"activeOnStart": true,
|
||||||
|
"beginsPattern": "Changes detected. Starting compilation",
|
||||||
|
"endsPattern": "Build done @"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
"isBackground": true,
|
||||||
|
"group": "build",
|
||||||
|
"runOptions": {
|
||||||
|
"instanceLimit": 1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Develop Cast",
|
||||||
|
"type": "gulp",
|
||||||
|
"task": "develop-cast",
|
||||||
|
"problemMatcher": {
|
||||||
|
"owner": "ha-build",
|
||||||
|
"source": "ha-build",
|
||||||
|
"fileLocation": "absolute",
|
||||||
|
"severity": "error",
|
||||||
|
"pattern": [
|
||||||
|
{
|
||||||
|
"regexp": "(SyntaxError): (.+): (.+) \\((\\d+):(\\d+)\\)",
|
||||||
|
"severity": 1,
|
||||||
|
"file": 2,
|
||||||
|
"message": 3,
|
||||||
|
"line": 4,
|
||||||
|
"column": 5
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"background": {
|
||||||
|
"activeOnStart": true,
|
||||||
|
"beginsPattern": "Changes detected. Starting compilation",
|
||||||
|
"endsPattern": "Build done @"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
"isBackground": true,
|
||||||
|
"group": "build",
|
||||||
|
"runOptions": {
|
||||||
|
"instanceLimit": 1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Run HA Core in devcontainer",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "script/core",
|
||||||
|
"isBackground": true,
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"problemMatcher": [],
|
||||||
|
"runOptions": {
|
||||||
|
"instanceLimit": 1
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"label": "Run HA Core for Supervisor in devcontainer",
|
||||||
|
"type": "shell",
|
||||||
|
"command": "HASSIO=${input:supervisorHost} HASSIO_TOKEN=${input:supervisorToken} script/core",
|
||||||
|
"isBackground": true,
|
||||||
|
"group": {
|
||||||
|
"kind": "build",
|
||||||
|
"isDefault": true
|
||||||
|
},
|
||||||
|
"problemMatcher": [],
|
||||||
|
"runOptions": {
|
||||||
|
"instanceLimit": 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"inputs": [
|
||||||
|
{
|
||||||
|
"id": "supervisorHost",
|
||||||
|
"type": "promptString",
|
||||||
|
"description": "The IP of the Supervisor host running the Remote API proxy add-on"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "supervisorToken",
|
||||||
|
"type": "promptString",
|
||||||
|
"description": "The token for the Remote API proxy add-on"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
39
build-scripts/README.md
Normal file
39
build-scripts/README.md
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
# Bundling Home Assistant Frontend
|
||||||
|
|
||||||
|
The Home Assistant build pipeline contains various steps to prepare a build.
|
||||||
|
|
||||||
|
- Generating icon files to be included
|
||||||
|
- Generating translation files to be included
|
||||||
|
- Converting TypeScript, CSS and JSON files to JavaScript
|
||||||
|
- Bundling
|
||||||
|
- Minifying the files
|
||||||
|
- Generating the HTML entrypoint files
|
||||||
|
- Generating the service worker
|
||||||
|
- Compressing the files
|
||||||
|
|
||||||
|
## Converting files
|
||||||
|
|
||||||
|
Currently in Home Assistant we use a bundler to convert TypeScript, CSS and JSON files to JavaScript files that the browser understands.
|
||||||
|
|
||||||
|
We currently rely on Webpack but also have experimental Rollup support. Both of these programs bundle the converted files in both production and development.
|
||||||
|
|
||||||
|
For development, bundling is optional. We just want to get the right files in the browser.
|
||||||
|
|
||||||
|
Responsibilities of the converter during development:
|
||||||
|
|
||||||
|
- Convert TypeScript to JavaScript
|
||||||
|
- Convert CSS to JavaScript that sets the content as the default export
|
||||||
|
- Convert JSON to JavaScript that sets the content as the default export
|
||||||
|
- Make sure import, dynamic import and web worker references work
|
||||||
|
- Add extensions where missing
|
||||||
|
- Resolve absolute package imports
|
||||||
|
- Filter out specific imports/packages
|
||||||
|
- Replace constants with values
|
||||||
|
|
||||||
|
In production, the following responsibilities are added:
|
||||||
|
|
||||||
|
- Minify HTML
|
||||||
|
- Bundle multiple imports so that the browser can fetch less files
|
||||||
|
- Generate a second version that is ES5 compatible
|
||||||
|
|
||||||
|
Configuration for all these steps are specified in [bundle.js](bundle.js).
|
@ -44,7 +44,7 @@ module.exports.definedVars = ({ isProdBuild, latestBuild, defineOverlay }) => ({
|
|||||||
});
|
});
|
||||||
|
|
||||||
module.exports.terserOptions = (latestBuild) => ({
|
module.exports.terserOptions = (latestBuild) => ({
|
||||||
safari10: true,
|
safari10: !latestBuild,
|
||||||
ecma: latestBuild ? undefined : 5,
|
ecma: latestBuild ? undefined : 5,
|
||||||
output: { comments: false },
|
output: { comments: false },
|
||||||
});
|
});
|
||||||
@ -117,7 +117,7 @@ BundleConfig {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
module.exports.config = {
|
module.exports.config = {
|
||||||
app({ isProdBuild, latestBuild, isStatsBuild }) {
|
app({ isProdBuild, latestBuild, isStatsBuild, isWDS }) {
|
||||||
return {
|
return {
|
||||||
entry: {
|
entry: {
|
||||||
service_worker: "./src/entrypoints/service_worker.ts",
|
service_worker: "./src/entrypoints/service_worker.ts",
|
||||||
@ -132,6 +132,7 @@ module.exports.config = {
|
|||||||
isProdBuild,
|
isProdBuild,
|
||||||
latestBuild,
|
latestBuild,
|
||||||
isStatsBuild,
|
isStatsBuild,
|
||||||
|
isWDS,
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -6,6 +6,9 @@ module.exports = {
|
|||||||
useRollup() {
|
useRollup() {
|
||||||
return process.env.ROLLUP === "1";
|
return process.env.ROLLUP === "1";
|
||||||
},
|
},
|
||||||
|
useWDS() {
|
||||||
|
return process.env.WDS === "1";
|
||||||
|
},
|
||||||
isProdBuild() {
|
isProdBuild() {
|
||||||
return (
|
return (
|
||||||
process.env.NODE_ENV === "production" || module.exports.isStatsBuild()
|
process.env.NODE_ENV === "production" || module.exports.isStatsBuild()
|
||||||
|
@ -12,6 +12,7 @@ require("./webpack.js");
|
|||||||
require("./service-worker.js");
|
require("./service-worker.js");
|
||||||
require("./entry-html.js");
|
require("./entry-html.js");
|
||||||
require("./rollup.js");
|
require("./rollup.js");
|
||||||
|
require("./wds.js");
|
||||||
|
|
||||||
gulp.task(
|
gulp.task(
|
||||||
"develop-app",
|
"develop-app",
|
||||||
@ -28,7 +29,11 @@ gulp.task(
|
|||||||
"build-translations"
|
"build-translations"
|
||||||
),
|
),
|
||||||
"copy-static-app",
|
"copy-static-app",
|
||||||
env.useRollup() ? "rollup-watch-app" : "webpack-watch-app"
|
env.useWDS()
|
||||||
|
? "wds-watch-app"
|
||||||
|
: env.useRollup()
|
||||||
|
? "rollup-watch-app"
|
||||||
|
: "webpack-watch-app"
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -19,6 +19,7 @@ const renderTemplate = (pth, data = {}, pathFunc = templatePath) => {
|
|||||||
return compiled({
|
return compiled({
|
||||||
...data,
|
...data,
|
||||||
useRollup: env.useRollup(),
|
useRollup: env.useRollup(),
|
||||||
|
useWDS: env.useWDS(),
|
||||||
renderTemplate,
|
renderTemplate,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
@ -90,10 +91,23 @@ gulp.task("gen-pages-prod", (done) => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
gulp.task("gen-index-app-dev", (done) => {
|
gulp.task("gen-index-app-dev", (done) => {
|
||||||
|
let latestAppJS, latestCoreJS, latestCustomPanelJS;
|
||||||
|
|
||||||
|
if (env.useWDS()) {
|
||||||
|
latestAppJS = "http://localhost:8000/src/entrypoints/app.ts";
|
||||||
|
latestCoreJS = "http://localhost:8000/src/entrypoints/core.ts";
|
||||||
|
latestCustomPanelJS =
|
||||||
|
"http://localhost:8000/src/entrypoints/custom-panel.ts";
|
||||||
|
} else {
|
||||||
|
latestAppJS = "/frontend_latest/app.js";
|
||||||
|
latestCoreJS = "/frontend_latest/core.js";
|
||||||
|
latestCustomPanelJS = "/frontend_latest/custom-panel.js";
|
||||||
|
}
|
||||||
|
|
||||||
const content = renderTemplate("index", {
|
const content = renderTemplate("index", {
|
||||||
latestAppJS: "/frontend_latest/app.js",
|
latestAppJS,
|
||||||
latestCoreJS: "/frontend_latest/core.js",
|
latestCoreJS,
|
||||||
latestCustomPanelJS: "/frontend_latest/custom-panel.js",
|
latestCustomPanelJS,
|
||||||
|
|
||||||
es5AppJS: "/frontend_es5/app.js",
|
es5AppJS: "/frontend_es5/app.js",
|
||||||
es5CoreJS: "/frontend_es5/core.js",
|
es5CoreJS: "/frontend_es5/core.js",
|
||||||
|
@ -33,21 +33,10 @@ String.prototype.rsplit = function (sep, maxsplit) {
|
|||||||
: split;
|
: split;
|
||||||
};
|
};
|
||||||
|
|
||||||
// Panel translations which should be split from the core translations. These
|
// Panel translations which should be split from the core translations.
|
||||||
// should mirror the fragment definitions in polymer.json, so that we load
|
const TRANSLATION_FRAGMENTS = Object.keys(
|
||||||
// additional resources at equivalent points.
|
require("../../src/translations/en.json").ui.panel
|
||||||
const TRANSLATION_FRAGMENTS = [
|
);
|
||||||
"config",
|
|
||||||
"history",
|
|
||||||
"logbook",
|
|
||||||
"mailbox",
|
|
||||||
"profile",
|
|
||||||
"shopping-list",
|
|
||||||
"page-authorize",
|
|
||||||
"page-demo",
|
|
||||||
"page-onboarding",
|
|
||||||
"developer-tools",
|
|
||||||
];
|
|
||||||
|
|
||||||
function recursiveFlatten(prefix, data) {
|
function recursiveFlatten(prefix, data) {
|
||||||
let output = {};
|
let output = {};
|
||||||
|
11
build-scripts/gulp/wds.js
Normal file
11
build-scripts/gulp/wds.js
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
// Tasks to run Rollup
|
||||||
|
const gulp = require("gulp");
|
||||||
|
const { startDevServer } = require("@web/dev-server");
|
||||||
|
|
||||||
|
gulp.task("wds-watch-app", () => {
|
||||||
|
startDevServer({
|
||||||
|
config: {
|
||||||
|
watch: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
});
|
@ -18,6 +18,14 @@ const bothBuilds = (createConfigFunc, params) => [
|
|||||||
createConfigFunc({ ...params, latestBuild: false }),
|
createConfigFunc({ ...params, latestBuild: false }),
|
||||||
];
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @param {{
|
||||||
|
* compiler: import("webpack").Compiler,
|
||||||
|
* contentBase: string,
|
||||||
|
* port: number,
|
||||||
|
* listenHost?: string
|
||||||
|
* }}
|
||||||
|
*/
|
||||||
const runDevServer = ({
|
const runDevServer = ({
|
||||||
compiler,
|
compiler,
|
||||||
contentBase,
|
contentBase,
|
||||||
@ -33,10 +41,13 @@ const runDevServer = ({
|
|||||||
throw err;
|
throw err;
|
||||||
}
|
}
|
||||||
// Server listening
|
// Server listening
|
||||||
log("[webpack-dev-server]", `http://localhost:${port}`);
|
log(
|
||||||
|
"[webpack-dev-server]",
|
||||||
|
`Project is running at http://localhost:${port}`
|
||||||
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
const handler = (done) => (err, stats) => {
|
const doneHandler = (done) => (err, stats) => {
|
||||||
if (err) {
|
if (err) {
|
||||||
log.error(err.stack || err);
|
log.error(err.stack || err);
|
||||||
if (err.details) {
|
if (err.details) {
|
||||||
@ -45,22 +56,31 @@ const handler = (done) => (err, stats) => {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
log(`Build done @ ${new Date().toLocaleTimeString()}`);
|
|
||||||
|
|
||||||
if (stats.hasErrors() || stats.hasWarnings()) {
|
if (stats.hasErrors() || stats.hasWarnings()) {
|
||||||
log.warn(stats.toString("minimal"));
|
console.log(stats.toString("minimal"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
log(`Build done @ ${new Date().toLocaleTimeString()}`);
|
||||||
|
|
||||||
if (done) {
|
if (done) {
|
||||||
done();
|
done();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const prodBuild = (conf) =>
|
||||||
|
new Promise((resolve) => {
|
||||||
|
webpack(
|
||||||
|
conf,
|
||||||
|
// Resolve promise when done. Because we pass a callback, webpack closes itself
|
||||||
|
doneHandler(resolve)
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
gulp.task("webpack-watch-app", () => {
|
gulp.task("webpack-watch-app", () => {
|
||||||
// we are not calling done, so this command will run forever
|
// This command will run forever because we don't close compiler
|
||||||
webpack(createAppConfig({ isProdBuild: false, latestBuild: true })).watch(
|
webpack(createAppConfig({ isProdBuild: false, latestBuild: true })).watch(
|
||||||
{ ignored: /build-translations/ },
|
{ ignored: /build-translations/ },
|
||||||
handler()
|
doneHandler()
|
||||||
);
|
);
|
||||||
gulp.watch(
|
gulp.watch(
|
||||||
path.join(paths.translations_src, "en.json"),
|
path.join(paths.translations_src, "en.json"),
|
||||||
@ -68,15 +88,12 @@ gulp.task("webpack-watch-app", () => {
|
|||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task(
|
gulp.task("webpack-prod-app", () =>
|
||||||
"webpack-prod-app",
|
prodBuild(
|
||||||
() =>
|
bothBuilds(createAppConfig, {
|
||||||
new Promise((resolve) =>
|
isProdBuild: true,
|
||||||
webpack(
|
})
|
||||||
bothBuilds(createAppConfig, { isProdBuild: true }),
|
)
|
||||||
handler(resolve)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
gulp.task("webpack-dev-server-demo", () => {
|
gulp.task("webpack-dev-server-demo", () => {
|
||||||
@ -87,17 +104,12 @@ gulp.task("webpack-dev-server-demo", () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task(
|
gulp.task("webpack-prod-demo", () =>
|
||||||
"webpack-prod-demo",
|
prodBuild(
|
||||||
() =>
|
bothBuilds(createDemoConfig, {
|
||||||
new Promise((resolve) =>
|
isProdBuild: true,
|
||||||
webpack(
|
})
|
||||||
bothBuilds(createDemoConfig, {
|
)
|
||||||
isProdBuild: true,
|
|
||||||
}),
|
|
||||||
handler(resolve)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
gulp.task("webpack-dev-server-cast", () => {
|
gulp.task("webpack-dev-server-cast", () => {
|
||||||
@ -110,41 +122,30 @@ gulp.task("webpack-dev-server-cast", () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task(
|
gulp.task("webpack-prod-cast", () =>
|
||||||
"webpack-prod-cast",
|
prodBuild(
|
||||||
() =>
|
bothBuilds(createCastConfig, {
|
||||||
new Promise((resolve) =>
|
isProdBuild: true,
|
||||||
webpack(
|
})
|
||||||
bothBuilds(createCastConfig, {
|
)
|
||||||
isProdBuild: true,
|
|
||||||
}),
|
|
||||||
|
|
||||||
handler(resolve)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
gulp.task("webpack-watch-hassio", () => {
|
gulp.task("webpack-watch-hassio", () => {
|
||||||
// we are not calling done, so this command will run forever
|
// This command will run forever because we don't close compiler
|
||||||
webpack(
|
webpack(
|
||||||
createHassioConfig({
|
createHassioConfig({
|
||||||
isProdBuild: false,
|
isProdBuild: false,
|
||||||
latestBuild: true,
|
latestBuild: true,
|
||||||
})
|
})
|
||||||
).watch({}, handler());
|
).watch({}, doneHandler());
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task(
|
gulp.task("webpack-prod-hassio", () =>
|
||||||
"webpack-prod-hassio",
|
prodBuild(
|
||||||
() =>
|
bothBuilds(createHassioConfig, {
|
||||||
new Promise((resolve) =>
|
isProdBuild: true,
|
||||||
webpack(
|
})
|
||||||
bothBuilds(createHassioConfig, {
|
)
|
||||||
isProdBuild: true,
|
|
||||||
}),
|
|
||||||
handler(resolve)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
gulp.task("webpack-dev-server-gallery", () => {
|
gulp.task("webpack-dev-server-gallery", () => {
|
||||||
@ -156,17 +157,11 @@ gulp.task("webpack-dev-server-gallery", () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
gulp.task(
|
gulp.task("webpack-prod-gallery", () =>
|
||||||
"webpack-prod-gallery",
|
prodBuild(
|
||||||
() =>
|
createGalleryConfig({
|
||||||
new Promise((resolve) =>
|
isProdBuild: true,
|
||||||
webpack(
|
latestBuild: true,
|
||||||
createGalleryConfig({
|
})
|
||||||
isProdBuild: true,
|
)
|
||||||
latestBuild: true,
|
|
||||||
}),
|
|
||||||
|
|
||||||
handler(resolve)
|
|
||||||
)
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
var path = require("path");
|
const path = require("path");
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
polymer_dir: path.resolve(__dirname, ".."),
|
polymer_dir: path.resolve(__dirname, ".."),
|
||||||
|
@ -1,5 +1,3 @@
|
|||||||
const path = require("path");
|
|
||||||
|
|
||||||
module.exports = function (userOptions = {}) {
|
module.exports = function (userOptions = {}) {
|
||||||
// Files need to be absolute paths.
|
// Files need to be absolute paths.
|
||||||
// This only works if the file has no exports
|
// This only works if the file has no exports
|
||||||
|
@ -3,7 +3,7 @@ const path = require("path");
|
|||||||
const commonjs = require("@rollup/plugin-commonjs");
|
const commonjs = require("@rollup/plugin-commonjs");
|
||||||
const resolve = require("@rollup/plugin-node-resolve");
|
const resolve = require("@rollup/plugin-node-resolve");
|
||||||
const json = require("@rollup/plugin-json");
|
const json = require("@rollup/plugin-json");
|
||||||
const babel = require("rollup-plugin-babel");
|
const babel = require("@rollup/plugin-babel").babel;
|
||||||
const replace = require("@rollup/plugin-replace");
|
const replace = require("@rollup/plugin-replace");
|
||||||
const visualizer = require("rollup-plugin-visualizer");
|
const visualizer = require("rollup-plugin-visualizer");
|
||||||
const { string } = require("rollup-plugin-string");
|
const { string } = require("rollup-plugin-string");
|
||||||
@ -31,6 +31,7 @@ const createRollupConfig = ({
|
|||||||
isStatsBuild,
|
isStatsBuild,
|
||||||
publicPath,
|
publicPath,
|
||||||
dontHash,
|
dontHash,
|
||||||
|
isWDS,
|
||||||
}) => {
|
}) => {
|
||||||
return {
|
return {
|
||||||
/**
|
/**
|
||||||
@ -61,6 +62,7 @@ const createRollupConfig = ({
|
|||||||
...bundle.babelOptions({ latestBuild }),
|
...bundle.babelOptions({ latestBuild }),
|
||||||
extensions,
|
extensions,
|
||||||
exclude: bundle.babelExclude(),
|
exclude: bundle.babelExclude(),
|
||||||
|
babelHelpers: isWDS ? "inline" : "bundled",
|
||||||
}),
|
}),
|
||||||
string({
|
string({
|
||||||
// Import certain extensions as strings
|
// Import certain extensions as strings
|
||||||
@ -69,19 +71,21 @@ const createRollupConfig = ({
|
|||||||
replace(
|
replace(
|
||||||
bundle.definedVars({ isProdBuild, latestBuild, defineOverlay })
|
bundle.definedVars({ isProdBuild, latestBuild, defineOverlay })
|
||||||
),
|
),
|
||||||
manifest({
|
!isWDS &&
|
||||||
publicPath,
|
manifest({
|
||||||
}),
|
publicPath,
|
||||||
worker(),
|
}),
|
||||||
dontHashPlugin({ dontHash }),
|
!isWDS && worker(),
|
||||||
isProdBuild && terser(bundle.terserOptions(latestBuild)),
|
!isWDS && dontHashPlugin({ dontHash }),
|
||||||
isStatsBuild &&
|
!isWDS && isProdBuild && terser(bundle.terserOptions(latestBuild)),
|
||||||
|
!isWDS &&
|
||||||
|
isStatsBuild &&
|
||||||
visualizer({
|
visualizer({
|
||||||
// https://github.com/btd/rollup-plugin-visualizer#options
|
// https://github.com/btd/rollup-plugin-visualizer#options
|
||||||
open: true,
|
open: true,
|
||||||
sourcemap: true,
|
sourcemap: true,
|
||||||
}),
|
}),
|
||||||
],
|
].filter(Boolean),
|
||||||
},
|
},
|
||||||
/**
|
/**
|
||||||
* @type { import("rollup").OutputOptions }
|
* @type { import("rollup").OutputOptions }
|
||||||
@ -108,12 +112,13 @@ const createRollupConfig = ({
|
|||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild }) => {
|
const createAppConfig = ({ isProdBuild, latestBuild, isStatsBuild, isWDS }) => {
|
||||||
return createRollupConfig(
|
return createRollupConfig(
|
||||||
bundle.config.app({
|
bundle.config.app({
|
||||||
isProdBuild,
|
isProdBuild,
|
||||||
latestBuild,
|
latestBuild,
|
||||||
isStatsBuild,
|
isStatsBuild,
|
||||||
|
isWDS,
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
@ -4,6 +4,21 @@ const TerserPlugin = require("terser-webpack-plugin");
|
|||||||
const ManifestPlugin = require("webpack-manifest-plugin");
|
const ManifestPlugin = require("webpack-manifest-plugin");
|
||||||
const paths = require("./paths.js");
|
const paths = require("./paths.js");
|
||||||
const bundle = require("./bundle");
|
const bundle = require("./bundle");
|
||||||
|
const log = require("fancy-log");
|
||||||
|
|
||||||
|
class LogStartCompilePlugin {
|
||||||
|
ignoredFirst = false;
|
||||||
|
|
||||||
|
apply(compiler) {
|
||||||
|
compiler.hooks.beforeCompile.tap("LogStartCompilePlugin", () => {
|
||||||
|
if (!this.ignoredFirst) {
|
||||||
|
this.ignoredFirst = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
log("Changes detected. Starting compilation");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const createWebpackConfig = ({
|
const createWebpackConfig = ({
|
||||||
entry,
|
entry,
|
||||||
@ -104,7 +119,8 @@ const createWebpackConfig = ({
|
|||||||
),
|
),
|
||||||
path.resolve(paths.polymer_dir, "src/resources/EventTarget-ponyfill.js")
|
path.resolve(paths.polymer_dir, "src/resources/EventTarget-ponyfill.js")
|
||||||
),
|
),
|
||||||
],
|
!isProdBuild && new LogStartCompilePlugin(),
|
||||||
|
].filter(Boolean),
|
||||||
resolve: {
|
resolve: {
|
||||||
extensions: [".ts", ".js", ".json"],
|
extensions: [".ts", ".js", ".json"],
|
||||||
},
|
},
|
||||||
|
@ -3,22 +3,10 @@ import { Lovelace } from "../../../src/panels/lovelace/types";
|
|||||||
import { DemoConfig } from "./types";
|
import { DemoConfig } from "./types";
|
||||||
|
|
||||||
export const demoConfigs: Array<() => Promise<DemoConfig>> = [
|
export const demoConfigs: Array<() => Promise<DemoConfig>> = [
|
||||||
() =>
|
() => import("./arsaboo").then((mod) => mod.demoArsaboo),
|
||||||
import(/* webpackChunkName: "arsaboo" */ "./arsaboo").then(
|
() => import("./teachingbirds").then((mod) => mod.demoTeachingbirds),
|
||||||
(mod) => mod.demoArsaboo
|
() => import("./kernehed").then((mod) => mod.demoKernehed),
|
||||||
),
|
() => import("./jimpower").then((mod) => mod.demoJimpower),
|
||||||
() =>
|
|
||||||
import(/* webpackChunkName: "teachingbirds" */ "./teachingbirds").then(
|
|
||||||
(mod) => mod.demoTeachingbirds
|
|
||||||
),
|
|
||||||
() =>
|
|
||||||
import(/* webpackChunkName: "kernehed" */ "./kernehed").then(
|
|
||||||
(mod) => mod.demoKernehed
|
|
||||||
),
|
|
||||||
() =>
|
|
||||||
import(/* webpackChunkName: "jimpower" */ "./jimpower").then(
|
|
||||||
(mod) => mod.demoJimpower
|
|
||||||
),
|
|
||||||
];
|
];
|
||||||
|
|
||||||
// eslint-disable-next-line import/no-mutable-exports
|
// eslint-disable-next-line import/no-mutable-exports
|
||||||
|
@ -9,5 +9,5 @@ export interface DemoConfig {
|
|||||||
authorUrl: string;
|
authorUrl: string;
|
||||||
lovelace: (localize: LocalizeFunc) => LovelaceConfig;
|
lovelace: (localize: LocalizeFunc) => LovelaceConfig;
|
||||||
entities: (localize: LocalizeFunc) => Entity[];
|
entities: (localize: LocalizeFunc) => Entity[];
|
||||||
theme: () => { [key: string]: string } | null;
|
theme: () => Record<string, string> | null;
|
||||||
}
|
}
|
||||||
|
@ -7,7 +7,5 @@ import "./ha-demo";
|
|||||||
|
|
||||||
/* polyfill for paper-dropdown */
|
/* polyfill for paper-dropdown */
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
import(
|
import("web-animations-js/web-animations-next-lite.min");
|
||||||
/* webpackChunkName: "polyfill-web-animations-next" */ "web-animations-js/web-animations-next-lite.min"
|
|
||||||
);
|
|
||||||
}, 1000);
|
}, 1000);
|
||||||
|
@ -21,15 +21,16 @@ class DemoCard extends PolymerElement {
|
|||||||
}
|
}
|
||||||
pre {
|
pre {
|
||||||
width: 400px;
|
width: 400px;
|
||||||
margin: 16px;
|
margin: 0 16px;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
|
color: var(--primary-text-color);
|
||||||
}
|
}
|
||||||
@media only screen and (max-width: 800px) {
|
@media only screen and (max-width: 800px) {
|
||||||
.root {
|
.root {
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
pre {
|
pre {
|
||||||
margin-left: 0;
|
margin: 16px 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
@ -26,8 +26,9 @@ class DemoMoreInfo extends PolymerElement {
|
|||||||
|
|
||||||
pre {
|
pre {
|
||||||
width: 400px;
|
width: 400px;
|
||||||
margin: 16px;
|
margin: 0 16px;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
|
color: var(--primary-text-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
@media only screen and (max-width: 800px) {
|
@media only screen and (max-width: 800px) {
|
||||||
@ -35,7 +36,7 @@ class DemoMoreInfo extends PolymerElement {
|
|||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
}
|
}
|
||||||
pre {
|
pre {
|
||||||
margin-left: 0;
|
margin: 16px 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
@ -73,13 +73,7 @@ const CONFIGS = [
|
|||||||
|
|
||||||
class DemoAlarmPanelEntity extends PolymerElement {
|
class DemoAlarmPanelEntity extends PolymerElement {
|
||||||
static get template() {
|
static get template() {
|
||||||
return html`
|
return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
|
||||||
<demo-cards
|
|
||||||
id="demos"
|
|
||||||
hass="[[hass]]"
|
|
||||||
configs="[[_configs]]"
|
|
||||||
></demo-cards>
|
|
||||||
`;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static get properties() {
|
static get properties() {
|
||||||
@ -88,7 +82,6 @@ class DemoAlarmPanelEntity extends PolymerElement {
|
|||||||
type: Object,
|
type: Object,
|
||||||
value: CONFIGS,
|
value: CONFIGS,
|
||||||
},
|
},
|
||||||
hass: Object,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -55,13 +55,7 @@ const CONFIGS = [
|
|||||||
|
|
||||||
class DemoConditional extends PolymerElement {
|
class DemoConditional extends PolymerElement {
|
||||||
static get template() {
|
static get template() {
|
||||||
return html`
|
return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
|
||||||
<demo-cards
|
|
||||||
id="demos"
|
|
||||||
hass="[[hass]]"
|
|
||||||
configs="[[_configs]]"
|
|
||||||
></demo-cards>
|
|
||||||
`;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static get properties() {
|
static get properties() {
|
||||||
@ -70,7 +64,6 @@ class DemoConditional extends PolymerElement {
|
|||||||
type: Object,
|
type: Object,
|
||||||
value: CONFIGS,
|
value: CONFIGS,
|
||||||
},
|
},
|
||||||
hass: Object,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -20,10 +20,10 @@ const CONFIGS = [
|
|||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
heading: "With Name",
|
heading: "With Name (defined in card)",
|
||||||
config: `
|
config: `
|
||||||
- type: button
|
- type: button
|
||||||
name: Bedroom
|
name: Custom Name
|
||||||
entity: light.bed_light
|
entity: light.bed_light
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
@ -32,7 +32,7 @@ const CONFIGS = [
|
|||||||
config: `
|
config: `
|
||||||
- type: button
|
- type: button
|
||||||
entity: light.bed_light
|
entity: light.bed_light
|
||||||
icon: mdi:hotel
|
icon: mdi:tools
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -71,13 +71,7 @@ const CONFIGS = [
|
|||||||
|
|
||||||
class DemoButtonEntity extends PolymerElement {
|
class DemoButtonEntity extends PolymerElement {
|
||||||
static get template() {
|
static get template() {
|
||||||
return html`
|
return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
|
||||||
<demo-cards
|
|
||||||
id="demos"
|
|
||||||
hass="[[hass]]"
|
|
||||||
configs="[[_configs]]"
|
|
||||||
></demo-cards>
|
|
||||||
`;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static get properties() {
|
static get properties() {
|
||||||
@ -86,7 +80,6 @@ class DemoButtonEntity extends PolymerElement {
|
|||||||
type: Object,
|
type: Object,
|
||||||
value: CONFIGS,
|
value: CONFIGS,
|
||||||
},
|
},
|
||||||
hass: Object,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -7,6 +7,8 @@ import "../components/demo-cards";
|
|||||||
|
|
||||||
const ENTITIES = [
|
const ENTITIES = [
|
||||||
getEntity("sensor", "brightness", "12", {}),
|
getEntity("sensor", "brightness", "12", {}),
|
||||||
|
getEntity("sensor", "brightness_medium", "53", {}),
|
||||||
|
getEntity("sensor", "brightness_high", "87", {}),
|
||||||
getEntity("plant", "bonsai", "ok", {}),
|
getEntity("plant", "bonsai", "ok", {}),
|
||||||
getEntity("sensor", "not_working", "unavailable", {}),
|
getEntity("sensor", "not_working", "unavailable", {}),
|
||||||
getEntity("sensor", "outside_humidity", "54", {
|
getEntity("sensor", "outside_humidity", "54", {
|
||||||
@ -21,16 +23,10 @@ const CONFIGS = [
|
|||||||
{
|
{
|
||||||
heading: "Basic example",
|
heading: "Basic example",
|
||||||
config: `
|
config: `
|
||||||
- type: gauge
|
|
||||||
entity: sensor.brightness
|
|
||||||
`,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
heading: "With title",
|
|
||||||
config: `
|
|
||||||
- type: gauge
|
- type: gauge
|
||||||
title: Humidity
|
title: Humidity
|
||||||
entity: sensor.outside_humidity
|
entity: sensor.outside_humidity
|
||||||
|
name: Outside Humidity
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -39,6 +35,7 @@ const CONFIGS = [
|
|||||||
- type: gauge
|
- type: gauge
|
||||||
entity: sensor.outside_temperature
|
entity: sensor.outside_temperature
|
||||||
unit_of_measurement: C
|
unit_of_measurement: C
|
||||||
|
name: Outside Temperature
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -46,19 +43,45 @@ const CONFIGS = [
|
|||||||
config: `
|
config: `
|
||||||
- type: gauge
|
- type: gauge
|
||||||
entity: sensor.brightness
|
entity: sensor.brightness
|
||||||
|
name: Brightness Low
|
||||||
severity:
|
severity:
|
||||||
red: 32
|
red: 75
|
||||||
green: 0
|
green: 0
|
||||||
yellow: 23
|
yellow: 50
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
heading: "Setting Min and Max Values",
|
heading: "Setting Severity Levels",
|
||||||
|
config: `
|
||||||
|
- type: gauge
|
||||||
|
entity: sensor.brightness_medium
|
||||||
|
name: Brightness Medium
|
||||||
|
severity:
|
||||||
|
red: 75
|
||||||
|
green: 0
|
||||||
|
yellow: 50
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Setting Severity Levels",
|
||||||
|
config: `
|
||||||
|
- type: gauge
|
||||||
|
entity: sensor.brightness_high
|
||||||
|
name: Brightness High
|
||||||
|
severity:
|
||||||
|
red: 75
|
||||||
|
green: 0
|
||||||
|
yellow: 50
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Setting Min (0) and Max (15) Values",
|
||||||
config: `
|
config: `
|
||||||
- type: gauge
|
- type: gauge
|
||||||
entity: sensor.brightness
|
entity: sensor.brightness
|
||||||
|
name: Brightness
|
||||||
min: 0
|
min: 0
|
||||||
max: 38
|
max: 15
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -8,29 +8,43 @@ import "../components/demo-cards";
|
|||||||
const ENTITIES = [
|
const ENTITIES = [
|
||||||
getEntity("light", "bed_light", "on", {
|
getEntity("light", "bed_light", "on", {
|
||||||
friendly_name: "Bed Light",
|
friendly_name: "Bed Light",
|
||||||
brightness: 130,
|
brightness: 255,
|
||||||
}),
|
}),
|
||||||
getEntity("light", "dim", "off", {
|
getEntity("light", "dim_on", "on", {
|
||||||
|
friendly_name: "Dining Room",
|
||||||
|
supported_features: 1,
|
||||||
|
brightness: 100,
|
||||||
|
}),
|
||||||
|
getEntity("light", "dim_off", "off", {
|
||||||
|
friendly_name: "Dining Room",
|
||||||
supported_features: 1,
|
supported_features: 1,
|
||||||
}),
|
}),
|
||||||
getEntity("light", "unavailable", "unavailable", {
|
getEntity("light", "unavailable", "unavailable", {
|
||||||
|
friendly_name: "Lost Light",
|
||||||
supported_features: 1,
|
supported_features: 1,
|
||||||
}),
|
}),
|
||||||
];
|
];
|
||||||
|
|
||||||
const CONFIGS = [
|
const CONFIGS = [
|
||||||
{
|
{
|
||||||
heading: "Basic example",
|
heading: "Switchable Light",
|
||||||
config: `
|
config: `
|
||||||
- type: light
|
- type: light
|
||||||
entity: light.bed_light
|
entity: light.bed_light
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
heading: "Dim",
|
heading: "Dimmable Light On",
|
||||||
config: `
|
config: `
|
||||||
- type: light
|
- type: light
|
||||||
entity: light.dim
|
entity: light.dim_on
|
||||||
|
`,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
heading: "Dimmable Light Off",
|
||||||
|
config: `
|
||||||
|
- type: light
|
||||||
|
entity: light.dim_off
|
||||||
`,
|
`,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
@ -163,13 +163,7 @@ const CONFIGS = [
|
|||||||
|
|
||||||
class DemoMap extends PolymerElement {
|
class DemoMap extends PolymerElement {
|
||||||
static get template() {
|
static get template() {
|
||||||
return html`
|
return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
|
||||||
<demo-cards
|
|
||||||
id="demos"
|
|
||||||
hass="[[hass]]"
|
|
||||||
configs="[[_configs]]"
|
|
||||||
></demo-cards>
|
|
||||||
`;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static get properties() {
|
static get properties() {
|
||||||
@ -178,7 +172,6 @@ class DemoMap extends PolymerElement {
|
|||||||
type: Object,
|
type: Object,
|
||||||
value: CONFIGS,
|
value: CONFIGS,
|
||||||
},
|
},
|
||||||
hass: Object,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -150,13 +150,7 @@ const CONFIGS = [
|
|||||||
|
|
||||||
class DemoHuiMediControlCard extends PolymerElement {
|
class DemoHuiMediControlCard extends PolymerElement {
|
||||||
static get template() {
|
static get template() {
|
||||||
return html`
|
return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
|
||||||
<demo-cards
|
|
||||||
id="demos"
|
|
||||||
hass="[[hass]]"
|
|
||||||
configs="[[_configs]]"
|
|
||||||
></demo-cards>
|
|
||||||
`;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static get properties() {
|
static get properties() {
|
||||||
@ -165,7 +159,6 @@ class DemoHuiMediControlCard extends PolymerElement {
|
|||||||
type: Object,
|
type: Object,
|
||||||
value: CONFIGS,
|
value: CONFIGS,
|
||||||
},
|
},
|
||||||
hass: Object,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -57,13 +57,7 @@ const CONFIGS = [
|
|||||||
|
|
||||||
class DemoHuiMediaPlayerRows extends PolymerElement {
|
class DemoHuiMediaPlayerRows extends PolymerElement {
|
||||||
static get template() {
|
static get template() {
|
||||||
return html`
|
return html` <demo-cards id="demos" configs="[[_configs]]"></demo-cards> `;
|
||||||
<demo-cards
|
|
||||||
id="demos"
|
|
||||||
hass="[[hass]]"
|
|
||||||
configs="[[_configs]]"
|
|
||||||
></demo-cards>
|
|
||||||
`;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static get properties() {
|
static get properties() {
|
||||||
@ -72,7 +66,6 @@ class DemoHuiMediaPlayerRows extends PolymerElement {
|
|||||||
type: Object,
|
type: Object,
|
||||||
value: CONFIGS,
|
value: CONFIGS,
|
||||||
},
|
},
|
||||||
hass: Object,
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -20,48 +20,47 @@ class HaGallery extends PolymerElement {
|
|||||||
static get template() {
|
static get template() {
|
||||||
return html`
|
return html`
|
||||||
<style include="iron-positioning ha-style">
|
<style include="iron-positioning ha-style">
|
||||||
:host {
|
:host {
|
||||||
-ms-user-select: initial;
|
-ms-user-select: initial;
|
||||||
-webkit-user-select: initial;
|
-webkit-user-select: initial;
|
||||||
-moz-user-select: initial;
|
-moz-user-select: initial;
|
||||||
}
|
}
|
||||||
app-header-layout {
|
app-header-layout {
|
||||||
min-height: 100vh;
|
min-height: 100vh;
|
||||||
}
|
}
|
||||||
ha-icon-button.invisible {
|
ha-icon-button.invisible {
|
||||||
visibility: hidden;
|
visibility: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pickers {
|
.pickers {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
align-items: start;
|
align-items: start;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pickers ha-card {
|
.pickers ha-card {
|
||||||
width: 400px;
|
width: 400px;
|
||||||
display: block;
|
display: block;
|
||||||
margin: 16px 8px;
|
margin: 16px 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.pickers ha-card:last-child {
|
.pickers ha-card:last-child {
|
||||||
margin-bottom: 16px;
|
margin-bottom: 16px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.intro {
|
.intro {
|
||||||
margin: -1em 0;
|
margin: -1em 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
p a {
|
p a {
|
||||||
color: var(--primary-color);
|
color: var(--primary-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
a {
|
|
||||||
color: var(--primary-text-color);
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
a {
|
||||||
|
color: var(--primary-text-color);
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
<app-header-layout>
|
<app-header-layout>
|
||||||
@ -70,32 +69,42 @@ class HaGallery extends PolymerElement {
|
|||||||
<ha-icon-button
|
<ha-icon-button
|
||||||
icon="hass:arrow-left"
|
icon="hass:arrow-left"
|
||||||
on-click="_backTapped"
|
on-click="_backTapped"
|
||||||
class$='[[_computeHeaderButtonClass(_demo)]]'
|
class$="[[_computeHeaderButtonClass(_demo)]]"
|
||||||
></ha-icon-button>
|
></ha-icon-button>
|
||||||
<div main-title>[[_withDefault(_demo, "Home Assistant Gallery")]]</div>
|
<div main-title>
|
||||||
|
[[_withDefault(_demo, "Home Assistant Gallery")]]
|
||||||
|
</div>
|
||||||
</app-toolbar>
|
</app-toolbar>
|
||||||
</app-header>
|
</app-header>
|
||||||
|
|
||||||
<div class='content'>
|
<div class="content">
|
||||||
<div id='demo'></div>
|
<div id="demo"></div>
|
||||||
<template is='dom-if' if='[[!_demo]]'>
|
<template is="dom-if" if="[[!_demo]]">
|
||||||
<div class='pickers'>
|
<div class="pickers">
|
||||||
<ha-card header="Lovelace card demos">
|
<ha-card header="Lovelace Card Demos">
|
||||||
<div class='card-content intro'>
|
<div class="card-content intro">
|
||||||
<p>
|
<p>
|
||||||
Lovelace has many different cards. Each card allows the user to tell a different story about what is going on in their house. These cards are very customizable, as no household is the same.
|
Lovelace has many different cards. Each card allows the user
|
||||||
|
to tell a different story about what is going on in their
|
||||||
|
house. These cards are very customizable, as no household is
|
||||||
|
the same.
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p>
|
<p>
|
||||||
This gallery helps our developers and designers to see all the different states that each card can be in.
|
This gallery helps our developers and designers to see all
|
||||||
|
the different states that each card can be in.
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p>
|
<p>
|
||||||
Check <a href='https://www.home-assistant.io/lovelace'>the official website</a> for instructions on how to get started with Lovelace.</a>.
|
Check
|
||||||
|
<a href="https://www.home-assistant.io/lovelace"
|
||||||
|
>the official website</a
|
||||||
|
>
|
||||||
|
for instructions on how to get started with Lovelace.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<template is='dom-repeat' items='[[_lovelaceDemos]]'>
|
<template is="dom-repeat" items="[[_lovelaceDemos]]">
|
||||||
<a href='#[[item]]'>
|
<a href="#[[item]]">
|
||||||
<paper-item>
|
<paper-item>
|
||||||
<paper-item-body>{{ item }}</paper-item-body>
|
<paper-item-body>{{ item }}</paper-item-body>
|
||||||
<ha-icon icon="hass:chevron-right"></ha-icon>
|
<ha-icon icon="hass:chevron-right"></ha-icon>
|
||||||
@ -104,14 +113,14 @@ class HaGallery extends PolymerElement {
|
|||||||
</template>
|
</template>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
|
|
||||||
<ha-card header="More Info demos">
|
<ha-card header="More Info Demos">
|
||||||
<div class='card-content intro'>
|
<div class="card-content intro">
|
||||||
<p>
|
<p>
|
||||||
More info screens show up when an entity is clicked.
|
More info screens show up when an entity is clicked.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<template is='dom-repeat' items='[[_moreInfoDemos]]'>
|
<template is="dom-repeat" items="[[_moreInfoDemos]]">
|
||||||
<a href='#[[item]]'>
|
<a href="#[[item]]">
|
||||||
<paper-item>
|
<paper-item>
|
||||||
<paper-item-body>{{ item }}</paper-item-body>
|
<paper-item-body>{{ item }}</paper-item-body>
|
||||||
<ha-icon icon="hass:chevron-right"></ha-icon>
|
<ha-icon icon="hass:chevron-right"></ha-icon>
|
||||||
@ -120,14 +129,14 @@ class HaGallery extends PolymerElement {
|
|||||||
</template>
|
</template>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
|
|
||||||
<ha-card header="Util demos">
|
<ha-card header="Util Demos">
|
||||||
<div class='card-content intro'>
|
<div class="card-content intro">
|
||||||
<p>
|
<p>
|
||||||
Test pages for our utility functions.
|
Test pages for our utility functions.
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<template is='dom-repeat' items='[[_utilDemos]]'>
|
<template is="dom-repeat" items="[[_utilDemos]]">
|
||||||
<a href='#[[item]]'>
|
<a href="#[[item]]">
|
||||||
<paper-item>
|
<paper-item>
|
||||||
<paper-item-body>{{ item }}</paper-item-body>
|
<paper-item-body>{{ item }}</paper-item-body>
|
||||||
<ha-icon icon="hass:chevron-right"></ha-icon>
|
<ha-icon icon="hass:chevron-right"></ha-icon>
|
||||||
@ -139,7 +148,10 @@ class HaGallery extends PolymerElement {
|
|||||||
</template>
|
</template>
|
||||||
</div>
|
</div>
|
||||||
</app-header-layout>
|
</app-header-layout>
|
||||||
<notification-manager hass=[[_fakeHass]] id='notifications'></notification-manager>
|
<notification-manager
|
||||||
|
hass="[[_fakeHass]]"
|
||||||
|
id="notifications"
|
||||||
|
></notification-manager>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -27,6 +27,8 @@ declare global {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const MAX_FILE_SIZE = 1 * 1024 * 1024 * 1024; // 1GB
|
||||||
|
|
||||||
@customElement("hassio-upload-snapshot")
|
@customElement("hassio-upload-snapshot")
|
||||||
export class HassioUploadSnapshot extends LitElement {
|
export class HassioUploadSnapshot extends LitElement {
|
||||||
public hass!: HomeAssistant;
|
public hass!: HomeAssistant;
|
||||||
@ -51,6 +53,20 @@ export class HassioUploadSnapshot extends LitElement {
|
|||||||
private async _uploadFile(ev) {
|
private async _uploadFile(ev) {
|
||||||
const file = ev.detail.files[0];
|
const file = ev.detail.files[0];
|
||||||
|
|
||||||
|
if (file.size > MAX_FILE_SIZE) {
|
||||||
|
showAlertDialog(this, {
|
||||||
|
title: "Snapshot file is too big",
|
||||||
|
text: html`The maximum allowed filesize is 1GB.<br />
|
||||||
|
<a
|
||||||
|
href="https://www.home-assistant.io/hassio/haos_common_tasks/#restoring-a-snapshot-on-a-new-install"
|
||||||
|
target="_blank"
|
||||||
|
>Have a look here on how to restore it.</a
|
||||||
|
>`,
|
||||||
|
confirmText: "ok",
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (!["application/x-tar"].includes(file.type)) {
|
if (!["application/x-tar"].includes(file.type)) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Unsupported file format",
|
title: "Unsupported file format",
|
||||||
|
@ -12,7 +12,7 @@ import { atLeastVersion } from "../../../src/common/config/version";
|
|||||||
import { navigate } from "../../../src/common/navigate";
|
import { navigate } from "../../../src/common/navigate";
|
||||||
import { compare } from "../../../src/common/string/compare";
|
import { compare } from "../../../src/common/string/compare";
|
||||||
import "../../../src/components/ha-card";
|
import "../../../src/components/ha-card";
|
||||||
import { HassioAddonInfo } from "../../../src/data/hassio/addon";
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
import { haStyle } from "../../../src/resources/styles";
|
import { haStyle } from "../../../src/resources/styles";
|
||||||
import { HomeAssistant } from "../../../src/types";
|
import { HomeAssistant } from "../../../src/types";
|
||||||
import "../components/hassio-card-content";
|
import "../components/hassio-card-content";
|
||||||
@ -22,14 +22,14 @@ import { hassioStyle } from "../resources/hassio-style";
|
|||||||
class HassioAddons extends LitElement {
|
class HassioAddons extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property({ attribute: false }) public addons?: HassioAddonInfo[];
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
return html`
|
return html`
|
||||||
<div class="content">
|
<div class="content">
|
||||||
<h1>Add-ons</h1>
|
<h1>Add-ons</h1>
|
||||||
<div class="card-group">
|
<div class="card-group">
|
||||||
${!this.addons?.length
|
${!this.supervisor.supervisor.addons?.length
|
||||||
? html`
|
? html`
|
||||||
<ha-card>
|
<ha-card>
|
||||||
<div class="card-content">
|
<div class="card-content">
|
||||||
@ -41,7 +41,7 @@ class HassioAddons extends LitElement {
|
|||||||
</div>
|
</div>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
`
|
`
|
||||||
: this.addons
|
: this.supervisor.supervisor.addons
|
||||||
.sort((a, b) => compare(a.name, b.name))
|
.sort((a, b) => compare(a.name, b.name))
|
||||||
.map(
|
.map(
|
||||||
(addon) => html`
|
(addon) => html`
|
||||||
|
@ -7,11 +7,7 @@ import {
|
|||||||
property,
|
property,
|
||||||
TemplateResult,
|
TemplateResult,
|
||||||
} from "lit-element";
|
} from "lit-element";
|
||||||
import { HassioHassOSInfo } from "../../../src/data/hassio/host";
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
import {
|
|
||||||
HassioHomeAssistantInfo,
|
|
||||||
HassioSupervisorInfo,
|
|
||||||
} from "../../../src/data/hassio/supervisor";
|
|
||||||
import "../../../src/layouts/hass-tabs-subpage";
|
import "../../../src/layouts/hass-tabs-subpage";
|
||||||
import { haStyle } from "../../../src/resources/styles";
|
import { haStyle } from "../../../src/resources/styles";
|
||||||
import { HomeAssistant, Route } from "../../../src/types";
|
import { HomeAssistant, Route } from "../../../src/types";
|
||||||
@ -23,16 +19,12 @@ import "./hassio-update";
|
|||||||
class HassioDashboard extends LitElement {
|
class HassioDashboard extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ type: Boolean }) public narrow!: boolean;
|
@property({ type: Boolean }) public narrow!: boolean;
|
||||||
|
|
||||||
@property({ attribute: false }) public route!: Route;
|
@property({ attribute: false }) public route!: Route;
|
||||||
|
|
||||||
@property({ attribute: false }) public supervisorInfo!: HassioSupervisorInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hassInfo!: HassioHomeAssistantInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hassOsInfo!: HassioHassOSInfo;
|
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
return html`
|
return html`
|
||||||
<hass-tabs-subpage
|
<hass-tabs-subpage
|
||||||
@ -47,13 +39,11 @@ class HassioDashboard extends LitElement {
|
|||||||
<div class="content">
|
<div class="content">
|
||||||
<hassio-update
|
<hassio-update
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.hassInfo=${this.hassInfo}
|
.supervisor=${this.supervisor}
|
||||||
.supervisorInfo=${this.supervisorInfo}
|
|
||||||
.hassOsInfo=${this.hassOsInfo}
|
|
||||||
></hassio-update>
|
></hassio-update>
|
||||||
<hassio-addons
|
<hassio-addons
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.addons=${this.supervisorInfo.addons}
|
.supervisor=${this.supervisor}
|
||||||
></hassio-addons>
|
></hassio-addons>
|
||||||
</div>
|
</div>
|
||||||
</hass-tabs-subpage>
|
</hass-tabs-subpage>
|
||||||
|
@ -23,6 +23,7 @@ import {
|
|||||||
HassioHomeAssistantInfo,
|
HassioHomeAssistantInfo,
|
||||||
HassioSupervisorInfo,
|
HassioSupervisorInfo,
|
||||||
} from "../../../src/data/hassio/supervisor";
|
} from "../../../src/data/hassio/supervisor";
|
||||||
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
import {
|
import {
|
||||||
showAlertDialog,
|
showAlertDialog,
|
||||||
showConfirmationDialog,
|
showConfirmationDialog,
|
||||||
@ -35,31 +36,20 @@ import { hassioStyle } from "../resources/hassio-style";
|
|||||||
export class HassioUpdate extends LitElement {
|
export class HassioUpdate extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property({ attribute: false }) public hassInfo?: HassioHomeAssistantInfo;
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ attribute: false }) public hassOsInfo?: HassioHassOSInfo;
|
private _pendingUpdates = memoizeOne((supervisor: Supervisor): number => {
|
||||||
|
return Object.keys(supervisor).filter(
|
||||||
@property({ attribute: false }) public supervisorInfo?: HassioSupervisorInfo;
|
(value) => supervisor[value].update_available
|
||||||
|
).length;
|
||||||
private _pendingUpdates = memoizeOne(
|
});
|
||||||
(
|
|
||||||
core?: HassioHomeAssistantInfo,
|
|
||||||
supervisor?: HassioSupervisorInfo,
|
|
||||||
os?: HassioHassOSInfo
|
|
||||||
): number => {
|
|
||||||
return [core, supervisor, os].filter(
|
|
||||||
(value) => !!value && value?.update_available
|
|
||||||
).length;
|
|
||||||
}
|
|
||||||
);
|
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
const updatesAvailable = this._pendingUpdates(
|
if (!this.supervisor) {
|
||||||
this.hassInfo,
|
return html``;
|
||||||
this.supervisorInfo,
|
}
|
||||||
this.hassOsInfo
|
|
||||||
);
|
|
||||||
|
|
||||||
|
const updatesAvailable = this._pendingUpdates(this.supervisor);
|
||||||
if (!updatesAvailable) {
|
if (!updatesAvailable) {
|
||||||
return html``;
|
return html``;
|
||||||
}
|
}
|
||||||
@ -74,26 +64,26 @@ export class HassioUpdate extends LitElement {
|
|||||||
<div class="card-group">
|
<div class="card-group">
|
||||||
${this._renderUpdateCard(
|
${this._renderUpdateCard(
|
||||||
"Home Assistant Core",
|
"Home Assistant Core",
|
||||||
this.hassInfo!,
|
this.supervisor.core,
|
||||||
"hassio/homeassistant/update",
|
"hassio/homeassistant/update",
|
||||||
`https://${
|
`https://${
|
||||||
this.hassInfo?.version_latest.includes("b") ? "rc" : "www"
|
this.supervisor.core.version_latest.includes("b") ? "rc" : "www"
|
||||||
}.home-assistant.io/latest-release-notes/`
|
}.home-assistant.io/latest-release-notes/`
|
||||||
)}
|
)}
|
||||||
${this._renderUpdateCard(
|
${this._renderUpdateCard(
|
||||||
"Supervisor",
|
"Supervisor",
|
||||||
this.supervisorInfo!,
|
this.supervisor.supervisor,
|
||||||
"hassio/supervisor/update",
|
"hassio/supervisor/update",
|
||||||
`https://github.com//home-assistant/hassio/releases/tag/${
|
`https://github.com//home-assistant/hassio/releases/tag/${
|
||||||
this.supervisorInfo!.version_latest
|
this.supervisor.supervisor.version_latest
|
||||||
}`
|
}`
|
||||||
)}
|
)}
|
||||||
${this.hassOsInfo
|
${this.supervisor.host.features.includes("hassos")
|
||||||
? this._renderUpdateCard(
|
? this._renderUpdateCard(
|
||||||
"Operating System",
|
"Operating System",
|
||||||
this.hassOsInfo,
|
this.supervisor.os,
|
||||||
"hassio/os/update",
|
"hassio/os/update",
|
||||||
`https://github.com//home-assistant/hassos/releases/tag/${this.hassOsInfo.version_latest}`
|
`https://github.com//home-assistant/hassos/releases/tag/${this.supervisor.os.version_latest}`
|
||||||
)
|
)
|
||||||
: ""}
|
: ""}
|
||||||
</div>
|
</div>
|
||||||
|
@ -11,10 +11,7 @@ export const showHassioMarkdownDialog = (
|
|||||||
): void => {
|
): void => {
|
||||||
fireEvent(element, "show-dialog", {
|
fireEvent(element, "show-dialog", {
|
||||||
dialogTag: "dialog-hassio-markdown",
|
dialogTag: "dialog-hassio-markdown",
|
||||||
dialogImport: () =>
|
dialogImport: () => import("./dialog-hassio-markdown"),
|
||||||
import(
|
|
||||||
/* webpackChunkName: "dialog-hassio-markdown" */ "./dialog-hassio-markdown"
|
|
||||||
),
|
|
||||||
dialogParams,
|
dialogParams,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
import "@material/mwc-button/mwc-button";
|
import "@material/mwc-button/mwc-button";
|
||||||
import "@material/mwc-icon-button";
|
import "@material/mwc-icon-button";
|
||||||
|
import "@material/mwc-list/mwc-list";
|
||||||
|
import "@material/mwc-list/mwc-list-item";
|
||||||
import "@material/mwc-tab";
|
import "@material/mwc-tab";
|
||||||
import "@material/mwc-tab-bar";
|
import "@material/mwc-tab-bar";
|
||||||
import { mdiClose } from "@mdi/js";
|
import { mdiClose } from "@mdi/js";
|
||||||
@ -16,18 +18,22 @@ import {
|
|||||||
} from "lit-element";
|
} from "lit-element";
|
||||||
import { cache } from "lit-html/directives/cache";
|
import { cache } from "lit-html/directives/cache";
|
||||||
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
import { fireEvent } from "../../../../src/common/dom/fire_event";
|
||||||
|
import "../../../../src/components/ha-chips";
|
||||||
import "../../../../src/components/ha-circular-progress";
|
import "../../../../src/components/ha-circular-progress";
|
||||||
import "../../../../src/components/ha-dialog";
|
import "../../../../src/components/ha-dialog";
|
||||||
|
import "../../../../src/components/ha-expansion-panel";
|
||||||
import "../../../../src/components/ha-formfield";
|
import "../../../../src/components/ha-formfield";
|
||||||
import "../../../../src/components/ha-header-bar";
|
import "../../../../src/components/ha-header-bar";
|
||||||
import "../../../../src/components/ha-radio";
|
import "../../../../src/components/ha-radio";
|
||||||
import type { HaRadio } from "../../../../src/components/ha-radio";
|
|
||||||
import "../../../../src/components/ha-related-items";
|
import "../../../../src/components/ha-related-items";
|
||||||
import "../../../../src/components/ha-svg-icon";
|
import "../../../../src/components/ha-svg-icon";
|
||||||
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
import { extractApiErrorMessage } from "../../../../src/data/hassio/common";
|
||||||
import {
|
import {
|
||||||
|
AccessPoints,
|
||||||
|
accesspointScan,
|
||||||
NetworkInterface,
|
NetworkInterface,
|
||||||
updateNetworkInterface,
|
updateNetworkInterface,
|
||||||
|
WifiConfiguration,
|
||||||
} from "../../../../src/data/hassio/network";
|
} from "../../../../src/data/hassio/network";
|
||||||
import {
|
import {
|
||||||
showAlertDialog,
|
showAlertDialog,
|
||||||
@ -38,54 +44,51 @@ import { haStyleDialog } from "../../../../src/resources/styles";
|
|||||||
import type { HomeAssistant } from "../../../../src/types";
|
import type { HomeAssistant } from "../../../../src/types";
|
||||||
import { HassioNetworkDialogParams } from "./show-dialog-network";
|
import { HassioNetworkDialogParams } from "./show-dialog-network";
|
||||||
|
|
||||||
|
const IP_VERSIONS = ["ipv4", "ipv6"];
|
||||||
|
|
||||||
@customElement("dialog-hassio-network")
|
@customElement("dialog-hassio-network")
|
||||||
export class DialogHassioNetwork extends LitElement
|
export class DialogHassioNetwork extends LitElement
|
||||||
implements HassDialog<HassioNetworkDialogParams> {
|
implements HassDialog<HassioNetworkDialogParams> {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
@internalProperty() private _prosessing = false;
|
@internalProperty() private _accessPoints?: AccessPoints;
|
||||||
|
|
||||||
@internalProperty() private _params?: HassioNetworkDialogParams;
|
|
||||||
|
|
||||||
@internalProperty() private _network!: {
|
|
||||||
interface: string;
|
|
||||||
data: NetworkInterface;
|
|
||||||
}[];
|
|
||||||
|
|
||||||
@internalProperty() private _curTabIndex = 0;
|
@internalProperty() private _curTabIndex = 0;
|
||||||
|
|
||||||
@internalProperty() private _device?: {
|
|
||||||
interface: string;
|
|
||||||
data: NetworkInterface;
|
|
||||||
};
|
|
||||||
|
|
||||||
@internalProperty() private _dirty = false;
|
@internalProperty() private _dirty = false;
|
||||||
|
|
||||||
|
@internalProperty() private _interface?: NetworkInterface;
|
||||||
|
|
||||||
|
@internalProperty() private _interfaces!: NetworkInterface[];
|
||||||
|
|
||||||
|
@internalProperty() private _params?: HassioNetworkDialogParams;
|
||||||
|
|
||||||
|
@internalProperty() private _processing = false;
|
||||||
|
|
||||||
|
@internalProperty() private _scanning = false;
|
||||||
|
|
||||||
|
@internalProperty() private _wifiConfiguration?: WifiConfiguration;
|
||||||
|
|
||||||
public async showDialog(params: HassioNetworkDialogParams): Promise<void> {
|
public async showDialog(params: HassioNetworkDialogParams): Promise<void> {
|
||||||
this._params = params;
|
this._params = params;
|
||||||
this._dirty = false;
|
this._dirty = false;
|
||||||
this._curTabIndex = 0;
|
this._curTabIndex = 0;
|
||||||
this._network = Object.keys(params.network?.interfaces)
|
this._interfaces = params.network.interfaces.sort((a, b) => {
|
||||||
.map((device) => ({
|
return a.primary > b.primary ? -1 : 1;
|
||||||
interface: device,
|
});
|
||||||
data: params.network.interfaces[device],
|
this._interface = { ...this._interfaces[this._curTabIndex] };
|
||||||
}))
|
|
||||||
.sort((a, b) => {
|
|
||||||
return a.data.primary > b.data.primary ? -1 : 1;
|
|
||||||
});
|
|
||||||
this._device = this._network[this._curTabIndex];
|
|
||||||
this._device.data.nameservers = String(this._device.data.nameservers);
|
|
||||||
await this.updateComplete;
|
await this.updateComplete;
|
||||||
}
|
}
|
||||||
|
|
||||||
public closeDialog(): void {
|
public closeDialog(): void {
|
||||||
this._params = undefined;
|
this._params = undefined;
|
||||||
this._prosessing = false;
|
this._processing = false;
|
||||||
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
fireEvent(this, "dialog-closed", { dialog: this.localName });
|
||||||
}
|
}
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
if (!this._params || !this._network) {
|
if (!this._params || !this._interface) {
|
||||||
return html``;
|
return html``;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -107,11 +110,11 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
<ha-svg-icon .path=${mdiClose}></ha-svg-icon>
|
<ha-svg-icon .path=${mdiClose}></ha-svg-icon>
|
||||||
</mwc-icon-button>
|
</mwc-icon-button>
|
||||||
</ha-header-bar>
|
</ha-header-bar>
|
||||||
${this._network.length > 1
|
${this._interfaces.length > 1
|
||||||
? html` <mwc-tab-bar
|
? html` <mwc-tab-bar
|
||||||
.activeIndex=${this._curTabIndex}
|
.activeIndex=${this._curTabIndex}
|
||||||
@MDCTabBar:activated=${this._handleTabActivated}
|
@MDCTabBar:activated=${this._handleTabActivated}
|
||||||
>${this._network.map(
|
>${this._interfaces.map(
|
||||||
(device) =>
|
(device) =>
|
||||||
html`<mwc-tab
|
html`<mwc-tab
|
||||||
.id=${device.interface}
|
.id=${device.interface}
|
||||||
@ -129,81 +132,301 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
|
|
||||||
private _renderTab() {
|
private _renderTab() {
|
||||||
return html` <div class="form container">
|
return html` <div class="form container">
|
||||||
<ha-formfield label="DHCP">
|
${IP_VERSIONS.map((version) =>
|
||||||
<ha-radio
|
this._interface![version] ? this._renderIPConfiguration(version) : ""
|
||||||
@change=${this._handleRadioValueChanged}
|
)}
|
||||||
value="dhcp"
|
${this._interface?.type === "wireless"
|
||||||
name="method"
|
? html`
|
||||||
?checked=${this._device!.data.method === "dhcp"}
|
<ha-expansion-panel outlined>
|
||||||
>
|
<span slot="title">Wi-Fi</span>
|
||||||
</ha-radio>
|
${this._interface?.wifi?.ssid
|
||||||
</ha-formfield>
|
? html`<p>Connected to: ${this._interface?.wifi?.ssid}</p>`
|
||||||
<ha-formfield label="Static">
|
: ""}
|
||||||
<ha-radio
|
<mwc-button
|
||||||
@change=${this._handleRadioValueChanged}
|
class="scan"
|
||||||
value="static"
|
@click=${this._scanForAP}
|
||||||
name="method"
|
.disabled=${this._scanning}
|
||||||
?checked=${this._device!.data.method === "static"}
|
>
|
||||||
>
|
${this._scanning
|
||||||
</ha-radio>
|
? html`<ha-circular-progress active size="small">
|
||||||
</ha-formfield>
|
</ha-circular-progress>`
|
||||||
${this._device!.data.method !== "dhcp"
|
: "Scan for accesspoints"}
|
||||||
? html` <paper-input
|
</mwc-button>
|
||||||
class="flex-auto"
|
${this._accessPoints &&
|
||||||
id="ip_address"
|
this._accessPoints.accesspoints &&
|
||||||
label="IP address/Netmask"
|
this._accessPoints.accesspoints.length !== 0
|
||||||
.value="${this._device!.data.ip_address}"
|
? html`
|
||||||
@value-changed=${this._handleInputValueChanged}
|
<mwc-list>
|
||||||
></paper-input>
|
${this._accessPoints.accesspoints
|
||||||
<paper-input
|
.filter((ap) => ap.ssid)
|
||||||
class="flex-auto"
|
.map(
|
||||||
id="gateway"
|
(ap) =>
|
||||||
label="Gateway address"
|
html`
|
||||||
.value="${this._device!.data.gateway}"
|
<mwc-list-item
|
||||||
@value-changed=${this._handleInputValueChanged}
|
twoline
|
||||||
></paper-input>
|
@click=${this._selectAP}
|
||||||
<paper-input
|
.activated=${ap.ssid ===
|
||||||
class="flex-auto"
|
this._wifiConfiguration?.ssid}
|
||||||
id="nameservers"
|
.ap=${ap}
|
||||||
label="DNS servers"
|
>
|
||||||
.value="${this._device!.data.nameservers as string}"
|
<span>${ap.ssid}</span>
|
||||||
@value-changed=${this._handleInputValueChanged}
|
<span slot="secondary">
|
||||||
></paper-input>
|
${ap.mac} - Strength: ${ap.signal}
|
||||||
NB!: If you are changing IP or gateway addresses, you might lose
|
</span>
|
||||||
the connection.`
|
</mwc-list-item>
|
||||||
|
`
|
||||||
|
)}
|
||||||
|
</mwc-list>
|
||||||
|
`
|
||||||
|
: ""}
|
||||||
|
${this._wifiConfiguration
|
||||||
|
? html`
|
||||||
|
<div class="radio-row">
|
||||||
|
<ha-formfield label="open">
|
||||||
|
<ha-radio
|
||||||
|
@change=${this._handleRadioValueChangedAp}
|
||||||
|
.ap=${this._wifiConfiguration}
|
||||||
|
value="open"
|
||||||
|
name="auth"
|
||||||
|
.checked=${this._wifiConfiguration.auth ===
|
||||||
|
undefined ||
|
||||||
|
this._wifiConfiguration.auth === "open"}
|
||||||
|
>
|
||||||
|
</ha-radio>
|
||||||
|
</ha-formfield>
|
||||||
|
<ha-formfield label="wep">
|
||||||
|
<ha-radio
|
||||||
|
@change=${this._handleRadioValueChangedAp}
|
||||||
|
.ap=${this._wifiConfiguration}
|
||||||
|
value="wep"
|
||||||
|
name="auth"
|
||||||
|
.checked=${this._wifiConfiguration.auth === "wep"}
|
||||||
|
>
|
||||||
|
</ha-radio>
|
||||||
|
</ha-formfield>
|
||||||
|
<ha-formfield label="wpa-psk">
|
||||||
|
<ha-radio
|
||||||
|
@change=${this._handleRadioValueChangedAp}
|
||||||
|
.ap=${this._wifiConfiguration}
|
||||||
|
value="wpa-psk"
|
||||||
|
name="auth"
|
||||||
|
.checked=${this._wifiConfiguration.auth ===
|
||||||
|
"wpa-psk"}
|
||||||
|
>
|
||||||
|
</ha-radio>
|
||||||
|
</ha-formfield>
|
||||||
|
</div>
|
||||||
|
${this._wifiConfiguration.auth === "wpa-psk" ||
|
||||||
|
this._wifiConfiguration.auth === "wep"
|
||||||
|
? html`
|
||||||
|
<paper-input
|
||||||
|
class="flex-auto"
|
||||||
|
type="password"
|
||||||
|
id="psk"
|
||||||
|
label="Password"
|
||||||
|
version="wifi"
|
||||||
|
@value-changed=${this
|
||||||
|
._handleInputValueChangedWifi}
|
||||||
|
>
|
||||||
|
</paper-input>
|
||||||
|
`
|
||||||
|
: ""}
|
||||||
|
`
|
||||||
|
: ""}
|
||||||
|
</ha-expansion-panel>
|
||||||
|
`
|
||||||
|
: ""}
|
||||||
|
${this._dirty
|
||||||
|
? html`<div class="warning">
|
||||||
|
If you are changing the Wi-Fi, IP or gateway addresses, you might
|
||||||
|
lose the connection!
|
||||||
|
</div>`
|
||||||
: ""}
|
: ""}
|
||||||
</div>
|
</div>
|
||||||
<div class="buttons">
|
<div class="buttons">
|
||||||
<mwc-button label="close" @click=${this.closeDialog}> </mwc-button>
|
<mwc-button label="close" @click=${this.closeDialog}> </mwc-button>
|
||||||
<mwc-button @click=${this._updateNetwork} ?disabled=${!this._dirty}>
|
<mwc-button @click=${this._updateNetwork} .disabled=${!this._dirty}>
|
||||||
${this._prosessing
|
${this._processing
|
||||||
? html`<ha-circular-progress active></ha-circular-progress>`
|
? html`<ha-circular-progress active size="small">
|
||||||
: "Update"}
|
</ha-circular-progress>`
|
||||||
|
: "Save"}
|
||||||
</mwc-button>
|
</mwc-button>
|
||||||
</div>`;
|
</div>`;
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _updateNetwork() {
|
private _selectAP(event) {
|
||||||
this._prosessing = true;
|
this._wifiConfiguration = event.currentTarget.ap;
|
||||||
let options: Partial<NetworkInterface> = {
|
this._dirty = true;
|
||||||
method: this._device!.data.method,
|
}
|
||||||
};
|
|
||||||
if (options.method !== "dhcp") {
|
private async _scanForAP() {
|
||||||
options = {
|
if (!this._interface) {
|
||||||
...options,
|
return;
|
||||||
address: this._device!.data.ip_address,
|
|
||||||
gateway: this._device!.data.gateway,
|
|
||||||
dns: String(this._device!.data.nameservers).split(","),
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
this._scanning = true;
|
||||||
try {
|
try {
|
||||||
await updateNetworkInterface(this.hass, this._device!.interface, options);
|
this._accessPoints = await accesspointScan(
|
||||||
|
this.hass,
|
||||||
|
this._interface.interface
|
||||||
|
);
|
||||||
|
} catch (err) {
|
||||||
|
showAlertDialog(this, {
|
||||||
|
title: "Failed to scan for accesspoints",
|
||||||
|
text: extractApiErrorMessage(err),
|
||||||
|
});
|
||||||
|
} finally {
|
||||||
|
this._scanning = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private _renderIPConfiguration(version: string) {
|
||||||
|
return html`
|
||||||
|
<ha-expansion-panel outlined>
|
||||||
|
<span slot="title">IPv${version.charAt(version.length - 1)}</span>
|
||||||
|
<div class="radio-row">
|
||||||
|
<ha-formfield label="DHCP">
|
||||||
|
<ha-radio
|
||||||
|
@change=${this._handleRadioValueChanged}
|
||||||
|
.version=${version}
|
||||||
|
value="auto"
|
||||||
|
name="${version}method"
|
||||||
|
.checked=${this._interface![version]?.method === "auto"}
|
||||||
|
>
|
||||||
|
</ha-radio>
|
||||||
|
</ha-formfield>
|
||||||
|
<ha-formfield label="Static">
|
||||||
|
<ha-radio
|
||||||
|
@change=${this._handleRadioValueChanged}
|
||||||
|
.version=${version}
|
||||||
|
value="static"
|
||||||
|
name="${version}method"
|
||||||
|
.checked=${this._interface![version]?.method === "static"}
|
||||||
|
>
|
||||||
|
</ha-radio>
|
||||||
|
</ha-formfield>
|
||||||
|
<ha-formfield label="Disabled" class="warning">
|
||||||
|
<ha-radio
|
||||||
|
@change=${this._handleRadioValueChanged}
|
||||||
|
.version=${version}
|
||||||
|
value="disabled"
|
||||||
|
name="${version}method"
|
||||||
|
.checked=${this._interface![version]?.method === "disabled"}
|
||||||
|
>
|
||||||
|
</ha-radio>
|
||||||
|
</ha-formfield>
|
||||||
|
</div>
|
||||||
|
${this._interface![version].method === "static"
|
||||||
|
? html`
|
||||||
|
<paper-input
|
||||||
|
class="flex-auto"
|
||||||
|
id="address"
|
||||||
|
label="IP address/Netmask"
|
||||||
|
.version=${version}
|
||||||
|
.value=${this._toString(this._interface![version].address)}
|
||||||
|
@value-changed=${this._handleInputValueChanged}
|
||||||
|
>
|
||||||
|
</paper-input>
|
||||||
|
<paper-input
|
||||||
|
class="flex-auto"
|
||||||
|
id="gateway"
|
||||||
|
label="Gateway address"
|
||||||
|
.version=${version}
|
||||||
|
.value=${this._interface![version].gateway}
|
||||||
|
@value-changed=${this._handleInputValueChanged}
|
||||||
|
>
|
||||||
|
</paper-input>
|
||||||
|
<paper-input
|
||||||
|
class="flex-auto"
|
||||||
|
id="nameservers"
|
||||||
|
label="DNS servers"
|
||||||
|
.version=${version}
|
||||||
|
.value=${this._toString(this._interface![version].nameservers)}
|
||||||
|
@value-changed=${this._handleInputValueChanged}
|
||||||
|
>
|
||||||
|
</paper-input>
|
||||||
|
`
|
||||||
|
: ""}
|
||||||
|
</ha-expansion-panel>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
_toArray(data: string | string[]): string[] {
|
||||||
|
if (Array.isArray(data)) {
|
||||||
|
if (data && typeof data[0] === "string") {
|
||||||
|
data = data[0];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!data) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
if (typeof data === "string") {
|
||||||
|
return data.replace(/ /g, "").split(",");
|
||||||
|
}
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
_toString(data: string | string[]): string {
|
||||||
|
if (!data) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
if (Array.isArray(data)) {
|
||||||
|
return data.join(", ");
|
||||||
|
}
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async _updateNetwork() {
|
||||||
|
this._processing = true;
|
||||||
|
let interfaceOptions: Partial<NetworkInterface> = {};
|
||||||
|
|
||||||
|
IP_VERSIONS.forEach((version) => {
|
||||||
|
interfaceOptions[version] = {
|
||||||
|
method: this._interface![version]?.method || "auto",
|
||||||
|
};
|
||||||
|
if (this._interface![version]?.method === "static") {
|
||||||
|
interfaceOptions[version] = {
|
||||||
|
...interfaceOptions[version],
|
||||||
|
address: this._toArray(this._interface![version]?.address),
|
||||||
|
gateway: this._interface![version]?.gateway,
|
||||||
|
nameservers: this._toArray(this._interface![version]?.nameservers),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (this._wifiConfiguration) {
|
||||||
|
interfaceOptions = {
|
||||||
|
...interfaceOptions,
|
||||||
|
wifi: {
|
||||||
|
ssid: this._wifiConfiguration.ssid,
|
||||||
|
mode: this._wifiConfiguration.mode,
|
||||||
|
auth: this._wifiConfiguration.auth || "open",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
if (interfaceOptions.wifi!.auth !== "open") {
|
||||||
|
interfaceOptions.wifi = {
|
||||||
|
...interfaceOptions.wifi,
|
||||||
|
psk: this._wifiConfiguration.psk,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
interfaceOptions.enabled =
|
||||||
|
this._wifiConfiguration !== undefined ||
|
||||||
|
interfaceOptions.ipv4?.method !== "disabled" ||
|
||||||
|
interfaceOptions.ipv6?.method !== "disabled";
|
||||||
|
|
||||||
|
try {
|
||||||
|
await updateNetworkInterface(
|
||||||
|
this.hass,
|
||||||
|
this._interface!.interface,
|
||||||
|
interfaceOptions
|
||||||
|
);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to change network settings",
|
title: "Failed to change network settings",
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
this._prosessing = false;
|
this._processing = false;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
this._params?.loadData();
|
this._params?.loadData();
|
||||||
@ -219,40 +442,73 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
dismissText: "no",
|
dismissText: "no",
|
||||||
});
|
});
|
||||||
if (!confirm) {
|
if (!confirm) {
|
||||||
this.requestUpdate("_device");
|
this.requestUpdate("_interface");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
this._curTabIndex = ev.detail.index;
|
this._curTabIndex = ev.detail.index;
|
||||||
this._device = this._network[ev.detail.index];
|
this._interface = { ...this._interfaces[ev.detail.index] };
|
||||||
this._device.data.nameservers = String(this._device.data.nameservers);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private _handleRadioValueChanged(ev: CustomEvent): void {
|
private _handleRadioValueChanged(ev: CustomEvent): void {
|
||||||
const value = (ev.target as HaRadio).value as "dhcp" | "static";
|
const value = (ev.target as any).value as "disabled" | "auto" | "static";
|
||||||
|
const version = (ev.target as any).version as "ipv4" | "ipv6";
|
||||||
|
|
||||||
if (!value || !this._device || this._device!.data.method === value) {
|
if (
|
||||||
|
!value ||
|
||||||
|
!this._interface ||
|
||||||
|
this._interface[version]!.method === value
|
||||||
|
) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this._dirty = true;
|
this._dirty = true;
|
||||||
|
|
||||||
this._device!.data.method = value;
|
this._interface[version]!.method = value;
|
||||||
this.requestUpdate("_device");
|
this.requestUpdate("_interface");
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleRadioValueChangedAp(ev: CustomEvent): void {
|
||||||
|
const value = ((ev.target as any).value as string) as
|
||||||
|
| "open"
|
||||||
|
| "wep"
|
||||||
|
| "wpa-psk";
|
||||||
|
this._wifiConfiguration!.auth = value;
|
||||||
|
this._dirty = true;
|
||||||
|
this.requestUpdate("_wifiConfiguration");
|
||||||
}
|
}
|
||||||
|
|
||||||
private _handleInputValueChanged(ev: CustomEvent): void {
|
private _handleInputValueChanged(ev: CustomEvent): void {
|
||||||
const value: string | null | undefined = (ev.target as PaperInputElement)
|
const value: string | null | undefined = (ev.target as PaperInputElement)
|
||||||
.value;
|
.value;
|
||||||
|
const version = (ev.target as any).version as "ipv4" | "ipv6";
|
||||||
const id = (ev.target as PaperInputElement).id;
|
const id = (ev.target as PaperInputElement).id;
|
||||||
|
|
||||||
if (!value || !this._device || this._device.data[id] === value) {
|
if (
|
||||||
|
!value ||
|
||||||
|
!this._interface ||
|
||||||
|
this._toString(this._interface[version]![id]) === this._toString(value)
|
||||||
|
) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
this._dirty = true;
|
this._dirty = true;
|
||||||
|
this._interface[version]![id] = value;
|
||||||
|
}
|
||||||
|
|
||||||
this._device.data[id] = value;
|
private _handleInputValueChangedWifi(ev: CustomEvent): void {
|
||||||
|
const value: string | null | undefined = (ev.target as PaperInputElement)
|
||||||
|
.value;
|
||||||
|
const id = (ev.target as PaperInputElement).id;
|
||||||
|
|
||||||
|
if (
|
||||||
|
!value ||
|
||||||
|
!this._wifiConfiguration ||
|
||||||
|
this._wifiConfiguration![id] === value
|
||||||
|
) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
this._dirty = true;
|
||||||
|
this._wifiConfiguration![id] = value;
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResult[] {
|
||||||
@ -299,12 +555,16 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
--mdc-theme-primary: var(--error-color);
|
--mdc-theme-primary: var(--error-color);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mwc-button.scan {
|
||||||
|
margin-left: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
:host([rtl]) app-toolbar {
|
:host([rtl]) app-toolbar {
|
||||||
direction: rtl;
|
direction: rtl;
|
||||||
text-align: right;
|
text-align: right;
|
||||||
}
|
}
|
||||||
.container {
|
.container {
|
||||||
padding: 20px 24px;
|
padding: 0 8px 4px;
|
||||||
}
|
}
|
||||||
.form {
|
.form {
|
||||||
margin-bottom: 53px;
|
margin-bottom: 53px;
|
||||||
@ -322,6 +582,23 @@ export class DialogHassioNetwork extends LitElement
|
|||||||
padding-bottom: max(env(safe-area-inset-bottom), 8px);
|
padding-bottom: max(env(safe-area-inset-bottom), 8px);
|
||||||
background-color: var(--mdc-theme-surface, #fff);
|
background-color: var(--mdc-theme-surface, #fff);
|
||||||
}
|
}
|
||||||
|
.warning {
|
||||||
|
color: var(--error-color);
|
||||||
|
--primary-color: var(--error-color);
|
||||||
|
}
|
||||||
|
div.warning {
|
||||||
|
margin: 12px 4px -12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
ha-expansion-panel {
|
||||||
|
margin: 4px 0;
|
||||||
|
}
|
||||||
|
paper-input {
|
||||||
|
padding: 0 14px;
|
||||||
|
}
|
||||||
|
mwc-list-item {
|
||||||
|
--mdc-list-side-padding: 10px;
|
||||||
|
}
|
||||||
`,
|
`,
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
@ -13,10 +13,7 @@ export const showNetworkDialog = (
|
|||||||
): void => {
|
): void => {
|
||||||
fireEvent(element, "show-dialog", {
|
fireEvent(element, "show-dialog", {
|
||||||
dialogTag: "dialog-hassio-network",
|
dialogTag: "dialog-hassio-network",
|
||||||
dialogImport: () =>
|
dialogImport: () => import("./dialog-hassio-network"),
|
||||||
import(
|
|
||||||
/* webpackChunkName: "dialog-hassio-network" */ "./dialog-hassio-network"
|
|
||||||
),
|
|
||||||
dialogParams,
|
dialogParams,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
@ -4,10 +4,7 @@ import "./dialog-hassio-registries";
|
|||||||
export const showRegistriesDialog = (element: HTMLElement): void => {
|
export const showRegistriesDialog = (element: HTMLElement): void => {
|
||||||
fireEvent(element, "show-dialog", {
|
fireEvent(element, "show-dialog", {
|
||||||
dialogTag: "dialog-hassio-registries",
|
dialogTag: "dialog-hassio-registries",
|
||||||
dialogImport: () =>
|
dialogImport: () => import("./dialog-hassio-registries"),
|
||||||
import(
|
|
||||||
/* webpackChunkName: "dialog-hassio-registries" */ "./dialog-hassio-registries"
|
|
||||||
),
|
|
||||||
dialogParams: {},
|
dialogParams: {},
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
@ -13,10 +13,7 @@ export const showRepositoriesDialog = (
|
|||||||
): void => {
|
): void => {
|
||||||
fireEvent(element, "show-dialog", {
|
fireEvent(element, "show-dialog", {
|
||||||
dialogTag: "dialog-hassio-repositories",
|
dialogTag: "dialog-hassio-repositories",
|
||||||
dialogImport: () =>
|
dialogImport: () => import("./dialog-hassio-repositories"),
|
||||||
import(
|
|
||||||
/* webpackChunkName: "dialog-hassio-repositories" */ "./dialog-hassio-repositories"
|
|
||||||
),
|
|
||||||
dialogParams,
|
dialogParams,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
@ -109,7 +109,7 @@ class HassioSnapshotDialog extends LitElement {
|
|||||||
return html``;
|
return html``;
|
||||||
}
|
}
|
||||||
return html`
|
return html`
|
||||||
<ha-dialog open stacked @closing=${this._closeDialog} .heading=${true}>
|
<ha-dialog open @closing=${this._closeDialog} .heading=${true}>
|
||||||
<div slot="heading">
|
<div slot="heading">
|
||||||
<ha-header-bar>
|
<ha-header-bar>
|
||||||
<span slot="title">
|
<span slot="title">
|
||||||
@ -191,47 +191,37 @@ class HassioSnapshotDialog extends LitElement {
|
|||||||
: ""}
|
: ""}
|
||||||
${this._error ? html` <p class="error">Error: ${this._error}</p> ` : ""}
|
${this._error ? html` <p class="error">Error: ${this._error}</p> ` : ""}
|
||||||
|
|
||||||
<div>Actions:</div>
|
<div class="button-row" slot="primaryAction">
|
||||||
${!this._onboarding
|
<mwc-button @click=${this._partialRestoreClicked}>
|
||||||
? html`<mwc-button
|
<ha-svg-icon .path=${mdiHistory} class="icon"></ha-svg-icon>
|
||||||
@click=${this._downloadClicked}
|
Restore Selected
|
||||||
slot="primaryAction"
|
</mwc-button>
|
||||||
>
|
${!this._onboarding
|
||||||
<ha-svg-icon .path=${mdiDownload} class="icon"></ha-svg-icon>
|
? html`
|
||||||
Download Snapshot
|
<mwc-button @click=${this._deleteClicked}>
|
||||||
</mwc-button>`
|
<ha-svg-icon .path=${mdiDelete} class="icon warning">
|
||||||
: ""}
|
</ha-svg-icon>
|
||||||
|
<span class="warning">Delete Snapshot</span>
|
||||||
<mwc-button
|
</mwc-button>
|
||||||
@click=${this._partialRestoreClicked}
|
`
|
||||||
slot="secondaryAction"
|
: ""}
|
||||||
>
|
</div>
|
||||||
<ha-svg-icon .path=${mdiHistory} class="icon"></ha-svg-icon>
|
<div class="button-row" slot="secondaryAction">
|
||||||
Restore Selected
|
${this._snapshot.type === "full"
|
||||||
</mwc-button>
|
? html`
|
||||||
${this._snapshot.type === "full"
|
<mwc-button @click=${this._fullRestoreClicked}>
|
||||||
? html`
|
<ha-svg-icon .path=${mdiHistory} class="icon"></ha-svg-icon>
|
||||||
<mwc-button
|
Restore Everything
|
||||||
@click=${this._fullRestoreClicked}
|
</mwc-button>
|
||||||
slot="secondaryAction"
|
`
|
||||||
>
|
: ""}
|
||||||
<ha-svg-icon .path=${mdiHistory} class="icon"></ha-svg-icon>
|
${!this._onboarding
|
||||||
Wipe & restore
|
? html`<mwc-button @click=${this._downloadClicked}>
|
||||||
</mwc-button>
|
<ha-svg-icon .path=${mdiDownload} class="icon"></ha-svg-icon>
|
||||||
`
|
Download Snapshot
|
||||||
: ""}
|
</mwc-button>`
|
||||||
${!this._onboarding
|
: ""}
|
||||||
? html`<mwc-button
|
</div>
|
||||||
@click=${this._deleteClicked}
|
|
||||||
slot="secondaryAction"
|
|
||||||
>
|
|
||||||
<ha-svg-icon
|
|
||||||
.path=${mdiDelete}
|
|
||||||
class="icon warning"
|
|
||||||
></ha-svg-icon>
|
|
||||||
<span class="warning">Delete Snapshot</span>
|
|
||||||
</mwc-button>`
|
|
||||||
: ""}
|
|
||||||
</ha-dialog>
|
</ha-dialog>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
@ -245,6 +235,14 @@ class HassioSnapshotDialog extends LitElement {
|
|||||||
display: block;
|
display: block;
|
||||||
margin: 4px;
|
margin: 4px;
|
||||||
}
|
}
|
||||||
|
mwc-button ha-svg-icon {
|
||||||
|
margin-right: 4px;
|
||||||
|
}
|
||||||
|
.button-row {
|
||||||
|
display: grid;
|
||||||
|
gap: 8px;
|
||||||
|
margin-right: 8px;
|
||||||
|
}
|
||||||
.details {
|
.details {
|
||||||
color: var(--secondary-text-color);
|
color: var(--secondary-text-color);
|
||||||
}
|
}
|
||||||
@ -252,10 +250,6 @@ class HassioSnapshotDialog extends LitElement {
|
|||||||
.error {
|
.error {
|
||||||
color: var(--error-color);
|
color: var(--error-color);
|
||||||
}
|
}
|
||||||
.buttons {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
}
|
|
||||||
.buttons li {
|
.buttons li {
|
||||||
list-style-type: none;
|
list-style-type: none;
|
||||||
}
|
}
|
||||||
|
@ -12,10 +12,7 @@ export const showHassioSnapshotDialog = (
|
|||||||
): void => {
|
): void => {
|
||||||
fireEvent(element, "show-dialog", {
|
fireEvent(element, "show-dialog", {
|
||||||
dialogTag: "dialog-hassio-snapshot",
|
dialogTag: "dialog-hassio-snapshot",
|
||||||
dialogImport: () =>
|
dialogImport: () => import("./dialog-hassio-snapshot"),
|
||||||
import(
|
|
||||||
/* webpackChunkName: "dialog-hassio-snapshot" */ "./dialog-hassio-snapshot"
|
|
||||||
),
|
|
||||||
dialogParams,
|
dialogParams,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
@ -13,10 +13,7 @@ export const showSnapshotUploadDialog = (
|
|||||||
): void => {
|
): void => {
|
||||||
fireEvent(element, "show-dialog", {
|
fireEvent(element, "show-dialog", {
|
||||||
dialogTag: "dialog-hassio-snapshot-upload",
|
dialogTag: "dialog-hassio-snapshot-upload",
|
||||||
dialogImport: () =>
|
dialogImport: () => import("./dialog-hassio-snapshot-upload"),
|
||||||
import(
|
|
||||||
/* webpackChunkName: "dialog-hassio-snapshot-upload" */ "./dialog-hassio-snapshot-upload"
|
|
||||||
),
|
|
||||||
dialogParams,
|
dialogParams,
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
@ -1,29 +1,22 @@
|
|||||||
import {
|
import { html, PropertyValues, customElement, property } from "lit-element";
|
||||||
html,
|
|
||||||
PropertyValues,
|
|
||||||
customElement,
|
|
||||||
LitElement,
|
|
||||||
property,
|
|
||||||
} from "lit-element";
|
|
||||||
import "./hassio-router";
|
import "./hassio-router";
|
||||||
import { urlSyncMixin } from "../../src/state/url-sync-mixin";
|
|
||||||
import { ProvideHassLitMixin } from "../../src/mixins/provide-hass-lit-mixin";
|
|
||||||
import { HomeAssistant, Route } from "../../src/types";
|
import { HomeAssistant, Route } from "../../src/types";
|
||||||
import { HassioPanelInfo } from "../../src/data/hassio/supervisor";
|
import { HassioPanelInfo } from "../../src/data/hassio/supervisor";
|
||||||
import { applyThemesOnElement } from "../../src/common/dom/apply_themes_on_element";
|
import { applyThemesOnElement } from "../../src/common/dom/apply_themes_on_element";
|
||||||
import { fireEvent } from "../../src/common/dom/fire_event";
|
import { fireEvent } from "../../src/common/dom/fire_event";
|
||||||
import { makeDialogManager } from "../../src/dialogs/make-dialog-manager";
|
import { makeDialogManager } from "../../src/dialogs/make-dialog-manager";
|
||||||
import { atLeastVersion } from "../../src/common/config/version";
|
import { atLeastVersion } from "../../src/common/config/version";
|
||||||
|
import { SupervisorBaseElement } from "./supervisor-base-element";
|
||||||
|
|
||||||
@customElement("hassio-main")
|
@customElement("hassio-main")
|
||||||
export class HassioMain extends urlSyncMixin(ProvideHassLitMixin(LitElement)) {
|
export class HassioMain extends SupervisorBaseElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property() public panel!: HassioPanelInfo;
|
@property({ attribute: false }) public panel!: HassioPanelInfo;
|
||||||
|
|
||||||
@property() public narrow!: boolean;
|
@property({ type: Boolean }) public narrow!: boolean;
|
||||||
|
|
||||||
@property() public route?: Route;
|
@property({ attribute: false }) public route?: Route;
|
||||||
|
|
||||||
protected firstUpdated(changedProps: PropertyValues) {
|
protected firstUpdated(changedProps: PropertyValues) {
|
||||||
super.firstUpdated(changedProps);
|
super.firstUpdated(changedProps);
|
||||||
@ -77,9 +70,13 @@ export class HassioMain extends urlSyncMixin(ProvideHassLitMixin(LitElement)) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
protected render() {
|
protected render() {
|
||||||
|
if (!this.supervisor || !this.hass) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
return html`
|
return html`
|
||||||
<hassio-router
|
<hassio-router
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
|
.supervisor=${this.supervisor}
|
||||||
.route=${this.route}
|
.route=${this.route}
|
||||||
.panel=${this.panel}
|
.panel=${this.panel}
|
||||||
.narrow=${this.narrow}
|
.narrow=${this.narrow}
|
||||||
|
@ -1,10 +1,5 @@
|
|||||||
import { customElement, property } from "lit-element";
|
import { customElement, property } from "lit-element";
|
||||||
import { HassioHassOSInfo, HassioHostInfo } from "../../src/data/hassio/host";
|
import { Supervisor } from "../../src/data/supervisor/supervisor";
|
||||||
import {
|
|
||||||
HassioHomeAssistantInfo,
|
|
||||||
HassioSupervisorInfo,
|
|
||||||
HassioInfo,
|
|
||||||
} from "../../src/data/hassio/supervisor";
|
|
||||||
import {
|
import {
|
||||||
HassRouterPage,
|
HassRouterPage,
|
||||||
RouterOptions,
|
RouterOptions,
|
||||||
@ -21,20 +16,12 @@ import "./system/hassio-system";
|
|||||||
class HassioPanelRouter extends HassRouterPage {
|
class HassioPanelRouter extends HassRouterPage {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ attribute: false }) public route!: Route;
|
@property({ attribute: false }) public route!: Route;
|
||||||
|
|
||||||
@property({ type: Boolean }) public narrow!: boolean;
|
@property({ type: Boolean }) public narrow!: boolean;
|
||||||
|
|
||||||
@property({ attribute: false }) public supervisorInfo?: HassioSupervisorInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hassioInfo!: HassioInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hostInfo?: HassioHostInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hassInfo?: HassioHomeAssistantInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hassOsInfo!: HassioHassOSInfo;
|
|
||||||
|
|
||||||
protected routerOptions: RouterOptions = {
|
protected routerOptions: RouterOptions = {
|
||||||
routes: {
|
routes: {
|
||||||
dashboard: {
|
dashboard: {
|
||||||
@ -54,13 +41,9 @@ class HassioPanelRouter extends HassRouterPage {
|
|||||||
|
|
||||||
protected updatePageEl(el) {
|
protected updatePageEl(el) {
|
||||||
el.hass = this.hass;
|
el.hass = this.hass;
|
||||||
|
el.supervisor = this.supervisor;
|
||||||
el.route = this.route;
|
el.route = this.route;
|
||||||
el.narrow = this.narrow;
|
el.narrow = this.narrow;
|
||||||
el.supervisorInfo = this.supervisorInfo;
|
|
||||||
el.hassioInfo = this.hassioInfo;
|
|
||||||
el.hostInfo = this.hostInfo;
|
|
||||||
el.hassInfo = this.hassInfo;
|
|
||||||
el.hassOsInfo = this.hassOsInfo;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,18 +1,13 @@
|
|||||||
import {
|
import {
|
||||||
|
css,
|
||||||
|
CSSResult,
|
||||||
customElement,
|
customElement,
|
||||||
html,
|
html,
|
||||||
LitElement,
|
LitElement,
|
||||||
property,
|
property,
|
||||||
TemplateResult,
|
TemplateResult,
|
||||||
css,
|
|
||||||
CSSResult,
|
|
||||||
} from "lit-element";
|
} from "lit-element";
|
||||||
import { HassioHassOSInfo, HassioHostInfo } from "../../src/data/hassio/host";
|
import { Supervisor } from "../../src/data/supervisor/supervisor";
|
||||||
import {
|
|
||||||
HassioHomeAssistantInfo,
|
|
||||||
HassioSupervisorInfo,
|
|
||||||
HassioInfo,
|
|
||||||
} from "../../src/data/hassio/supervisor";
|
|
||||||
import { HomeAssistant, Route } from "../../src/types";
|
import { HomeAssistant, Route } from "../../src/types";
|
||||||
import "./hassio-panel-router";
|
import "./hassio-panel-router";
|
||||||
|
|
||||||
@ -20,34 +15,19 @@ import "./hassio-panel-router";
|
|||||||
class HassioPanel extends LitElement {
|
class HassioPanel extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ type: Boolean }) public narrow!: boolean;
|
@property({ type: Boolean }) public narrow!: boolean;
|
||||||
|
|
||||||
@property({ attribute: false }) public route!: Route;
|
@property({ attribute: false }) public route!: Route;
|
||||||
|
|
||||||
@property({ attribute: false }) public supervisorInfo!: HassioSupervisorInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hassioInfo!: HassioInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hostInfo!: HassioHostInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hassInfo!: HassioHomeAssistantInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hassOsInfo!: HassioHassOSInfo;
|
|
||||||
|
|
||||||
protected render(): TemplateResult {
|
protected render(): TemplateResult {
|
||||||
if (!this.supervisorInfo) {
|
|
||||||
return html``;
|
|
||||||
}
|
|
||||||
return html`
|
return html`
|
||||||
<hassio-panel-router
|
<hassio-panel-router
|
||||||
.route=${this.route}
|
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
|
.supervisor=${this.supervisor}
|
||||||
|
.route=${this.route}
|
||||||
.narrow=${this.narrow}
|
.narrow=${this.narrow}
|
||||||
.supervisorInfo=${this.supervisorInfo}
|
|
||||||
.hassioInfo=${this.hassioInfo}
|
|
||||||
.hostInfo=${this.hostInfo}
|
|
||||||
.hassInfo=${this.hassInfo}
|
|
||||||
.hassOsInfo=${this.hassOsInfo}
|
|
||||||
></hassio-panel-router>
|
></hassio-panel-router>
|
||||||
`;
|
`;
|
||||||
}
|
}
|
||||||
|
@ -1,24 +1,6 @@
|
|||||||
import {
|
import { customElement, property } from "lit-element";
|
||||||
customElement,
|
import { HassioPanelInfo } from "../../src/data/hassio/supervisor";
|
||||||
property,
|
import { Supervisor } from "../../src/data/supervisor/supervisor";
|
||||||
internalProperty,
|
|
||||||
PropertyValues,
|
|
||||||
} from "lit-element";
|
|
||||||
import {
|
|
||||||
fetchHassioHassOsInfo,
|
|
||||||
fetchHassioHostInfo,
|
|
||||||
HassioHassOSInfo,
|
|
||||||
HassioHostInfo,
|
|
||||||
} from "../../src/data/hassio/host";
|
|
||||||
import {
|
|
||||||
fetchHassioHomeAssistantInfo,
|
|
||||||
fetchHassioSupervisorInfo,
|
|
||||||
fetchHassioInfo,
|
|
||||||
HassioHomeAssistantInfo,
|
|
||||||
HassioInfo,
|
|
||||||
HassioPanelInfo,
|
|
||||||
HassioSupervisorInfo,
|
|
||||||
} from "../../src/data/hassio/supervisor";
|
|
||||||
import {
|
import {
|
||||||
HassRouterPage,
|
HassRouterPage,
|
||||||
RouterOptions,
|
RouterOptions,
|
||||||
@ -32,9 +14,11 @@ import "./hassio-panel";
|
|||||||
class HassioRouter extends HassRouterPage {
|
class HassioRouter extends HassRouterPage {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property() public panel!: HassioPanelInfo;
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property() public narrow!: boolean;
|
@property({ attribute: false }) public panel!: HassioPanelInfo;
|
||||||
|
|
||||||
|
@property({ type: Boolean }) public narrow!: boolean;
|
||||||
|
|
||||||
protected routerOptions: RouterOptions = {
|
protected routerOptions: RouterOptions = {
|
||||||
// Hass.io has a page with tabs, so we route all non-matching routes to it.
|
// Hass.io has a page with tabs, so we route all non-matching routes to it.
|
||||||
@ -51,47 +35,22 @@ class HassioRouter extends HassRouterPage {
|
|||||||
system: "dashboard",
|
system: "dashboard",
|
||||||
addon: {
|
addon: {
|
||||||
tag: "hassio-addon-dashboard",
|
tag: "hassio-addon-dashboard",
|
||||||
load: () =>
|
load: () => import("./addon-view/hassio-addon-dashboard"),
|
||||||
import(
|
|
||||||
/* webpackChunkName: "hassio-addon-dashboard" */ "./addon-view/hassio-addon-dashboard"
|
|
||||||
),
|
|
||||||
},
|
},
|
||||||
ingress: {
|
ingress: {
|
||||||
tag: "hassio-ingress-view",
|
tag: "hassio-ingress-view",
|
||||||
load: () =>
|
load: () => import("./ingress-view/hassio-ingress-view"),
|
||||||
import(
|
|
||||||
/* webpackChunkName: "hassio-ingress-view" */ "./ingress-view/hassio-ingress-view"
|
|
||||||
),
|
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@internalProperty() private _supervisorInfo?: HassioSupervisorInfo;
|
|
||||||
|
|
||||||
@internalProperty() private _hostInfo?: HassioHostInfo;
|
|
||||||
|
|
||||||
@internalProperty() private _hassioInfo?: HassioInfo;
|
|
||||||
|
|
||||||
@internalProperty() private _hassOsInfo?: HassioHassOSInfo;
|
|
||||||
|
|
||||||
@internalProperty() private _hassInfo?: HassioHomeAssistantInfo;
|
|
||||||
|
|
||||||
protected firstUpdated(changedProps: PropertyValues) {
|
|
||||||
super.firstUpdated(changedProps);
|
|
||||||
this.addEventListener("hass-api-called", (ev) => this._apiCalled(ev));
|
|
||||||
}
|
|
||||||
|
|
||||||
protected updatePageEl(el) {
|
protected updatePageEl(el) {
|
||||||
// the tabs page does its own routing so needs full route.
|
// the tabs page does its own routing so needs full route.
|
||||||
const route = el.nodeName === "HASSIO-PANEL" ? this.route : this.routeTail;
|
const route = el.nodeName === "HASSIO-PANEL" ? this.route : this.routeTail;
|
||||||
|
|
||||||
el.hass = this.hass;
|
el.hass = this.hass;
|
||||||
|
el.supervisor = this.supervisor;
|
||||||
el.narrow = this.narrow;
|
el.narrow = this.narrow;
|
||||||
el.supervisorInfo = this._supervisorInfo;
|
|
||||||
el.hassioInfo = this._hassioInfo;
|
|
||||||
el.hostInfo = this._hostInfo;
|
|
||||||
el.hassInfo = this._hassInfo;
|
|
||||||
el.hassOsInfo = this._hassOsInfo;
|
|
||||||
el.route = route;
|
el.route = route;
|
||||||
|
|
||||||
if (el.localName === "hassio-ingress-view") {
|
if (el.localName === "hassio-ingress-view") {
|
||||||
@ -102,45 +61,12 @@ class HassioRouter extends HassRouterPage {
|
|||||||
private async _fetchData() {
|
private async _fetchData() {
|
||||||
if (this.panel.config && this.panel.config.ingress) {
|
if (this.panel.config && this.panel.config.ingress) {
|
||||||
this._redirectIngress(this.panel.config.ingress);
|
this._redirectIngress(this.panel.config.ingress);
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const [supervisorInfo, hostInfo, hassInfo, hassioInfo] = await Promise.all([
|
|
||||||
fetchHassioSupervisorInfo(this.hass),
|
|
||||||
fetchHassioHostInfo(this.hass),
|
|
||||||
fetchHassioHomeAssistantInfo(this.hass),
|
|
||||||
fetchHassioInfo(this.hass),
|
|
||||||
]);
|
|
||||||
this._supervisorInfo = supervisorInfo;
|
|
||||||
this._hassioInfo = hassioInfo;
|
|
||||||
this._hostInfo = hostInfo;
|
|
||||||
this._hassInfo = hassInfo;
|
|
||||||
|
|
||||||
if (this._hostInfo.features && this._hostInfo.features.includes("hassos")) {
|
|
||||||
this._hassOsInfo = await fetchHassioHassOsInfo(this.hass);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private _redirectIngress(addonSlug: string) {
|
private _redirectIngress(addonSlug: string) {
|
||||||
this.route = { prefix: "/hassio", path: `/ingress/${addonSlug}` };
|
this.route = { prefix: "/hassio", path: `/ingress/${addonSlug}` };
|
||||||
}
|
}
|
||||||
|
|
||||||
private _apiCalled(ev) {
|
|
||||||
if (!ev.detail.success) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
let tries = 1;
|
|
||||||
|
|
||||||
const tryUpdate = () => {
|
|
||||||
this._fetchData().catch(() => {
|
|
||||||
tries += 1;
|
|
||||||
setTimeout(tryUpdate, Math.min(tries, 5) * 1000);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
tryUpdate();
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
declare global {
|
declare global {
|
||||||
|
@ -26,7 +26,6 @@ import {
|
|||||||
TemplateResult,
|
TemplateResult,
|
||||||
} from "lit-element";
|
} from "lit-element";
|
||||||
import { atLeastVersion } from "../../../src/common/config/version";
|
import { atLeastVersion } from "../../../src/common/config/version";
|
||||||
import { fireEvent } from "../../../src/common/dom/fire_event";
|
|
||||||
import "../../../src/components/buttons/ha-progress-button";
|
import "../../../src/components/buttons/ha-progress-button";
|
||||||
import "../../../src/components/ha-button-menu";
|
import "../../../src/components/ha-button-menu";
|
||||||
import "../../../src/components/ha-card";
|
import "../../../src/components/ha-card";
|
||||||
@ -41,7 +40,7 @@ import {
|
|||||||
HassioSnapshot,
|
HassioSnapshot,
|
||||||
reloadHassioSnapshots,
|
reloadHassioSnapshots,
|
||||||
} from "../../../src/data/hassio/snapshot";
|
} from "../../../src/data/hassio/snapshot";
|
||||||
import { HassioSupervisorInfo } from "../../../src/data/hassio/supervisor";
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
import "../../../src/layouts/hass-tabs-subpage";
|
import "../../../src/layouts/hass-tabs-subpage";
|
||||||
import { PolymerChangedEvent } from "../../../src/polymer-types";
|
import { PolymerChangedEvent } from "../../../src/polymer-types";
|
||||||
import { haStyle } from "../../../src/resources/styles";
|
import { haStyle } from "../../../src/resources/styles";
|
||||||
@ -67,7 +66,7 @@ class HassioSnapshots extends LitElement {
|
|||||||
|
|
||||||
@property({ attribute: false }) public route!: Route;
|
@property({ attribute: false }) public route!: Route;
|
||||||
|
|
||||||
@property({ attribute: false }) public supervisorInfo!: HassioSupervisorInfo;
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@internalProperty() private _snapshotName = "";
|
@internalProperty() private _snapshotName = "";
|
||||||
|
|
||||||
@ -266,7 +265,7 @@ class HassioSnapshots extends LitElement {
|
|||||||
|
|
||||||
protected updated(changedProps: PropertyValues) {
|
protected updated(changedProps: PropertyValues) {
|
||||||
if (changedProps.has("supervisorInfo")) {
|
if (changedProps.has("supervisorInfo")) {
|
||||||
this._addonList = this.supervisorInfo.addons
|
this._addonList = this.supervisor.supervisor.addons
|
||||||
.map((addon) => ({
|
.map((addon) => ({
|
||||||
slug: addon.slug,
|
slug: addon.slug,
|
||||||
name: addon.name,
|
name: addon.name,
|
||||||
@ -372,7 +371,6 @@ class HassioSnapshots extends LitElement {
|
|||||||
await createHassioPartialSnapshot(this.hass, data);
|
await createHassioPartialSnapshot(this.hass, data);
|
||||||
}
|
}
|
||||||
this._updateSnapshots();
|
this._updateSnapshots();
|
||||||
fireEvent(this, "hass-api-called", { success: true, response: null });
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
this._error = extractApiErrorMessage(err);
|
this._error = extractApiErrorMessage(err);
|
||||||
}
|
}
|
||||||
|
69
hassio/src/supervisor-base-element.ts
Normal file
69
hassio/src/supervisor-base-element.ts
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
import { LitElement, property, PropertyValues } from "lit-element";
|
||||||
|
import {
|
||||||
|
fetchHassioHassOsInfo,
|
||||||
|
fetchHassioHostInfo,
|
||||||
|
} from "../../src/data/hassio/host";
|
||||||
|
import { fetchNetworkInfo } from "../../src/data/hassio/network";
|
||||||
|
import { fetchHassioResolution } from "../../src/data/hassio/resolution";
|
||||||
|
import {
|
||||||
|
fetchHassioHomeAssistantInfo,
|
||||||
|
fetchHassioInfo,
|
||||||
|
fetchHassioSupervisorInfo,
|
||||||
|
} from "../../src/data/hassio/supervisor";
|
||||||
|
import { Supervisor } from "../../src/data/supervisor/supervisor";
|
||||||
|
import { ProvideHassLitMixin } from "../../src/mixins/provide-hass-lit-mixin";
|
||||||
|
import { urlSyncMixin } from "../../src/state/url-sync-mixin";
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HASSDomEvents {
|
||||||
|
"supervisor-update": Partial<Supervisor>;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export class SupervisorBaseElement extends urlSyncMixin(
|
||||||
|
ProvideHassLitMixin(LitElement)
|
||||||
|
) {
|
||||||
|
@property({ attribute: false }) public supervisor?: Supervisor;
|
||||||
|
|
||||||
|
protected _updateSupervisor(obj: Partial<Supervisor>): void {
|
||||||
|
this.supervisor = { ...this.supervisor!, ...obj };
|
||||||
|
}
|
||||||
|
|
||||||
|
protected firstUpdated(changedProps: PropertyValues): void {
|
||||||
|
super.firstUpdated(changedProps);
|
||||||
|
this._initSupervisor();
|
||||||
|
this.addEventListener("supervisor-update", (ev) =>
|
||||||
|
this._updateSupervisor(ev.detail)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private async _initSupervisor(): Promise<void> {
|
||||||
|
const [
|
||||||
|
supervisor,
|
||||||
|
host,
|
||||||
|
core,
|
||||||
|
info,
|
||||||
|
os,
|
||||||
|
network,
|
||||||
|
resolution,
|
||||||
|
] = await Promise.all([
|
||||||
|
fetchHassioSupervisorInfo(this.hass),
|
||||||
|
fetchHassioHostInfo(this.hass),
|
||||||
|
fetchHassioHomeAssistantInfo(this.hass),
|
||||||
|
fetchHassioInfo(this.hass),
|
||||||
|
fetchHassioHassOsInfo(this.hass),
|
||||||
|
fetchNetworkInfo(this.hass),
|
||||||
|
fetchHassioResolution(this.hass),
|
||||||
|
]);
|
||||||
|
|
||||||
|
this.supervisor = {
|
||||||
|
supervisor,
|
||||||
|
host,
|
||||||
|
core,
|
||||||
|
info,
|
||||||
|
os,
|
||||||
|
network,
|
||||||
|
resolution,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
@ -8,12 +8,12 @@ import {
|
|||||||
CSSResult,
|
CSSResult,
|
||||||
customElement,
|
customElement,
|
||||||
html,
|
html,
|
||||||
internalProperty,
|
|
||||||
LitElement,
|
LitElement,
|
||||||
property,
|
property,
|
||||||
TemplateResult,
|
TemplateResult,
|
||||||
} from "lit-element";
|
} from "lit-element";
|
||||||
import memoizeOne from "memoize-one";
|
import memoizeOne from "memoize-one";
|
||||||
|
import { fireEvent } from "../../../src/common/dom/fire_event";
|
||||||
import "../../../src/components/buttons/ha-progress-button";
|
import "../../../src/components/buttons/ha-progress-button";
|
||||||
import "../../../src/components/ha-button-menu";
|
import "../../../src/components/ha-button-menu";
|
||||||
import "../../../src/components/ha-card";
|
import "../../../src/components/ha-card";
|
||||||
@ -27,8 +27,6 @@ import {
|
|||||||
changeHostOptions,
|
changeHostOptions,
|
||||||
configSyncOS,
|
configSyncOS,
|
||||||
fetchHassioHostInfo,
|
fetchHassioHostInfo,
|
||||||
HassioHassOSInfo,
|
|
||||||
HassioHostInfo as HassioHostInfoType,
|
|
||||||
rebootHost,
|
rebootHost,
|
||||||
shutdownHost,
|
shutdownHost,
|
||||||
updateOS,
|
updateOS,
|
||||||
@ -37,7 +35,7 @@ import {
|
|||||||
fetchNetworkInfo,
|
fetchNetworkInfo,
|
||||||
NetworkInfo,
|
NetworkInfo,
|
||||||
} from "../../../src/data/hassio/network";
|
} from "../../../src/data/hassio/network";
|
||||||
import { HassioInfo } from "../../../src/data/hassio/supervisor";
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
import {
|
import {
|
||||||
showAlertDialog,
|
showAlertDialog,
|
||||||
showConfirmationDialog,
|
showConfirmationDialog,
|
||||||
@ -53,28 +51,22 @@ import { hassioStyle } from "../resources/hassio-style";
|
|||||||
class HassioHostInfo extends LitElement {
|
class HassioHostInfo extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property() public hostInfo!: HassioHostInfoType;
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ attribute: false }) public hassioInfo!: HassioInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hassOsInfo!: HassioHassOSInfo;
|
|
||||||
|
|
||||||
@internalProperty() public _networkInfo?: NetworkInfo;
|
|
||||||
|
|
||||||
protected render(): TemplateResult | void {
|
protected render(): TemplateResult | void {
|
||||||
const primaryIpAddress = this.hostInfo.features.includes("network")
|
const primaryIpAddress = this.supervisor.host.features.includes("network")
|
||||||
? this._primaryIpAddress(this._networkInfo!)
|
? this._primaryIpAddress(this.supervisor.network!)
|
||||||
: "";
|
: "";
|
||||||
return html`
|
return html`
|
||||||
<ha-card header="Host System">
|
<ha-card header="Host System">
|
||||||
<div class="card-content">
|
<div class="card-content">
|
||||||
${this.hostInfo.features.includes("hostname")
|
${this.supervisor.host.features.includes("hostname")
|
||||||
? html`<ha-settings-row>
|
? html`<ha-settings-row>
|
||||||
<span slot="heading">
|
<span slot="heading">
|
||||||
Hostname
|
Hostname
|
||||||
</span>
|
</span>
|
||||||
<span slot="description">
|
<span slot="description">
|
||||||
${this.hostInfo.hostname}
|
${this.supervisor.host.hostname}
|
||||||
</span>
|
</span>
|
||||||
<mwc-button
|
<mwc-button
|
||||||
title="Change the hostname"
|
title="Change the hostname"
|
||||||
@ -84,7 +76,7 @@ class HassioHostInfo extends LitElement {
|
|||||||
</mwc-button>
|
</mwc-button>
|
||||||
</ha-settings-row>`
|
</ha-settings-row>`
|
||||||
: ""}
|
: ""}
|
||||||
${this.hostInfo.features.includes("network")
|
${this.supervisor.host.features.includes("network")
|
||||||
? html` <ha-settings-row>
|
? html` <ha-settings-row>
|
||||||
<span slot="heading">
|
<span slot="heading">
|
||||||
IP Address
|
IP Address
|
||||||
@ -106,10 +98,9 @@ class HassioHostInfo extends LitElement {
|
|||||||
Operating System
|
Operating System
|
||||||
</span>
|
</span>
|
||||||
<span slot="description">
|
<span slot="description">
|
||||||
${this.hostInfo.operating_system}
|
${this.supervisor.host.operating_system}
|
||||||
</span>
|
</span>
|
||||||
${this.hostInfo.features.includes("hassos") &&
|
${this.supervisor.os.update_available
|
||||||
this.hassOsInfo.update_available
|
|
||||||
? html`
|
? html`
|
||||||
<ha-progress-button
|
<ha-progress-button
|
||||||
title="Update the host OS"
|
title="Update the host OS"
|
||||||
@ -120,29 +111,29 @@ class HassioHostInfo extends LitElement {
|
|||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
</ha-settings-row>
|
</ha-settings-row>
|
||||||
${!this.hostInfo.features.includes("hassos")
|
${!this.supervisor.host.features.includes("hassos")
|
||||||
? html`<ha-settings-row>
|
? html`<ha-settings-row>
|
||||||
<span slot="heading">
|
<span slot="heading">
|
||||||
Docker version
|
Docker version
|
||||||
</span>
|
</span>
|
||||||
<span slot="description">
|
<span slot="description">
|
||||||
${this.hassioInfo.docker}
|
${this.supervisor.info.docker}
|
||||||
</span>
|
</span>
|
||||||
</ha-settings-row>`
|
</ha-settings-row>`
|
||||||
: ""}
|
: ""}
|
||||||
${this.hostInfo.deployment
|
${this.supervisor.host.deployment
|
||||||
? html`<ha-settings-row>
|
? html`<ha-settings-row>
|
||||||
<span slot="heading">
|
<span slot="heading">
|
||||||
Deployment
|
Deployment
|
||||||
</span>
|
</span>
|
||||||
<span slot="description">
|
<span slot="description">
|
||||||
${this.hostInfo.deployment}
|
${this.supervisor.host.deployment}
|
||||||
</span>
|
</span>
|
||||||
</ha-settings-row>`
|
</ha-settings-row>`
|
||||||
: ""}
|
: ""}
|
||||||
</div>
|
</div>
|
||||||
<div class="card-actions">
|
<div class="card-actions">
|
||||||
${this.hostInfo.features.includes("reboot")
|
${this.supervisor.host.features.includes("reboot")
|
||||||
? html`
|
? html`
|
||||||
<ha-progress-button
|
<ha-progress-button
|
||||||
title="Reboot the host OS"
|
title="Reboot the host OS"
|
||||||
@ -153,7 +144,7 @@ class HassioHostInfo extends LitElement {
|
|||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
`
|
`
|
||||||
: ""}
|
: ""}
|
||||||
${this.hostInfo.features.includes("shutdown")
|
${this.supervisor.host.features.includes("shutdown")
|
||||||
? html`
|
? html`
|
||||||
<ha-progress-button
|
<ha-progress-button
|
||||||
title="Shutdown the host OS"
|
title="Shutdown the host OS"
|
||||||
@ -175,7 +166,7 @@ class HassioHostInfo extends LitElement {
|
|||||||
<mwc-list-item title="Show a list of hardware">
|
<mwc-list-item title="Show a list of hardware">
|
||||||
Hardware
|
Hardware
|
||||||
</mwc-list-item>
|
</mwc-list-item>
|
||||||
${this.hostInfo.features.includes("hassos")
|
${this.supervisor.host.features.includes("hassos")
|
||||||
? html`<mwc-list-item
|
? html`<mwc-list-item
|
||||||
title="Load HassOS configs or updates from USB"
|
title="Load HassOS configs or updates from USB"
|
||||||
>
|
>
|
||||||
@ -193,12 +184,10 @@ class HassioHostInfo extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private _primaryIpAddress = memoizeOne((network_info: NetworkInfo) => {
|
private _primaryIpAddress = memoizeOne((network_info: NetworkInfo) => {
|
||||||
if (!network_info) {
|
if (!network_info || !network_info.interfaces) {
|
||||||
return "";
|
return "";
|
||||||
}
|
}
|
||||||
return Object.keys(network_info?.interfaces)
|
return network_info.interfaces.find((a) => a.primary)?.ipv4?.address![0];
|
||||||
.map((device) => network_info.interfaces[device])
|
|
||||||
.find((device) => device.primary)?.ip_address;
|
|
||||||
});
|
});
|
||||||
|
|
||||||
private async _handleMenuAction(ev: CustomEvent<ActionDetail>) {
|
private async _handleMenuAction(ev: CustomEvent<ActionDetail>) {
|
||||||
@ -316,13 +305,13 @@ class HassioHostInfo extends LitElement {
|
|||||||
|
|
||||||
private async _changeNetworkClicked(): Promise<void> {
|
private async _changeNetworkClicked(): Promise<void> {
|
||||||
showNetworkDialog(this, {
|
showNetworkDialog(this, {
|
||||||
network: this._networkInfo!,
|
network: this.supervisor.network!,
|
||||||
loadData: () => this._loadData(),
|
loadData: () => this._loadData(),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _changeHostnameClicked(): Promise<void> {
|
private async _changeHostnameClicked(): Promise<void> {
|
||||||
const curHostname: string = this.hostInfo.hostname;
|
const curHostname: string = this.supervisor.host.hostname;
|
||||||
const hostname = await showPromptDialog(this, {
|
const hostname = await showPromptDialog(this, {
|
||||||
title: "Change Hostname",
|
title: "Change Hostname",
|
||||||
inputLabel: "Please enter a new hostname:",
|
inputLabel: "Please enter a new hostname:",
|
||||||
@ -333,7 +322,8 @@ class HassioHostInfo extends LitElement {
|
|||||||
if (hostname && hostname !== curHostname) {
|
if (hostname && hostname !== curHostname) {
|
||||||
try {
|
try {
|
||||||
await changeHostOptions(this.hass, { hostname });
|
await changeHostOptions(this.hass, { hostname });
|
||||||
this.hostInfo = await fetchHassioHostInfo(this.hass);
|
const host = await fetchHassioHostInfo(this.hass);
|
||||||
|
fireEvent(this, "supervisor-update", { host });
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Setting hostname failed",
|
title: "Setting hostname failed",
|
||||||
@ -346,7 +336,8 @@ class HassioHostInfo extends LitElement {
|
|||||||
private async _importFromUSB(): Promise<void> {
|
private async _importFromUSB(): Promise<void> {
|
||||||
try {
|
try {
|
||||||
await configSyncOS(this.hass);
|
await configSyncOS(this.hass);
|
||||||
this.hostInfo = await fetchHassioHostInfo(this.hass);
|
const host = await fetchHassioHostInfo(this.hass);
|
||||||
|
fireEvent(this, "supervisor-update", { host });
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to import from USB",
|
title: "Failed to import from USB",
|
||||||
@ -356,7 +347,8 @@ class HassioHostInfo extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private async _loadData(): Promise<void> {
|
private async _loadData(): Promise<void> {
|
||||||
this._networkInfo = await fetchNetworkInfo(this.hass);
|
const network = await fetchNetworkInfo(this.hass);
|
||||||
|
fireEvent(this, "supervisor-update", { network });
|
||||||
}
|
}
|
||||||
|
|
||||||
static get styles(): CSSResult[] {
|
static get styles(): CSSResult[] {
|
||||||
|
@ -13,16 +13,15 @@ import "../../../src/components/ha-card";
|
|||||||
import "../../../src/components/ha-settings-row";
|
import "../../../src/components/ha-settings-row";
|
||||||
import "../../../src/components/ha-switch";
|
import "../../../src/components/ha-switch";
|
||||||
import { extractApiErrorMessage } from "../../../src/data/hassio/common";
|
import { extractApiErrorMessage } from "../../../src/data/hassio/common";
|
||||||
import { HassioHostInfo as HassioHostInfoType } from "../../../src/data/hassio/host";
|
|
||||||
import { fetchHassioResolution } from "../../../src/data/hassio/resolution";
|
|
||||||
import {
|
import {
|
||||||
fetchHassioSupervisorInfo,
|
fetchHassioSupervisorInfo,
|
||||||
HassioSupervisorInfo as HassioSupervisorInfoType,
|
|
||||||
reloadSupervisor,
|
reloadSupervisor,
|
||||||
|
restartSupervisor,
|
||||||
setSupervisorOption,
|
setSupervisorOption,
|
||||||
SupervisorOptions,
|
SupervisorOptions,
|
||||||
updateSupervisor,
|
updateSupervisor,
|
||||||
} from "../../../src/data/hassio/supervisor";
|
} from "../../../src/data/hassio/supervisor";
|
||||||
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
import {
|
import {
|
||||||
showAlertDialog,
|
showAlertDialog,
|
||||||
showConfirmationDialog,
|
showConfirmationDialog,
|
||||||
@ -32,7 +31,7 @@ import { HomeAssistant } from "../../../src/types";
|
|||||||
import { documentationUrl } from "../../../src/util/documentation-url";
|
import { documentationUrl } from "../../../src/util/documentation-url";
|
||||||
import { hassioStyle } from "../resources/hassio-style";
|
import { hassioStyle } from "../resources/hassio-style";
|
||||||
|
|
||||||
const ISSUES = {
|
const UNSUPPORTED_REASON = {
|
||||||
container: {
|
container: {
|
||||||
title: "Containers known to cause issues",
|
title: "Containers known to cause issues",
|
||||||
url: "/more-info/unsupported/container",
|
url: "/more-info/unsupported/container",
|
||||||
@ -46,6 +45,10 @@ const ISSUES = {
|
|||||||
title: "Docker Version",
|
title: "Docker Version",
|
||||||
url: "/more-info/unsupported/docker_version",
|
url: "/more-info/unsupported/docker_version",
|
||||||
},
|
},
|
||||||
|
job_conditions: {
|
||||||
|
title: "Ignored job conditions",
|
||||||
|
url: "/more-info/unsupported/job_conditions",
|
||||||
|
},
|
||||||
lxc: { title: "LXC", url: "/more-info/unsupported/lxc" },
|
lxc: { title: "LXC", url: "/more-info/unsupported/lxc" },
|
||||||
network_manager: {
|
network_manager: {
|
||||||
title: "Network Manager",
|
title: "Network Manager",
|
||||||
@ -59,14 +62,30 @@ const ISSUES = {
|
|||||||
systemd: { title: "Systemd", url: "/more-info/unsupported/systemd" },
|
systemd: { title: "Systemd", url: "/more-info/unsupported/systemd" },
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const UNHEALTHY_REASON = {
|
||||||
|
privileged: {
|
||||||
|
title: "Supervisor is not privileged",
|
||||||
|
url: "/more-info/unsupported/privileged",
|
||||||
|
},
|
||||||
|
supervisor: {
|
||||||
|
title: "Supervisor was not able to update",
|
||||||
|
url: "/more-info/unhealthy/supervisor",
|
||||||
|
},
|
||||||
|
setup: {
|
||||||
|
title: "Setup of the Supervisor failed",
|
||||||
|
url: "/more-info/unhealthy/setup",
|
||||||
|
},
|
||||||
|
docker: {
|
||||||
|
title: "The Docker environment is not working properly",
|
||||||
|
url: "/more-info/unhealthy/docker",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
@customElement("hassio-supervisor-info")
|
@customElement("hassio-supervisor-info")
|
||||||
class HassioSupervisorInfo extends LitElement {
|
class HassioSupervisorInfo extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property({ attribute: false })
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
public supervisorInfo!: HassioSupervisorInfoType;
|
|
||||||
|
|
||||||
@property() public hostInfo!: HassioHostInfoType;
|
|
||||||
|
|
||||||
protected render(): TemplateResult | void {
|
protected render(): TemplateResult | void {
|
||||||
return html`
|
return html`
|
||||||
@ -77,7 +96,7 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
Version
|
Version
|
||||||
</span>
|
</span>
|
||||||
<span slot="description">
|
<span slot="description">
|
||||||
${this.supervisorInfo.version}
|
${this.supervisor.supervisor.version}
|
||||||
</span>
|
</span>
|
||||||
</ha-settings-row>
|
</ha-settings-row>
|
||||||
<ha-settings-row>
|
<ha-settings-row>
|
||||||
@ -85,9 +104,9 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
Newest Version
|
Newest Version
|
||||||
</span>
|
</span>
|
||||||
<span slot="description">
|
<span slot="description">
|
||||||
${this.supervisorInfo.version_latest}
|
${this.supervisor.supervisor.version_latest}
|
||||||
</span>
|
</span>
|
||||||
${this.supervisorInfo.update_available
|
${this.supervisor.supervisor.update_available
|
||||||
? html`
|
? html`
|
||||||
<ha-progress-button
|
<ha-progress-button
|
||||||
title="Update the supervisor"
|
title="Update the supervisor"
|
||||||
@ -103,9 +122,9 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
Channel
|
Channel
|
||||||
</span>
|
</span>
|
||||||
<span slot="description">
|
<span slot="description">
|
||||||
${this.supervisorInfo.channel}
|
${this.supervisor.supervisor.channel}
|
||||||
</span>
|
</span>
|
||||||
${this.supervisorInfo.channel === "beta"
|
${this.supervisor.supervisor.channel === "beta"
|
||||||
? html`
|
? html`
|
||||||
<ha-progress-button
|
<ha-progress-button
|
||||||
@click=${this._toggleBeta}
|
@click=${this._toggleBeta}
|
||||||
@ -114,7 +133,7 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
Leave beta channel
|
Leave beta channel
|
||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
`
|
`
|
||||||
: this.supervisorInfo.channel === "stable"
|
: this.supervisor.supervisor.channel === "stable"
|
||||||
? html`
|
? html`
|
||||||
<ha-progress-button
|
<ha-progress-button
|
||||||
@click=${this._toggleBeta}
|
@click=${this._toggleBeta}
|
||||||
@ -126,7 +145,7 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
: ""}
|
: ""}
|
||||||
</ha-settings-row>
|
</ha-settings-row>
|
||||||
|
|
||||||
${this.supervisorInfo?.supported
|
${this.supervisor.supervisor.supported
|
||||||
? html` <ha-settings-row three-line>
|
? html` <ha-settings-row three-line>
|
||||||
<span slot="heading">
|
<span slot="heading">
|
||||||
Share Diagnostics
|
Share Diagnostics
|
||||||
@ -143,7 +162,7 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
</div>
|
</div>
|
||||||
<ha-switch
|
<ha-switch
|
||||||
haptic
|
haptic
|
||||||
.checked=${this.supervisorInfo.diagnostics}
|
.checked=${this.supervisor.supervisor.diagnostics}
|
||||||
@change=${this._toggleDiagnostics}
|
@change=${this._toggleDiagnostics}
|
||||||
></ha-switch>
|
></ha-switch>
|
||||||
</ha-settings-row>`
|
</ha-settings-row>`
|
||||||
@ -157,14 +176,33 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
Learn more
|
Learn more
|
||||||
</button>
|
</button>
|
||||||
</div>`}
|
</div>`}
|
||||||
|
${!this.supervisor.supervisor.healthy
|
||||||
|
? html`<div class="error">
|
||||||
|
Your installtion is running in an unhealthy state.
|
||||||
|
<button
|
||||||
|
class="link"
|
||||||
|
title="Learn more about why your system is marked as unhealthy"
|
||||||
|
@click=${this._unhealthyDialog}
|
||||||
|
>
|
||||||
|
Learn more
|
||||||
|
</button>
|
||||||
|
</div>`
|
||||||
|
: ""}
|
||||||
</div>
|
</div>
|
||||||
<div class="card-actions">
|
<div class="card-actions">
|
||||||
<ha-progress-button
|
<ha-progress-button
|
||||||
@click=${this._supervisorReload}
|
@click=${this._supervisorReload}
|
||||||
title="Reload parts of the supervisor"
|
title="Reload parts of the Supervisor"
|
||||||
>
|
>
|
||||||
Reload
|
Reload
|
||||||
</ha-progress-button>
|
</ha-progress-button>
|
||||||
|
<ha-progress-button
|
||||||
|
class="warning"
|
||||||
|
@click=${this._supervisorRestart}
|
||||||
|
title="Restart the Supervisor"
|
||||||
|
>
|
||||||
|
Restart
|
||||||
|
</ha-progress-button>
|
||||||
</div>
|
</div>
|
||||||
</ha-card>
|
</ha-card>
|
||||||
`;
|
`;
|
||||||
@ -174,7 +212,7 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
const button = ev.currentTarget as any;
|
const button = ev.currentTarget as any;
|
||||||
button.progress = true;
|
button.progress = true;
|
||||||
|
|
||||||
if (this.supervisorInfo.channel === "stable") {
|
if (this.supervisor.supervisor.channel === "stable") {
|
||||||
const confirmed = await showConfirmationDialog(this, {
|
const confirmed = await showConfirmationDialog(this, {
|
||||||
title: "WARNING",
|
title: "WARNING",
|
||||||
text: html` Beta releases are for testers and early adopters and can
|
text: html` Beta releases are for testers and early adopters and can
|
||||||
@ -203,18 +241,19 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
const data: Partial<SupervisorOptions> = {
|
const data: Partial<SupervisorOptions> = {
|
||||||
channel: this.supervisorInfo.channel === "stable" ? "beta" : "stable",
|
channel:
|
||||||
|
this.supervisor.supervisor.channel === "stable" ? "beta" : "stable",
|
||||||
};
|
};
|
||||||
await setSupervisorOption(this.hass, data);
|
await setSupervisorOption(this.hass, data);
|
||||||
await reloadSupervisor(this.hass);
|
await this._reloadSupervisor();
|
||||||
fireEvent(this, "hass-api-called", { success: true, response: null });
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to set supervisor option",
|
title: "Failed to set supervisor option",
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
|
} finally {
|
||||||
|
button.progress = false;
|
||||||
}
|
}
|
||||||
button.progress = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _supervisorReload(ev: CustomEvent): Promise<void> {
|
private async _supervisorReload(ev: CustomEvent): Promise<void> {
|
||||||
@ -222,15 +261,37 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
button.progress = true;
|
button.progress = true;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await reloadSupervisor(this.hass);
|
await this._reloadSupervisor();
|
||||||
this.supervisorInfo = await fetchHassioSupervisorInfo(this.hass);
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
showAlertDialog(this, {
|
showAlertDialog(this, {
|
||||||
title: "Failed to reload the supervisor",
|
title: "Failed to reload the supervisor",
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
|
} finally {
|
||||||
|
button.progress = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private async _reloadSupervisor(): Promise<void> {
|
||||||
|
await reloadSupervisor(this.hass);
|
||||||
|
const supervisor = await fetchHassioSupervisorInfo(this.hass);
|
||||||
|
fireEvent(this, "supervisor-update", { supervisor });
|
||||||
|
}
|
||||||
|
|
||||||
|
private async _supervisorRestart(ev: CustomEvent): Promise<void> {
|
||||||
|
const button = ev.currentTarget as any;
|
||||||
|
button.progress = true;
|
||||||
|
|
||||||
|
try {
|
||||||
|
await restartSupervisor(this.hass);
|
||||||
|
} catch (err) {
|
||||||
|
showAlertDialog(this, {
|
||||||
|
title: "Failed to restart the supervisor",
|
||||||
|
text: extractApiErrorMessage(err),
|
||||||
|
});
|
||||||
|
} finally {
|
||||||
|
button.progress = false;
|
||||||
}
|
}
|
||||||
button.progress = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _supervisorUpdate(ev: CustomEvent): Promise<void> {
|
private async _supervisorUpdate(ev: CustomEvent): Promise<void> {
|
||||||
@ -239,7 +300,7 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
|
|
||||||
const confirmed = await showConfirmationDialog(this, {
|
const confirmed = await showConfirmationDialog(this, {
|
||||||
title: "Update Supervisor",
|
title: "Update Supervisor",
|
||||||
text: `Are you sure you want to update supervisor to version ${this.supervisorInfo.version_latest}?`,
|
text: `Are you sure you want to update supervisor to version ${this.supervisor.supervisor.version_latest}?`,
|
||||||
confirmText: "update",
|
confirmText: "update",
|
||||||
dismissText: "cancel",
|
dismissText: "cancel",
|
||||||
});
|
});
|
||||||
@ -256,8 +317,9 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
title: "Failed to update the supervisor",
|
title: "Failed to update the supervisor",
|
||||||
text: extractApiErrorMessage(err),
|
text: extractApiErrorMessage(err),
|
||||||
});
|
});
|
||||||
|
} finally {
|
||||||
|
button.progress = false;
|
||||||
}
|
}
|
||||||
button.progress = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private async _diagnosticsInformationDialog(): Promise<void> {
|
private async _diagnosticsInformationDialog(): Promise<void> {
|
||||||
@ -276,22 +338,53 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
private async _unsupportedDialog(): Promise<void> {
|
private async _unsupportedDialog(): Promise<void> {
|
||||||
const resolution = await fetchHassioResolution(this.hass);
|
|
||||||
await showAlertDialog(this, {
|
await showAlertDialog(this, {
|
||||||
title: "You are running an unsupported installation",
|
title: "You are running an unsupported installation",
|
||||||
text: html`Below is a list of issues found with your installation, click
|
text: html`Below is a list of issues found with your installation, click
|
||||||
on the links to learn how you can resolve the issues. <br /><br />
|
on the links to learn how you can resolve the issues. <br /><br />
|
||||||
<ul>
|
<ul>
|
||||||
${resolution.unsupported.map(
|
${this.supervisor.resolution.unsupported.map(
|
||||||
(issue) => html`
|
(issue) => html`
|
||||||
<li>
|
<li>
|
||||||
${ISSUES[issue]
|
${UNSUPPORTED_REASON[issue]
|
||||||
? html`<a
|
? html`<a
|
||||||
href="${documentationUrl(this.hass, ISSUES[issue].url)}"
|
href="${documentationUrl(
|
||||||
|
this.hass,
|
||||||
|
UNSUPPORTED_REASON[issue].url
|
||||||
|
)}"
|
||||||
target="_blank"
|
target="_blank"
|
||||||
rel="noreferrer"
|
rel="noreferrer"
|
||||||
>
|
>
|
||||||
${ISSUES[issue].title}
|
${UNSUPPORTED_REASON[issue].title}
|
||||||
|
</a>`
|
||||||
|
: issue}
|
||||||
|
</li>
|
||||||
|
`
|
||||||
|
)}
|
||||||
|
</ul>`,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
private async _unhealthyDialog(): Promise<void> {
|
||||||
|
await showAlertDialog(this, {
|
||||||
|
title: "Your installation is unhealthy",
|
||||||
|
text: html`Running an unhealthy installation will cause issues. Below is a
|
||||||
|
list of issues found with your installation, click on the links to learn
|
||||||
|
how you can resolve the issues. <br /><br />
|
||||||
|
<ul>
|
||||||
|
${this.supervisor.resolution.unhealthy.map(
|
||||||
|
(issue) => html`
|
||||||
|
<li>
|
||||||
|
${UNHEALTHY_REASON[issue]
|
||||||
|
? html`<a
|
||||||
|
href="${documentationUrl(
|
||||||
|
this.hass,
|
||||||
|
UNHEALTHY_REASON[issue].url
|
||||||
|
)}"
|
||||||
|
target="_blank"
|
||||||
|
rel="noreferrer"
|
||||||
|
>
|
||||||
|
${UNHEALTHY_REASON[issue].title}
|
||||||
</a>`
|
</a>`
|
||||||
: issue}
|
: issue}
|
||||||
</li>
|
</li>
|
||||||
@ -304,7 +397,7 @@ class HassioSupervisorInfo extends LitElement {
|
|||||||
private async _toggleDiagnostics(): Promise<void> {
|
private async _toggleDiagnostics(): Promise<void> {
|
||||||
try {
|
try {
|
||||||
const data: SupervisorOptions = {
|
const data: SupervisorOptions = {
|
||||||
diagnostics: !this.supervisorInfo?.diagnostics,
|
diagnostics: !this.supervisor.supervisor?.diagnostics,
|
||||||
};
|
};
|
||||||
await setSupervisorOption(this.hass, data);
|
await setSupervisorOption(this.hass, data);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
@ -19,6 +19,7 @@ import "../../../src/components/ha-card";
|
|||||||
import "../../../src/components/ha-settings-row";
|
import "../../../src/components/ha-settings-row";
|
||||||
import { fetchHassioStats, HassioStats } from "../../../src/data/hassio/common";
|
import { fetchHassioStats, HassioStats } from "../../../src/data/hassio/common";
|
||||||
import { HassioHostInfo } from "../../../src/data/hassio/host";
|
import { HassioHostInfo } from "../../../src/data/hassio/host";
|
||||||
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
import { haStyle } from "../../../src/resources/styles";
|
import { haStyle } from "../../../src/resources/styles";
|
||||||
import { HomeAssistant } from "../../../src/types";
|
import { HomeAssistant } from "../../../src/types";
|
||||||
import { bytesToString } from "../../../src/util/bytes-to-string";
|
import { bytesToString } from "../../../src/util/bytes-to-string";
|
||||||
@ -32,7 +33,7 @@ import { hassioStyle } from "../resources/hassio-style";
|
|||||||
class HassioSystemMetrics extends LitElement {
|
class HassioSystemMetrics extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
@property() public hostInfo!: HassioHostInfo;
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@internalProperty() private _supervisorMetrics?: HassioStats;
|
@internalProperty() private _supervisorMetrics?: HassioStats;
|
||||||
|
|
||||||
@ -64,8 +65,8 @@ class HassioSystemMetrics extends LitElement {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
description: "Used Space",
|
description: "Used Space",
|
||||||
value: this._getUsedSpace(this.hostInfo),
|
value: this._getUsedSpace(this.supervisor.host),
|
||||||
tooltip: `${this.hostInfo.disk_used} GB/${this.hostInfo.disk_total} GB`,
|
tooltip: `${this.supervisor.host.disk_used} GB/${this.supervisor.host.disk_total} GB`,
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
|
||||||
|
@ -7,14 +7,7 @@ import {
|
|||||||
property,
|
property,
|
||||||
TemplateResult,
|
TemplateResult,
|
||||||
} from "lit-element";
|
} from "lit-element";
|
||||||
import {
|
import { Supervisor } from "../../../src/data/supervisor/supervisor";
|
||||||
HassioHassOSInfo,
|
|
||||||
HassioHostInfo,
|
|
||||||
} from "../../../src/data/hassio/host";
|
|
||||||
import {
|
|
||||||
HassioInfo,
|
|
||||||
HassioSupervisorInfo,
|
|
||||||
} from "../../../src/data/hassio/supervisor";
|
|
||||||
import "../../../src/layouts/hass-tabs-subpage";
|
import "../../../src/layouts/hass-tabs-subpage";
|
||||||
import { haStyle } from "../../../src/resources/styles";
|
import { haStyle } from "../../../src/resources/styles";
|
||||||
import { HomeAssistant, Route } from "../../../src/types";
|
import { HomeAssistant, Route } from "../../../src/types";
|
||||||
@ -29,18 +22,12 @@ import "./hassio-system-metrics";
|
|||||||
class HassioSystem extends LitElement {
|
class HassioSystem extends LitElement {
|
||||||
@property({ attribute: false }) public hass!: HomeAssistant;
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public supervisor!: Supervisor;
|
||||||
|
|
||||||
@property({ type: Boolean }) public narrow!: boolean;
|
@property({ type: Boolean }) public narrow!: boolean;
|
||||||
|
|
||||||
@property({ attribute: false }) public route!: Route;
|
@property({ attribute: false }) public route!: Route;
|
||||||
|
|
||||||
@property() public supervisorInfo!: HassioSupervisorInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hassioInfo!: HassioInfo;
|
|
||||||
|
|
||||||
@property() public hostInfo!: HassioHostInfo;
|
|
||||||
|
|
||||||
@property({ attribute: false }) public hassOsInfo!: HassioHassOSInfo;
|
|
||||||
|
|
||||||
protected render(): TemplateResult | void {
|
protected render(): TemplateResult | void {
|
||||||
return html`
|
return html`
|
||||||
<hass-tabs-subpage
|
<hass-tabs-subpage
|
||||||
@ -56,18 +43,15 @@ class HassioSystem extends LitElement {
|
|||||||
<div class="card-group">
|
<div class="card-group">
|
||||||
<hassio-supervisor-info
|
<hassio-supervisor-info
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.hostInfo=${this.hostInfo}
|
.supervisor=${this.supervisor}
|
||||||
.supervisorInfo=${this.supervisorInfo}
|
|
||||||
></hassio-supervisor-info>
|
></hassio-supervisor-info>
|
||||||
<hassio-host-info
|
<hassio-host-info
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.hassioInfo=${this.hassioInfo}
|
.supervisor=${this.supervisor}
|
||||||
.hostInfo=${this.hostInfo}
|
|
||||||
.hassOsInfo=${this.hassOsInfo}
|
|
||||||
></hassio-host-info>
|
></hassio-host-info>
|
||||||
<hassio-system-metrics
|
<hassio-system-metrics
|
||||||
.hass=${this.hass}
|
.hass=${this.hass}
|
||||||
.hostInfo=${this.hostInfo}
|
.supervisor=${this.supervisor}
|
||||||
></hassio-system-metrics>
|
></hassio-system-metrics>
|
||||||
</div>
|
</div>
|
||||||
<hassio-supervisor-log .hass=${this.hass}></hassio-supervisor-log>
|
<hassio-supervisor-log .hass=${this.hass}></hassio-supervisor-log>
|
||||||
|
15
package.json
15
package.json
@ -83,6 +83,9 @@
|
|||||||
"@types/sortablejs": "^1.10.6",
|
"@types/sortablejs": "^1.10.6",
|
||||||
"@vaadin/vaadin-combo-box": "^5.0.10",
|
"@vaadin/vaadin-combo-box": "^5.0.10",
|
||||||
"@vaadin/vaadin-date-picker": "^4.0.7",
|
"@vaadin/vaadin-date-picker": "^4.0.7",
|
||||||
|
"@vibrant/color": "^3.2.1-alpha.1",
|
||||||
|
"@vibrant/core": "^3.2.1-alpha.1",
|
||||||
|
"@vibrant/quantizer-mmcq": "^3.2.1-alpha.1",
|
||||||
"@vue/web-component-wrapper": "^1.2.0",
|
"@vue/web-component-wrapper": "^1.2.0",
|
||||||
"@webcomponents/webcomponentsjs": "^2.2.7",
|
"@webcomponents/webcomponentsjs": "^2.2.7",
|
||||||
"chart.js": "~2.8.0",
|
"chart.js": "~2.8.0",
|
||||||
@ -90,7 +93,6 @@
|
|||||||
"codemirror": "^5.49.0",
|
"codemirror": "^5.49.0",
|
||||||
"comlink": "^4.3.0",
|
"comlink": "^4.3.0",
|
||||||
"core-js": "^3.6.5",
|
"core-js": "^3.6.5",
|
||||||
"cpx": "^1.5.0",
|
|
||||||
"cropperjs": "^1.5.7",
|
"cropperjs": "^1.5.7",
|
||||||
"deep-clone-simple": "^1.1.1",
|
"deep-clone-simple": "^1.1.1",
|
||||||
"deep-freeze": "^0.0.1",
|
"deep-freeze": "^0.0.1",
|
||||||
@ -110,7 +112,7 @@
|
|||||||
"marked": "^1.1.1",
|
"marked": "^1.1.1",
|
||||||
"mdn-polyfills": "^5.16.0",
|
"mdn-polyfills": "^5.16.0",
|
||||||
"memoize-one": "^5.0.2",
|
"memoize-one": "^5.0.2",
|
||||||
"node-vibrant": "^3.1.6",
|
"node-vibrant": "3.2.1-alpha.1",
|
||||||
"proxy-polyfill": "^0.3.1",
|
"proxy-polyfill": "^0.3.1",
|
||||||
"punycode": "^2.1.1",
|
"punycode": "^2.1.1",
|
||||||
"qrcode": "^1.4.4",
|
"qrcode": "^1.4.4",
|
||||||
@ -121,6 +123,8 @@
|
|||||||
"superstruct": "^0.10.12",
|
"superstruct": "^0.10.12",
|
||||||
"tinykeys": "^1.1.1",
|
"tinykeys": "^1.1.1",
|
||||||
"unfetch": "^4.1.0",
|
"unfetch": "^4.1.0",
|
||||||
|
"vis-data": "^7.1.1",
|
||||||
|
"vis-network": "^8.5.4",
|
||||||
"vue": "^2.6.11",
|
"vue": "^2.6.11",
|
||||||
"vue2-daterange-picker": "^0.5.1",
|
"vue2-daterange-picker": "^0.5.1",
|
||||||
"web-animations-js": "^2.3.2",
|
"web-animations-js": "^2.3.2",
|
||||||
@ -142,6 +146,9 @@
|
|||||||
"@babel/plugin-syntax-import-meta": "^7.10.4",
|
"@babel/plugin-syntax-import-meta": "^7.10.4",
|
||||||
"@babel/preset-env": "^7.11.5",
|
"@babel/preset-env": "^7.11.5",
|
||||||
"@babel/preset-typescript": "^7.10.4",
|
"@babel/preset-typescript": "^7.10.4",
|
||||||
|
"@koa/cors": "^3.1.0",
|
||||||
|
"@open-wc/dev-server-hmr": "^0.0.2",
|
||||||
|
"@rollup/plugin-babel": "^5.2.1",
|
||||||
"@rollup/plugin-commonjs": "^11.1.0",
|
"@rollup/plugin-commonjs": "^11.1.0",
|
||||||
"@rollup/plugin-json": "^4.0.3",
|
"@rollup/plugin-json": "^4.0.3",
|
||||||
"@rollup/plugin-node-resolve": "^7.1.3",
|
"@rollup/plugin-node-resolve": "^7.1.3",
|
||||||
@ -160,8 +167,11 @@
|
|||||||
"@types/webspeechapi": "^0.0.29",
|
"@types/webspeechapi": "^0.0.29",
|
||||||
"@typescript-eslint/eslint-plugin": "^4.4.0",
|
"@typescript-eslint/eslint-plugin": "^4.4.0",
|
||||||
"@typescript-eslint/parser": "^4.4.0",
|
"@typescript-eslint/parser": "^4.4.0",
|
||||||
|
"@web/dev-server": "^0.0.24",
|
||||||
|
"@web/dev-server-rollup": "^0.2.11",
|
||||||
"babel-loader": "^8.1.0",
|
"babel-loader": "^8.1.0",
|
||||||
"chai": "^4.2.0",
|
"chai": "^4.2.0",
|
||||||
|
"cpx": "^1.5.0",
|
||||||
"del": "^4.0.0",
|
"del": "^4.0.0",
|
||||||
"eslint": "^6.8.0",
|
"eslint": "^6.8.0",
|
||||||
"eslint-config-airbnb-typescript": "^7.2.1",
|
"eslint-config-airbnb-typescript": "^7.2.1",
|
||||||
@ -195,7 +205,6 @@
|
|||||||
"raw-loader": "^2.0.0",
|
"raw-loader": "^2.0.0",
|
||||||
"require-dir": "^1.2.0",
|
"require-dir": "^1.2.0",
|
||||||
"rollup": "^2.8.2",
|
"rollup": "^2.8.2",
|
||||||
"rollup-plugin-babel": "^4.4.0",
|
|
||||||
"rollup-plugin-string": "^3.0.0",
|
"rollup-plugin-string": "^3.0.0",
|
||||||
"rollup-plugin-terser": "^5.3.0",
|
"rollup-plugin-terser": "^5.3.0",
|
||||||
"rollup-plugin-visualizer": "^4.0.4",
|
"rollup-plugin-visualizer": "^4.0.4",
|
||||||
|
40
polymer.json
40
polymer.json
@ -1,40 +0,0 @@
|
|||||||
{
|
|
||||||
"entrypoint": "index.html",
|
|
||||||
"shell": "src/entrypoints/app.js",
|
|
||||||
"fragments": [
|
|
||||||
"src/panels/config/ha-panel-config.js",
|
|
||||||
"src/panels/dev-event/ha-panel-dev-event.js",
|
|
||||||
"src/panels/dev-info/ha-panel-dev-info.js",
|
|
||||||
"src/panels/dev-mqtt/ha-panel-dev-mqtt.js",
|
|
||||||
"src/panels/dev-service/ha-panel-dev-service.js",
|
|
||||||
"src/panels/dev-state/ha-panel-dev-state.js",
|
|
||||||
"src/panels/dev-template/ha-panel-dev-template.js",
|
|
||||||
"src/panels/history/ha-panel-history.js",
|
|
||||||
"src/panels/iframe/ha-panel-iframe.js",
|
|
||||||
"src/panels/logbook/ha-panel-logbook.js",
|
|
||||||
"src/panels/map/ha-panel-map.js",
|
|
||||||
"src/panels/mailbox/ha-panel-mailbox.js",
|
|
||||||
"hassio/src/entrypoint.js"
|
|
||||||
],
|
|
||||||
"sources": ["src/**/*", "!src/translations/*"],
|
|
||||||
"lint": {
|
|
||||||
"rules": ["polymer-3"],
|
|
||||||
"ignoreWarnings": ["could-not-resolve-reference", "could-not-load"],
|
|
||||||
"filesToIgnore": [
|
|
||||||
"**/*.html",
|
|
||||||
"**/src/panels/config/js/**/*.js",
|
|
||||||
"**/ha-iconset-svg.js",
|
|
||||||
"**/ha-script-editor.js",
|
|
||||||
"**/ha-automation-editor.js",
|
|
||||||
"**/ha-big-calendar.js"
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"builds": [
|
|
||||||
{
|
|
||||||
"preset": "es5-bundled"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"preset": "es6-bundled"
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
Binary file not shown.
Before Width: | Height: | Size: 30 KiB |
55
script/core
Executable file
55
script/core
Executable file
@ -0,0 +1,55 @@
|
|||||||
|
#!/bin/sh
|
||||||
|
# Helper to start Home Assistant Core inside the devcontainer
|
||||||
|
|
||||||
|
# Stop on errors
|
||||||
|
set -e
|
||||||
|
|
||||||
|
if [ -z "${DEVCONTAINER}" ]; then
|
||||||
|
echo "This task should only run inside a devcontainer, for local install HA Core in a venv."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -z "${CODESPACES}" ]; then
|
||||||
|
WORKSPACE="/root/workspace/frontend"
|
||||||
|
else
|
||||||
|
WORKSPACE="/workspaces/frontend"
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ -z $(which hass) ]; then
|
||||||
|
echo "Installing Home Asstant core from dev."
|
||||||
|
python3 -m pip install --upgrade \
|
||||||
|
colorlog \
|
||||||
|
git+git://github.com/home-assistant/home-assistant.git@dev
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ ! -d "${WORKSPACE}/config" ]; then
|
||||||
|
echo "Creating default configuration."
|
||||||
|
mkdir -p "${WORKSPACE}/config";
|
||||||
|
hass --script ensure_config -c config
|
||||||
|
echo "demo:
|
||||||
|
|
||||||
|
logger:
|
||||||
|
default: info
|
||||||
|
logs:
|
||||||
|
homeassistant.components.frontend: debug
|
||||||
|
" >> "${WORKSPACE}/config/configuration.yaml"
|
||||||
|
|
||||||
|
if [ ! -z "${HASSIO}" ]; then
|
||||||
|
echo "
|
||||||
|
# frontend:
|
||||||
|
# development_repo: ${WORKSPACE}
|
||||||
|
|
||||||
|
hassio:
|
||||||
|
development_repo: ${WORKSPACE}" >> "${WORKSPACE}/config/configuration.yaml"
|
||||||
|
else
|
||||||
|
echo "
|
||||||
|
frontend:
|
||||||
|
development_repo: ${WORKSPACE}
|
||||||
|
|
||||||
|
# hassio:
|
||||||
|
# development_repo: ${WORKSPACE}" >> "${WORKSPACE}/config/configuration.yaml"
|
||||||
|
fi
|
||||||
|
|
||||||
|
fi
|
||||||
|
|
||||||
|
hass -c "${WORKSPACE}/config"
|
2
setup.py
2
setup.py
@ -2,7 +2,7 @@ from setuptools import setup, find_packages
|
|||||||
|
|
||||||
setup(
|
setup(
|
||||||
name="home-assistant-frontend",
|
name="home-assistant-frontend",
|
||||||
version="20201111.2",
|
version="20201126.0",
|
||||||
description="The Home Assistant frontend",
|
description="The Home Assistant frontend",
|
||||||
url="https://github.com/home-assistant/home-assistant-polymer",
|
url="https://github.com/home-assistant/home-assistant-polymer",
|
||||||
author="The Home Assistant Authors",
|
author="The Home Assistant Authors",
|
||||||
|
@ -18,7 +18,7 @@ import "./ha-auth-flow";
|
|||||||
import { extractSearchParamsObject } from "../common/url/search-params";
|
import { extractSearchParamsObject } from "../common/url/search-params";
|
||||||
import punycode from "punycode";
|
import punycode from "punycode";
|
||||||
|
|
||||||
import(/* webpackChunkName: "pick-auth-provider" */ "./ha-pick-auth-provider");
|
import("./ha-pick-auth-provider");
|
||||||
|
|
||||||
class HaAuthorize extends litLocalizeLiteMixin(LitElement) {
|
class HaAuthorize extends litLocalizeLiteMixin(LitElement) {
|
||||||
@property() public clientId?: string;
|
@property() public clientId?: string;
|
||||||
|
@ -22,3 +22,8 @@ export const rgbContrast = (
|
|||||||
|
|
||||||
return (lum2 + 0.05) / (lum1 + 0.05);
|
return (lum2 + 0.05) / (lum1 + 0.05);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const getRGBContrastRatio = (
|
||||||
|
rgb1: [number, number, number],
|
||||||
|
rgb2: [number, number, number]
|
||||||
|
) => Math.round((rgbContrast(rgb1, rgb2) + Number.EPSILON) * 100) / 100;
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
|
import { Theme } from "../../data/ws-themes";
|
||||||
import { darkStyles, derivedStyles } from "../../resources/styles";
|
import { darkStyles, derivedStyles } from "../../resources/styles";
|
||||||
import { HomeAssistant, Theme } from "../../types";
|
import type { HomeAssistant } from "../../types";
|
||||||
import {
|
import {
|
||||||
hex2rgb,
|
hex2rgb,
|
||||||
lab2hex,
|
lab2hex,
|
||||||
@ -13,10 +14,10 @@ import { rgbContrast } from "../color/rgb";
|
|||||||
|
|
||||||
interface ProcessedTheme {
|
interface ProcessedTheme {
|
||||||
keys: { [key: string]: "" };
|
keys: { [key: string]: "" };
|
||||||
styles: { [key: string]: string };
|
styles: Record<string, string>;
|
||||||
}
|
}
|
||||||
|
|
||||||
let PROCESSED_THEMES: { [key: string]: ProcessedTheme } = {};
|
let PROCESSED_THEMES: Record<string, ProcessedTheme> = {};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Apply a theme to an element by setting the CSS variables on it.
|
* Apply a theme to an element by setting the CSS variables on it.
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import { directive, NodePart, Part } from "lit-html";
|
import { directive, NodePart, Part } from "lit-html";
|
||||||
|
|
||||||
export const dynamicElement = directive(
|
export const dynamicElement = directive(
|
||||||
(tag: string, properties?: { [key: string]: any }) => (part: Part): void => {
|
(tag: string, properties?: Record<string, any>) => (part: Part): void => {
|
||||||
if (!(part instanceof NodePart)) {
|
if (!(part instanceof NodePart)) {
|
||||||
throw new Error(
|
throw new Error(
|
||||||
"dynamicElementDirective can only be used in content bindings"
|
"dynamicElementDirective can only be used in content bindings"
|
||||||
|
@ -13,13 +13,12 @@ export const setupLeafletMap = async (
|
|||||||
throw new Error("Cannot setup Leaflet map on disconnected element");
|
throw new Error("Cannot setup Leaflet map on disconnected element");
|
||||||
}
|
}
|
||||||
// eslint-disable-next-line
|
// eslint-disable-next-line
|
||||||
const Leaflet = ((await import(
|
const Leaflet = ((await import("leaflet")) as any)
|
||||||
/* webpackChunkName: "leaflet" */ "leaflet"
|
.default as LeafletModuleType;
|
||||||
)) as any).default as LeafletModuleType;
|
|
||||||
Leaflet.Icon.Default.imagePath = "/static/images/leaflet/images/";
|
Leaflet.Icon.Default.imagePath = "/static/images/leaflet/images/";
|
||||||
|
|
||||||
if (draw) {
|
if (draw) {
|
||||||
await import(/* webpackChunkName: "leaflet-draw" */ "leaflet-draw");
|
await import("leaflet-draw");
|
||||||
}
|
}
|
||||||
|
|
||||||
const map = Leaflet.map(mapElement);
|
const map = Leaflet.map(mapElement);
|
||||||
|
@ -5,7 +5,7 @@ import { formatDateTime } from "../datetime/format_date_time";
|
|||||||
import { formatTime } from "../datetime/format_time";
|
import { formatTime } from "../datetime/format_time";
|
||||||
import { LocalizeFunc } from "../translations/localize";
|
import { LocalizeFunc } from "../translations/localize";
|
||||||
import { computeStateDomain } from "./compute_state_domain";
|
import { computeStateDomain } from "./compute_state_domain";
|
||||||
import { numberFormat } from "../string/number-format";
|
import { formatNumber } from "../string/format_number";
|
||||||
|
|
||||||
export const computeStateDisplay = (
|
export const computeStateDisplay = (
|
||||||
localize: LocalizeFunc,
|
localize: LocalizeFunc,
|
||||||
@ -20,7 +20,7 @@ export const computeStateDisplay = (
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (stateObj.attributes.unit_of_measurement) {
|
if (stateObj.attributes.unit_of_measurement) {
|
||||||
return `${numberFormat(compareState, language)} ${
|
return `${formatNumber(compareState, language)} ${
|
||||||
stateObj.attributes.unit_of_measurement
|
stateObj.attributes.unit_of_measurement
|
||||||
}`;
|
}`;
|
||||||
}
|
}
|
||||||
|
@ -78,3 +78,25 @@ export const coverIcon = (state?: string, stateObj?: HassEntity): string => {
|
|||||||
return "hass:window-open";
|
return "hass:window-open";
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
export const computeOpenIcon = (stateObj: HassEntity): string => {
|
||||||
|
switch (stateObj.attributes.device_class) {
|
||||||
|
case "awning":
|
||||||
|
case "door":
|
||||||
|
case "gate":
|
||||||
|
return "hass:arrow-expand-horizontal";
|
||||||
|
default:
|
||||||
|
return "hass:arrow-up";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const computeCloseIcon = (stateObj: HassEntity): string => {
|
||||||
|
switch (stateObj.attributes.device_class) {
|
||||||
|
case "awning":
|
||||||
|
case "door":
|
||||||
|
case "gate":
|
||||||
|
return "hass:arrow-collapse-horizontal";
|
||||||
|
default:
|
||||||
|
return "hass:arrow-down";
|
||||||
|
}
|
||||||
|
};
|
||||||
|
@ -77,6 +77,11 @@ export const domainIcon = (
|
|||||||
return "hass:calendar";
|
return "hass:calendar";
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case "sun":
|
||||||
|
return stateObj?.state === "above_horizon"
|
||||||
|
? FIXED_DOMAIN_ICONS[domain]
|
||||||
|
: "hass:weather-night";
|
||||||
}
|
}
|
||||||
|
|
||||||
if (domain in FIXED_DOMAIN_ICONS) {
|
if (domain in FIXED_DOMAIN_ICONS) {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { HassEntities } from "home-assistant-js-websocket";
|
import { HassEntities } from "home-assistant-js-websocket";
|
||||||
import { GroupEntity } from "../../types";
|
import type { GroupEntity } from "../../data/group";
|
||||||
import { DEFAULT_VIEW_ENTITY_ID } from "../const";
|
import { DEFAULT_VIEW_ENTITY_ID } from "../const";
|
||||||
|
|
||||||
// Return an ordered array of available views
|
// Return an ordered array of available views
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { HassEntities } from "home-assistant-js-websocket";
|
import { HassEntities } from "home-assistant-js-websocket";
|
||||||
import { GroupEntity } from "../../types";
|
import { GroupEntity } from "../../data/group";
|
||||||
|
|
||||||
export const getGroupEntities = (
|
export const getGroupEntities = (
|
||||||
entities: HassEntities,
|
entities: HassEntities,
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { HassEntities } from "home-assistant-js-websocket";
|
import { HassEntities } from "home-assistant-js-websocket";
|
||||||
import { GroupEntity } from "../../types";
|
import { GroupEntity } from "../../data/group";
|
||||||
import { computeDomain } from "./compute_domain";
|
import { computeDomain } from "./compute_domain";
|
||||||
import { getGroupEntities } from "./get_group_entities";
|
import { getGroupEntities } from "./get_group_entities";
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { HassEntities } from "home-assistant-js-websocket";
|
import { HassEntities } from "home-assistant-js-websocket";
|
||||||
import { GroupEntity } from "../../types";
|
import { GroupEntity } from "../../data/group";
|
||||||
import { computeDomain } from "./compute_domain";
|
import { computeDomain } from "./compute_domain";
|
||||||
|
|
||||||
// Split a collection into a list of groups and a 'rest' list of ungrouped
|
// Split a collection into a list of groups and a 'rest' list of ungrouped
|
||||||
|
132
src/common/image/extract_color.ts
Normal file
132
src/common/image/extract_color.ts
Normal file
@ -0,0 +1,132 @@
|
|||||||
|
import Vibrant from "node-vibrant/lib/browser";
|
||||||
|
import MMCQ from "@vibrant/quantizer-mmcq";
|
||||||
|
import { BasicPipeline } from "@vibrant/core/lib/pipeline";
|
||||||
|
import { Swatch, Vec3 } from "@vibrant/color";
|
||||||
|
import { getRGBContrastRatio } from "../color/rgb";
|
||||||
|
|
||||||
|
const CONTRAST_RATIO = 4.5;
|
||||||
|
|
||||||
|
// How much the total diff between 2 RGB colors can be
|
||||||
|
// to be considered similar.
|
||||||
|
const COLOR_SIMILARITY_THRESHOLD = 150;
|
||||||
|
|
||||||
|
// For debug purposes, is being tree shaken.
|
||||||
|
const DEBUG_COLOR = __DEV__ && false;
|
||||||
|
|
||||||
|
const logColor = (
|
||||||
|
color: Swatch,
|
||||||
|
label = `${color.hex} - ${color.population}`
|
||||||
|
) =>
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.log(
|
||||||
|
`%c${label}`,
|
||||||
|
`color: ${color.bodyTextColor}; background-color: ${color.hex}`
|
||||||
|
);
|
||||||
|
|
||||||
|
const customGenerator = (colors: Swatch[]) => {
|
||||||
|
colors.sort((colorA, colorB) => colorB.population - colorA.population);
|
||||||
|
|
||||||
|
const backgroundColor = colors[0];
|
||||||
|
let foregroundColor: Vec3 | undefined;
|
||||||
|
|
||||||
|
const contrastRatios = new Map<string, number>();
|
||||||
|
const approvedContrastRatio = (hex: string, rgb: Swatch["rgb"]) => {
|
||||||
|
if (!contrastRatios.has(hex)) {
|
||||||
|
contrastRatios.set(hex, getRGBContrastRatio(backgroundColor.rgb, rgb));
|
||||||
|
}
|
||||||
|
|
||||||
|
return contrastRatios.get(hex)! > CONTRAST_RATIO;
|
||||||
|
};
|
||||||
|
|
||||||
|
// We take each next color and find one that has better contrast.
|
||||||
|
for (let i = 1; i < colors.length && foregroundColor === undefined; i++) {
|
||||||
|
// If this color matches, score, take it.
|
||||||
|
if (approvedContrastRatio(colors[i].hex, colors[i].rgb)) {
|
||||||
|
if (DEBUG_COLOR) {
|
||||||
|
logColor(colors[i], "PICKED");
|
||||||
|
}
|
||||||
|
foregroundColor = colors[i].rgb;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// This color has the wrong contrast ratio, but it is the right color.
|
||||||
|
// Let's find similar colors that might have the right contrast ratio
|
||||||
|
|
||||||
|
const currentColor = colors[i];
|
||||||
|
if (DEBUG_COLOR) {
|
||||||
|
logColor(colors[i], "Finding similar color with better contrast");
|
||||||
|
}
|
||||||
|
|
||||||
|
for (let j = i + 1; j < colors.length; j++) {
|
||||||
|
const compareColor = colors[j];
|
||||||
|
|
||||||
|
// difference. 0 is same, 765 max difference
|
||||||
|
const diffScore =
|
||||||
|
Math.abs(currentColor.rgb[0] - compareColor.rgb[0]) +
|
||||||
|
Math.abs(currentColor.rgb[1] - compareColor.rgb[1]) +
|
||||||
|
Math.abs(currentColor.rgb[2] - compareColor.rgb[2]);
|
||||||
|
|
||||||
|
if (DEBUG_COLOR) {
|
||||||
|
logColor(colors[j], `${colors[j].hex} - ${diffScore}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (diffScore > COLOR_SIMILARITY_THRESHOLD) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (approvedContrastRatio(compareColor.hex, compareColor.rgb)) {
|
||||||
|
if (DEBUG_COLOR) {
|
||||||
|
logColor(compareColor, "PICKED");
|
||||||
|
}
|
||||||
|
foregroundColor = compareColor.rgb;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (foregroundColor === undefined) {
|
||||||
|
foregroundColor =
|
||||||
|
// @ts-expect-error
|
||||||
|
backgroundColor.getYiq() < 200 ? [255, 255, 255] : [0, 0, 0];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (DEBUG_COLOR) {
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.log();
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.log(
|
||||||
|
"%cPicked colors",
|
||||||
|
`color: ${foregroundColor}; background-color: ${backgroundColor.hex}; font-weight: bold; padding: 16px;`
|
||||||
|
);
|
||||||
|
colors.forEach((color) => logColor(color));
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.log();
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
foreground: new Swatch(foregroundColor, 0),
|
||||||
|
background: backgroundColor,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
Vibrant.use(
|
||||||
|
new BasicPipeline().filter
|
||||||
|
.register(
|
||||||
|
"default",
|
||||||
|
(r: number, g: number, b: number, a: number) =>
|
||||||
|
a >= 125 && !(r > 250 && g > 250 && b > 250)
|
||||||
|
)
|
||||||
|
.quantizer.register("mmcq", MMCQ)
|
||||||
|
// Our generator has different output
|
||||||
|
// @ts-expect-error
|
||||||
|
.generator.register("default", customGenerator)
|
||||||
|
);
|
||||||
|
|
||||||
|
export const extractColors = (url: string, downsampleColors = 16) =>
|
||||||
|
new Vibrant(url, {
|
||||||
|
colorCount: downsampleColors,
|
||||||
|
})
|
||||||
|
.getPalette()
|
||||||
|
.then(({ foreground, background }) => {
|
||||||
|
return { background: background!, foreground: foreground! };
|
||||||
|
});
|
54
src/common/string/format_number.ts
Normal file
54
src/common/string/format_number.ts
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
/**
|
||||||
|
* Formats a number based on the specified language with thousands separator(s) and decimal character for better legibility.
|
||||||
|
*
|
||||||
|
* @param num The number to format
|
||||||
|
* @param language The language to use when formatting the number
|
||||||
|
*/
|
||||||
|
export const formatNumber = (
|
||||||
|
num: string | number,
|
||||||
|
language: string,
|
||||||
|
options?: Intl.NumberFormatOptions
|
||||||
|
): string => {
|
||||||
|
// Polyfill for Number.isNaN, which is more reliable than the global isNaN()
|
||||||
|
Number.isNaN =
|
||||||
|
Number.isNaN ||
|
||||||
|
function isNaN(input) {
|
||||||
|
return typeof input === "number" && isNaN(input);
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!Number.isNaN(Number(num)) && Intl) {
|
||||||
|
return new Intl.NumberFormat(
|
||||||
|
language,
|
||||||
|
getDefaultFormatOptions(num, options)
|
||||||
|
).format(Number(num));
|
||||||
|
}
|
||||||
|
return num.toString();
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Generates default options for Intl.NumberFormat
|
||||||
|
* @param num The number to be formatted
|
||||||
|
* @param options The Intl.NumberFormatOptions that should be included in the returned options
|
||||||
|
*/
|
||||||
|
const getDefaultFormatOptions = (
|
||||||
|
num: string | number,
|
||||||
|
options?: Intl.NumberFormatOptions
|
||||||
|
): Intl.NumberFormatOptions => {
|
||||||
|
const defaultOptions: Intl.NumberFormatOptions = options || {};
|
||||||
|
|
||||||
|
if (typeof num !== "string") {
|
||||||
|
return defaultOptions;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Keep decimal trailing zeros if they are present in a string numeric value
|
||||||
|
if (
|
||||||
|
!options ||
|
||||||
|
(!options.minimumFractionDigits && !options.maximumFractionDigits)
|
||||||
|
) {
|
||||||
|
const digits = num.indexOf(".") > -1 ? num.split(".")[1].length : 0;
|
||||||
|
defaultOptions.minimumFractionDigits = digits;
|
||||||
|
defaultOptions.maximumFractionDigits = digits;
|
||||||
|
}
|
||||||
|
|
||||||
|
return defaultOptions;
|
||||||
|
};
|
@ -1,22 +0,0 @@
|
|||||||
/**
|
|
||||||
* Formats a number based on the specified language with thousands separator(s) and decimal character for better legibility.
|
|
||||||
*
|
|
||||||
* @param num The number to format
|
|
||||||
* @param language The language to use when formatting the number
|
|
||||||
*/
|
|
||||||
export const numberFormat = (
|
|
||||||
num: string | number,
|
|
||||||
language: string
|
|
||||||
): string => {
|
|
||||||
// Polyfill for Number.isNaN, which is more reliable that the global isNaN()
|
|
||||||
Number.isNaN =
|
|
||||||
Number.isNaN ||
|
|
||||||
function isNaN(input) {
|
|
||||||
return typeof input === "number" && isNaN(input);
|
|
||||||
};
|
|
||||||
|
|
||||||
if (!Number.isNaN(Number(num)) && Intl) {
|
|
||||||
return new Intl.NumberFormat(language).format(Number(num));
|
|
||||||
}
|
|
||||||
return num.toString();
|
|
||||||
};
|
|
@ -102,7 +102,7 @@ export const computeLocalize = async (
|
|||||||
export const localizeKey = (
|
export const localizeKey = (
|
||||||
localize: LocalizeFunc,
|
localize: LocalizeFunc,
|
||||||
key: string,
|
key: string,
|
||||||
placeholders?: { [key: string]: string }
|
placeholders?: Record<string, string>
|
||||||
) => {
|
) => {
|
||||||
const args: [string, ...string[]] = [key];
|
const args: [string, ...string[]] = [key];
|
||||||
if (placeholders) {
|
if (placeholders) {
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
export const extractSearchParamsObject = (): { [key: string]: string } => {
|
export const extractSearchParamsObject = (): Record<string, string> => {
|
||||||
const query = {};
|
const query = {};
|
||||||
const searchParams = new URLSearchParams(location.search);
|
const searchParams = new URLSearchParams(location.search);
|
||||||
for (const [key, value] of searchParams.entries()) {
|
for (const [key, value] of searchParams.entries()) {
|
||||||
|
@ -42,6 +42,10 @@ interface Device {
|
|||||||
id: string;
|
id: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export type HaDevicePickerDeviceFilterFunc = (
|
||||||
|
device: DeviceRegistryEntry
|
||||||
|
) => boolean;
|
||||||
|
|
||||||
const rowRenderer = (root: HTMLElement, _owner, model: { item: Device }) => {
|
const rowRenderer = (root: HTMLElement, _owner, model: { item: Device }) => {
|
||||||
if (!root.firstElementChild) {
|
if (!root.firstElementChild) {
|
||||||
root.innerHTML = `
|
root.innerHTML = `
|
||||||
@ -102,6 +106,8 @@ export class HaDevicePicker extends SubscribeMixin(LitElement) {
|
|||||||
@property({ type: Array, attribute: "include-device-classes" })
|
@property({ type: Array, attribute: "include-device-classes" })
|
||||||
public includeDeviceClasses?: string[];
|
public includeDeviceClasses?: string[];
|
||||||
|
|
||||||
|
@property() public deviceFilter?: HaDevicePickerDeviceFilterFunc;
|
||||||
|
|
||||||
@property({ type: Boolean })
|
@property({ type: Boolean })
|
||||||
private _opened?: boolean;
|
private _opened?: boolean;
|
||||||
|
|
||||||
@ -112,7 +118,8 @@ export class HaDevicePicker extends SubscribeMixin(LitElement) {
|
|||||||
entities: EntityRegistryEntry[],
|
entities: EntityRegistryEntry[],
|
||||||
includeDomains: this["includeDomains"],
|
includeDomains: this["includeDomains"],
|
||||||
excludeDomains: this["excludeDomains"],
|
excludeDomains: this["excludeDomains"],
|
||||||
includeDeviceClasses: this["includeDeviceClasses"]
|
includeDeviceClasses: this["includeDeviceClasses"],
|
||||||
|
deviceFilter: this["deviceFilter"]
|
||||||
): Device[] => {
|
): Device[] => {
|
||||||
if (!devices.length) {
|
if (!devices.length) {
|
||||||
return [];
|
return [];
|
||||||
@ -180,6 +187,14 @@ export class HaDevicePicker extends SubscribeMixin(LitElement) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (deviceFilter) {
|
||||||
|
inputDevices = inputDevices.filter(
|
||||||
|
(device) =>
|
||||||
|
// We always want to include the device of the current value
|
||||||
|
device.id === this.value || deviceFilter!(device)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
const outputDevices = inputDevices.map((device) => {
|
const outputDevices = inputDevices.map((device) => {
|
||||||
return {
|
return {
|
||||||
id: device.id,
|
id: device.id,
|
||||||
@ -224,7 +239,8 @@ export class HaDevicePicker extends SubscribeMixin(LitElement) {
|
|||||||
this.entities,
|
this.entities,
|
||||||
this.includeDomains,
|
this.includeDomains,
|
||||||
this.excludeDomains,
|
this.excludeDomains,
|
||||||
this.includeDeviceClasses
|
this.includeDeviceClasses,
|
||||||
|
this.deviceFilter
|
||||||
);
|
);
|
||||||
return html`
|
return html`
|
||||||
<vaadin-combo-box-light
|
<vaadin-combo-box-light
|
||||||
|
@ -230,9 +230,7 @@ class HaChartBase extends mixinBehaviors(
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (scriptsLoaded === null) {
|
if (scriptsLoaded === null) {
|
||||||
scriptsLoaded = import(
|
scriptsLoaded = import("../../resources/ha-chart-scripts.js");
|
||||||
/* webpackChunkName: "load_chart" */ "../../resources/ha-chart-scripts.js"
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
scriptsLoaded.then((ChartModule) => {
|
scriptsLoaded.then((ChartModule) => {
|
||||||
this.ChartClass = ChartModule.default;
|
this.ChartClass = ChartModule.default;
|
||||||
|
@ -21,6 +21,7 @@ import { timerTimeRemaining } from "../../common/entity/timer_time_remaining";
|
|||||||
import { HomeAssistant } from "../../types";
|
import { HomeAssistant } from "../../types";
|
||||||
import "../ha-label-badge";
|
import "../ha-label-badge";
|
||||||
import { UNAVAILABLE, UNKNOWN } from "../../data/entity";
|
import { UNAVAILABLE, UNKNOWN } from "../../data/entity";
|
||||||
|
import { formatNumber } from "../../common/string/format_number";
|
||||||
|
|
||||||
@customElement("ha-state-label-badge")
|
@customElement("ha-state-label-badge")
|
||||||
export class HaStateLabelBadge extends LitElement {
|
export class HaStateLabelBadge extends LitElement {
|
||||||
@ -115,7 +116,7 @@ export class HaStateLabelBadge extends LitElement {
|
|||||||
: state.state === UNKNOWN
|
: state.state === UNKNOWN
|
||||||
? "-"
|
? "-"
|
||||||
: state.attributes.unit_of_measurement
|
: state.attributes.unit_of_measurement
|
||||||
? state.state
|
? formatNumber(state.state, this.hass!.language)
|
||||||
: computeStateDisplay(
|
: computeStateDisplay(
|
||||||
this.hass!.localize,
|
this.hass!.localize,
|
||||||
state,
|
state,
|
||||||
@ -154,11 +155,8 @@ export class HaStateLabelBadge extends LitElement {
|
|||||||
case "device_tracker":
|
case "device_tracker":
|
||||||
case "updater":
|
case "updater":
|
||||||
case "person":
|
case "person":
|
||||||
return stateIcon(state);
|
|
||||||
case "sun":
|
case "sun":
|
||||||
return state.state === "above_horizon"
|
return stateIcon(state);
|
||||||
? domainIcon(domain)
|
|
||||||
: "hass:brightness-3";
|
|
||||||
case "timer":
|
case "timer":
|
||||||
return state.state === "active"
|
return state.state === "active"
|
||||||
? "hass:timer-outline"
|
? "hass:timer-outline"
|
||||||
|
@ -1,152 +0,0 @@
|
|||||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
|
|
||||||
import "@polymer/paper-tooltip/paper-tooltip";
|
|
||||||
/* eslint-plugin-disable lit */
|
|
||||||
import LocalizeMixin from "../../mixins/localize-mixin";
|
|
||||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
|
|
||||||
import { computeStateName } from "../../common/entity/compute_state_name";
|
|
||||||
import { computeRTL } from "../../common/util/compute_rtl";
|
|
||||||
import "../ha-relative-time";
|
|
||||||
import "./state-badge";
|
|
||||||
|
|
||||||
class StateInfo extends LocalizeMixin(PolymerElement) {
|
|
||||||
static get template() {
|
|
||||||
return html`
|
|
||||||
${this.styleTemplate} ${this.stateBadgeTemplate} ${this.infoTemplate}
|
|
||||||
`;
|
|
||||||
}
|
|
||||||
|
|
||||||
static get styleTemplate() {
|
|
||||||
return html`
|
|
||||||
<style>
|
|
||||||
:host {
|
|
||||||
@apply --paper-font-body1;
|
|
||||||
min-width: 120px;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
state-badge {
|
|
||||||
float: left;
|
|
||||||
}
|
|
||||||
|
|
||||||
:host([rtl]) state-badge {
|
|
||||||
float: right;
|
|
||||||
}
|
|
||||||
|
|
||||||
.info {
|
|
||||||
margin-left: 56px;
|
|
||||||
}
|
|
||||||
|
|
||||||
:host([rtl]) .info {
|
|
||||||
margin-right: 56px;
|
|
||||||
margin-left: 0;
|
|
||||||
text-align: right;
|
|
||||||
}
|
|
||||||
|
|
||||||
.name {
|
|
||||||
@apply --paper-font-common-nowrap;
|
|
||||||
color: var(--primary-text-color);
|
|
||||||
line-height: 40px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.name[in-dialog],
|
|
||||||
:host([secondary-line]) .name {
|
|
||||||
line-height: 20px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.time-ago,
|
|
||||||
.extra-info,
|
|
||||||
.extra-info > * {
|
|
||||||
@apply --paper-font-common-nowrap;
|
|
||||||
color: var(--secondary-text-color);
|
|
||||||
}
|
|
||||||
|
|
||||||
.row {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: row;
|
|
||||||
flex-wrap: no-wrap;
|
|
||||||
width: 100%;
|
|
||||||
justify-content: space-between;
|
|
||||||
margin: 0 2px 4px 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.row:last-child {
|
|
||||||
margin-bottom: 0px;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
`;
|
|
||||||
}
|
|
||||||
|
|
||||||
static get stateBadgeTemplate() {
|
|
||||||
return html` <state-badge state-obj="[[stateObj]]"></state-badge> `;
|
|
||||||
}
|
|
||||||
|
|
||||||
static get infoTemplate() {
|
|
||||||
return html`
|
|
||||||
<div class="info">
|
|
||||||
<div class="name" in-dialog$="[[inDialog]]">
|
|
||||||
[[computeStateName(stateObj)]]
|
|
||||||
</div>
|
|
||||||
<template is="dom-if" if="[[inDialog]]">
|
|
||||||
<div class="time-ago">
|
|
||||||
<ha-relative-time
|
|
||||||
id="last_changed"
|
|
||||||
hass="[[hass]]"
|
|
||||||
datetime="[[stateObj.last_changed]]"
|
|
||||||
></ha-relative-time>
|
|
||||||
<paper-tooltip animation-delay="0" for="last_changed">
|
|
||||||
<div>
|
|
||||||
<div class="row">
|
|
||||||
<span class="column-name">
|
|
||||||
[[localize('ui.dialogs.more_info_control.last_changed')]]:
|
|
||||||
</span>
|
|
||||||
<ha-relative-time
|
|
||||||
hass="[[hass]]"
|
|
||||||
datetime="[[stateObj.last_changed]]"
|
|
||||||
></ha-relative-time>
|
|
||||||
</div>
|
|
||||||
<div class="row">
|
|
||||||
<span>
|
|
||||||
[[localize('ui.dialogs.more_info_control.last_updated')]]:
|
|
||||||
</span>
|
|
||||||
<ha-relative-time
|
|
||||||
hass="[[hass]]"
|
|
||||||
datetime="[[stateObj.last_updated]]"
|
|
||||||
></ha-relative-time>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</paper-tooltip>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
<template is="dom-if" if="[[!inDialog]]">
|
|
||||||
<div class="extra-info"><slot> </slot></div>
|
|
||||||
</template>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
}
|
|
||||||
|
|
||||||
static get properties() {
|
|
||||||
return {
|
|
||||||
hass: Object,
|
|
||||||
stateObj: Object,
|
|
||||||
inDialog: {
|
|
||||||
type: Boolean,
|
|
||||||
value: () => false,
|
|
||||||
},
|
|
||||||
rtl: {
|
|
||||||
type: Boolean,
|
|
||||||
reflectToAttribute: true,
|
|
||||||
computed: "computeRTL(hass)",
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
computeStateName(stateObj) {
|
|
||||||
return computeStateName(stateObj);
|
|
||||||
}
|
|
||||||
|
|
||||||
computeRTL(hass) {
|
|
||||||
return computeRTL(hass);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
customElements.define("state-info", StateInfo);
|
|
160
src/components/entity/state-info.ts
Normal file
160
src/components/entity/state-info.ts
Normal file
@ -0,0 +1,160 @@
|
|||||||
|
import "@polymer/paper-tooltip/paper-tooltip";
|
||||||
|
import {
|
||||||
|
css,
|
||||||
|
CSSResult,
|
||||||
|
customElement,
|
||||||
|
html,
|
||||||
|
LitElement,
|
||||||
|
property,
|
||||||
|
TemplateResult,
|
||||||
|
} from "lit-element";
|
||||||
|
import type { HassEntity } from "home-assistant-js-websocket";
|
||||||
|
|
||||||
|
import { computeStateName } from "../../common/entity/compute_state_name";
|
||||||
|
import { computeRTL } from "../../common/util/compute_rtl";
|
||||||
|
import type { HomeAssistant } from "../../types";
|
||||||
|
|
||||||
|
import "../ha-relative-time";
|
||||||
|
import "./state-badge";
|
||||||
|
|
||||||
|
@customElement("state-info")
|
||||||
|
class StateInfo extends LitElement {
|
||||||
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public stateObj?: HassEntity;
|
||||||
|
|
||||||
|
@property({ type: Boolean }) public inDialog = false;
|
||||||
|
|
||||||
|
// property used only in css
|
||||||
|
@property({ type: Boolean, reflect: true }) public rtl = false;
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
if (!this.hass || !this.stateObj) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
|
||||||
|
return html`<state-badge
|
||||||
|
.stateObj=${this.stateObj}
|
||||||
|
.stateColor=${true}
|
||||||
|
></state-badge>
|
||||||
|
<div class="info">
|
||||||
|
<div class="name" .inDialog=${this.inDialog}>
|
||||||
|
${computeStateName(this.stateObj)}
|
||||||
|
</div>
|
||||||
|
${this.inDialog
|
||||||
|
? html`<div class="time-ago">
|
||||||
|
<ha-relative-time
|
||||||
|
id="last_changed"
|
||||||
|
.hass=${this.hass}
|
||||||
|
.datetime=${this.stateObj.last_changed}
|
||||||
|
></ha-relative-time>
|
||||||
|
<paper-tooltip animation-delay="0" for="last_changed">
|
||||||
|
<div>
|
||||||
|
<div class="row">
|
||||||
|
<span class="column-name">
|
||||||
|
${this.hass.localize(
|
||||||
|
"ui.dialogs.more_info_control.last_changed"
|
||||||
|
)}:
|
||||||
|
</span>
|
||||||
|
<ha-relative-time
|
||||||
|
.hass=${this.hass}
|
||||||
|
.datetime=${this.stateObj.last_changed}
|
||||||
|
></ha-relative-time>
|
||||||
|
</div>
|
||||||
|
<div class="row">
|
||||||
|
<span>
|
||||||
|
${this.hass.localize(
|
||||||
|
"ui.dialogs.more_info_control.last_updated"
|
||||||
|
)}:
|
||||||
|
</span>
|
||||||
|
<ha-relative-time
|
||||||
|
.hass=${this.hass}
|
||||||
|
.datetime=${this.stateObj.last_updated}
|
||||||
|
></ha-relative-time>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</paper-tooltip>
|
||||||
|
</div>`
|
||||||
|
: html`<div class="extra-info"><slot> </slot></div>`}
|
||||||
|
</div>`;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected updated(changedProps) {
|
||||||
|
super.updated(changedProps);
|
||||||
|
if (!changedProps.has("hass")) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const oldHass = changedProps.get("hass") as HomeAssistant | undefined;
|
||||||
|
if (!oldHass || oldHass.language !== this.hass.language) {
|
||||||
|
this.rtl = computeRTL(this.hass);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles(): CSSResult {
|
||||||
|
return css`
|
||||||
|
:host {
|
||||||
|
min-width: 120px;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
|
||||||
|
state-badge {
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([rtl]) state-badge {
|
||||||
|
float: right;
|
||||||
|
}
|
||||||
|
|
||||||
|
.info {
|
||||||
|
margin-left: 56px;
|
||||||
|
}
|
||||||
|
|
||||||
|
:host([rtl]) .info {
|
||||||
|
margin-right: 56px;
|
||||||
|
margin-left: 0;
|
||||||
|
text-align: right;
|
||||||
|
}
|
||||||
|
|
||||||
|
.name {
|
||||||
|
color: var(--primary-text-color);
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
|
||||||
|
.name[in-dialog],
|
||||||
|
:host([secondary-line]) .name {
|
||||||
|
line-height: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.time-ago,
|
||||||
|
.extra-info,
|
||||||
|
.extra-info > * {
|
||||||
|
color: var(--secondary-text-color);
|
||||||
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
}
|
||||||
|
|
||||||
|
.row {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
flex-wrap: no-wrap;
|
||||||
|
width: 100%;
|
||||||
|
justify-content: space-between;
|
||||||
|
margin: 0 2px 4px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.row:last-child {
|
||||||
|
margin-bottom: 0px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"state-info": StateInfo;
|
||||||
|
}
|
||||||
|
}
|
@ -107,7 +107,7 @@ class HaAttributes extends LitElement {
|
|||||||
(!Array.isArray(value) && value instanceof Object)
|
(!Array.isArray(value) && value instanceof Object)
|
||||||
) {
|
) {
|
||||||
if (!jsYamlPromise) {
|
if (!jsYamlPromise) {
|
||||||
jsYamlPromise = import(/* webpackChunkName: "js-yaml" */ "js-yaml");
|
jsYamlPromise = import("js-yaml");
|
||||||
}
|
}
|
||||||
const yaml = jsYamlPromise.then((jsYaml) => jsYaml.safeDump(value));
|
const yaml = jsYamlPromise.then((jsYaml) => jsYaml.safeDump(value));
|
||||||
return html` <pre>${until(yaml, "")}</pre> `;
|
return html` <pre>${until(yaml, "")}</pre> `;
|
||||||
|
121
src/components/ha-blueprint-picker.ts
Normal file
121
src/components/ha-blueprint-picker.ts
Normal file
@ -0,0 +1,121 @@
|
|||||||
|
import "@polymer/paper-dropdown-menu/paper-dropdown-menu-light";
|
||||||
|
import "@polymer/paper-item/paper-item";
|
||||||
|
import "@polymer/paper-listbox/paper-listbox";
|
||||||
|
import {
|
||||||
|
css,
|
||||||
|
CSSResult,
|
||||||
|
customElement,
|
||||||
|
html,
|
||||||
|
LitElement,
|
||||||
|
property,
|
||||||
|
TemplateResult,
|
||||||
|
} from "lit-element";
|
||||||
|
import memoizeOne from "memoize-one";
|
||||||
|
import { fireEvent } from "../common/dom/fire_event";
|
||||||
|
import { compare } from "../common/string/compare";
|
||||||
|
import { Blueprint, Blueprints, fetchBlueprints } from "../data/blueprint";
|
||||||
|
import { HomeAssistant } from "../types";
|
||||||
|
|
||||||
|
@customElement("ha-blueprint-picker")
|
||||||
|
class HaBluePrintPicker extends LitElement {
|
||||||
|
public hass?: HomeAssistant;
|
||||||
|
|
||||||
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@property() public value = "";
|
||||||
|
|
||||||
|
@property() public domain = "automation";
|
||||||
|
|
||||||
|
@property() public blueprints?: Blueprints;
|
||||||
|
|
||||||
|
@property({ type: Boolean }) public disabled = false;
|
||||||
|
|
||||||
|
private _processedBlueprints = memoizeOne((blueprints?: Blueprints) => {
|
||||||
|
if (!blueprints) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
const result = Object.entries(blueprints)
|
||||||
|
.filter(([_path, blueprint]) => !("error" in blueprint))
|
||||||
|
.map(([path, blueprint]) => ({
|
||||||
|
...(blueprint as Blueprint).metadata,
|
||||||
|
path,
|
||||||
|
}));
|
||||||
|
return result.sort((a, b) => compare(a.name, b.name));
|
||||||
|
});
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
if (!this.hass) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
return html`
|
||||||
|
<paper-dropdown-menu-light
|
||||||
|
.label=${this.label ||
|
||||||
|
this.hass.localize("ui.components.blueprint-picker.label")}
|
||||||
|
.disabled=${this.disabled}
|
||||||
|
>
|
||||||
|
<paper-listbox
|
||||||
|
slot="dropdown-content"
|
||||||
|
.selected=${this.value}
|
||||||
|
attr-for-selected="data-blueprint-path"
|
||||||
|
@iron-select=${this._blueprintChanged}
|
||||||
|
>
|
||||||
|
<paper-item data-blueprint-path="">
|
||||||
|
${this.hass.localize(
|
||||||
|
"ui.components.blueprint-picker.select_blueprint"
|
||||||
|
)}
|
||||||
|
</paper-item>
|
||||||
|
${this._processedBlueprints(this.blueprints).map(
|
||||||
|
(blueprint) => html`
|
||||||
|
<paper-item data-blueprint-path=${blueprint.path}>
|
||||||
|
${blueprint.name}
|
||||||
|
</paper-item>
|
||||||
|
`
|
||||||
|
)}
|
||||||
|
</paper-listbox>
|
||||||
|
</paper-dropdown-menu-light>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
protected firstUpdated(changedProps) {
|
||||||
|
super.firstUpdated(changedProps);
|
||||||
|
if (this.blueprints === undefined) {
|
||||||
|
fetchBlueprints(this.hass!, this.domain).then((blueprints) => {
|
||||||
|
this.blueprints = blueprints;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private _blueprintChanged(ev) {
|
||||||
|
const newValue = ev.detail.item.dataset.blueprintPath;
|
||||||
|
|
||||||
|
if (newValue !== this.value) {
|
||||||
|
this.value = ev.detail.value;
|
||||||
|
setTimeout(() => {
|
||||||
|
fireEvent(this, "value-changed", { value: newValue });
|
||||||
|
fireEvent(this, "change");
|
||||||
|
}, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles(): CSSResult {
|
||||||
|
return css`
|
||||||
|
:host {
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
paper-dropdown-menu-light {
|
||||||
|
width: 100%;
|
||||||
|
min-width: 200px;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
paper-listbox {
|
||||||
|
min-width: 200px;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"ha-blueprint-picker": HaBluePrintPicker;
|
||||||
|
}
|
||||||
|
}
|
@ -13,11 +13,12 @@ import { fireEvent } from "../common/dom/fire_event";
|
|||||||
import { computeStateName } from "../common/entity/compute_state_name";
|
import { computeStateName } from "../common/entity/compute_state_name";
|
||||||
import { supportsFeature } from "../common/entity/supports-feature";
|
import { supportsFeature } from "../common/entity/supports-feature";
|
||||||
import {
|
import {
|
||||||
|
CameraEntity,
|
||||||
CAMERA_SUPPORT_STREAM,
|
CAMERA_SUPPORT_STREAM,
|
||||||
computeMJPEGStreamUrl,
|
computeMJPEGStreamUrl,
|
||||||
fetchStreamUrl,
|
fetchStreamUrl,
|
||||||
} from "../data/camera";
|
} from "../data/camera";
|
||||||
import { CameraEntity, HomeAssistant } from "../types";
|
import { HomeAssistant } from "../types";
|
||||||
import "./ha-hls-player";
|
import "./ha-hls-player";
|
||||||
|
|
||||||
@customElement("ha-camera-stream")
|
@customElement("ha-camera-stream")
|
||||||
|
@ -11,6 +11,7 @@ import { HassEntity } from "home-assistant-js-websocket";
|
|||||||
|
|
||||||
import { CLIMATE_PRESET_NONE } from "../data/climate";
|
import { CLIMATE_PRESET_NONE } from "../data/climate";
|
||||||
import type { HomeAssistant } from "../types";
|
import type { HomeAssistant } from "../types";
|
||||||
|
import { formatNumber } from "../common/string/format_number";
|
||||||
|
|
||||||
@customElement("ha-climate-state")
|
@customElement("ha-climate-state")
|
||||||
class HaClimateState extends LitElement {
|
class HaClimateState extends LitElement {
|
||||||
@ -51,11 +52,17 @@ class HaClimateState extends LitElement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (this.stateObj.attributes.current_temperature != null) {
|
if (this.stateObj.attributes.current_temperature != null) {
|
||||||
return `${this.stateObj.attributes.current_temperature} ${this.hass.config.unit_system.temperature}`;
|
return `${formatNumber(
|
||||||
|
this.stateObj.attributes.current_temperature,
|
||||||
|
this.hass!.language
|
||||||
|
)} ${this.hass.config.unit_system.temperature}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.stateObj.attributes.current_humidity != null) {
|
if (this.stateObj.attributes.current_humidity != null) {
|
||||||
return `${this.stateObj.attributes.current_humidity} %`;
|
return `${formatNumber(
|
||||||
|
this.stateObj.attributes.current_humidity,
|
||||||
|
this.hass!.language
|
||||||
|
)} %`;
|
||||||
}
|
}
|
||||||
|
|
||||||
return undefined;
|
return undefined;
|
||||||
@ -70,21 +77,39 @@ class HaClimateState extends LitElement {
|
|||||||
this.stateObj.attributes.target_temp_low != null &&
|
this.stateObj.attributes.target_temp_low != null &&
|
||||||
this.stateObj.attributes.target_temp_high != null
|
this.stateObj.attributes.target_temp_high != null
|
||||||
) {
|
) {
|
||||||
return `${this.stateObj.attributes.target_temp_low}-${this.stateObj.attributes.target_temp_high} ${this.hass.config.unit_system.temperature}`;
|
return `${formatNumber(
|
||||||
|
this.stateObj.attributes.target_temp_low,
|
||||||
|
this.hass!.language
|
||||||
|
)}-${formatNumber(
|
||||||
|
this.stateObj.attributes.target_temp_high,
|
||||||
|
this.hass!.language
|
||||||
|
)} ${this.hass.config.unit_system.temperature}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.stateObj.attributes.temperature != null) {
|
if (this.stateObj.attributes.temperature != null) {
|
||||||
return `${this.stateObj.attributes.temperature} ${this.hass.config.unit_system.temperature}`;
|
return `${formatNumber(
|
||||||
|
this.stateObj.attributes.temperature,
|
||||||
|
this.hass!.language
|
||||||
|
)} ${this.hass.config.unit_system.temperature}`;
|
||||||
}
|
}
|
||||||
if (
|
if (
|
||||||
this.stateObj.attributes.target_humidity_low != null &&
|
this.stateObj.attributes.target_humidity_low != null &&
|
||||||
this.stateObj.attributes.target_humidity_high != null
|
this.stateObj.attributes.target_humidity_high != null
|
||||||
) {
|
) {
|
||||||
return `${this.stateObj.attributes.target_humidity_low}-${this.stateObj.attributes.target_humidity_high}%`;
|
return `${formatNumber(
|
||||||
|
this.stateObj.attributes.target_humidity_low,
|
||||||
|
this.hass!.language
|
||||||
|
)}-${formatNumber(
|
||||||
|
this.stateObj.attributes.target_humidity_high,
|
||||||
|
this.hass!.language
|
||||||
|
)}%`;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this.stateObj.attributes.humidity != null) {
|
if (this.stateObj.attributes.humidity != null) {
|
||||||
return `${this.stateObj.attributes.humidity} %`;
|
return `${formatNumber(
|
||||||
|
this.stateObj.attributes.humidity,
|
||||||
|
this.hass!.language
|
||||||
|
)} %`;
|
||||||
}
|
}
|
||||||
|
|
||||||
return "";
|
return "";
|
||||||
|
@ -1,126 +0,0 @@
|
|||||||
import "./ha-icon-button";
|
|
||||||
import { html } from "@polymer/polymer/lib/utils/html-tag";
|
|
||||||
/* eslint-plugin-disable lit */
|
|
||||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
|
|
||||||
import { UNAVAILABLE } from "../data/entity";
|
|
||||||
import CoverEntity from "../util/cover-model";
|
|
||||||
|
|
||||||
class HaCoverControls extends PolymerElement {
|
|
||||||
static get template() {
|
|
||||||
return html`
|
|
||||||
<style>
|
|
||||||
.state {
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
[invisible] {
|
|
||||||
visibility: hidden !important;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|
||||||
<div class="state">
|
|
||||||
<ha-icon-button
|
|
||||||
aria-label="Open cover"
|
|
||||||
icon="[[computeOpenIcon(stateObj)]]"
|
|
||||||
on-click="onOpenTap"
|
|
||||||
invisible$="[[!entityObj.supportsOpen]]"
|
|
||||||
disabled="[[computeOpenDisabled(stateObj, entityObj)]]"
|
|
||||||
></ha-icon-button>
|
|
||||||
<ha-icon-button
|
|
||||||
aria-label="Stop the cover from moving"
|
|
||||||
icon="hass:stop"
|
|
||||||
on-click="onStopTap"
|
|
||||||
invisible$="[[!entityObj.supportsStop]]"
|
|
||||||
disabled="[[computeStopDisabled(stateObj)]]"
|
|
||||||
></ha-icon-button>
|
|
||||||
<ha-icon-button
|
|
||||||
aria-label="Close cover"
|
|
||||||
icon="[[computeCloseIcon(stateObj)]]"
|
|
||||||
on-click="onCloseTap"
|
|
||||||
invisible$="[[!entityObj.supportsClose]]"
|
|
||||||
disabled="[[computeClosedDisabled(stateObj, entityObj)]]"
|
|
||||||
></ha-icon-button>
|
|
||||||
</div>
|
|
||||||
`;
|
|
||||||
}
|
|
||||||
|
|
||||||
static get properties() {
|
|
||||||
return {
|
|
||||||
hass: {
|
|
||||||
type: Object,
|
|
||||||
},
|
|
||||||
stateObj: {
|
|
||||||
type: Object,
|
|
||||||
},
|
|
||||||
entityObj: {
|
|
||||||
type: Object,
|
|
||||||
computed: "computeEntityObj(hass, stateObj)",
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
computeEntityObj(hass, stateObj) {
|
|
||||||
return new CoverEntity(hass, stateObj);
|
|
||||||
}
|
|
||||||
|
|
||||||
computeOpenIcon(stateObj) {
|
|
||||||
switch (stateObj.attributes.device_class) {
|
|
||||||
case "awning":
|
|
||||||
case "door":
|
|
||||||
case "gate":
|
|
||||||
return "hass:arrow-expand-horizontal";
|
|
||||||
default:
|
|
||||||
return "hass:arrow-up";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
computeCloseIcon(stateObj) {
|
|
||||||
switch (stateObj.attributes.device_class) {
|
|
||||||
case "awning":
|
|
||||||
case "door":
|
|
||||||
case "gate":
|
|
||||||
return "hass:arrow-collapse-horizontal";
|
|
||||||
default:
|
|
||||||
return "hass:arrow-down";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
computeStopDisabled(stateObj) {
|
|
||||||
if (stateObj.state === UNAVAILABLE) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
computeOpenDisabled(stateObj, entityObj) {
|
|
||||||
if (stateObj.state === UNAVAILABLE) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
const assumedState = stateObj.attributes.assumed_state === true;
|
|
||||||
return (entityObj.isFullyOpen || entityObj.isOpening) && !assumedState;
|
|
||||||
}
|
|
||||||
|
|
||||||
computeClosedDisabled(stateObj, entityObj) {
|
|
||||||
if (stateObj.state === UNAVAILABLE) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
const assumedState = stateObj.attributes.assumed_state === true;
|
|
||||||
return (entityObj.isFullyClosed || entityObj.isClosing) && !assumedState;
|
|
||||||
}
|
|
||||||
|
|
||||||
onOpenTap(ev) {
|
|
||||||
ev.stopPropagation();
|
|
||||||
this.entityObj.openCover();
|
|
||||||
}
|
|
||||||
|
|
||||||
onCloseTap(ev) {
|
|
||||||
ev.stopPropagation();
|
|
||||||
this.entityObj.closeCover();
|
|
||||||
}
|
|
||||||
|
|
||||||
onStopTap(ev) {
|
|
||||||
ev.stopPropagation();
|
|
||||||
this.entityObj.stopCover();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
customElements.define("ha-cover-controls", HaCoverControls);
|
|
135
src/components/ha-cover-controls.ts
Normal file
135
src/components/ha-cover-controls.ts
Normal file
@ -0,0 +1,135 @@
|
|||||||
|
import {
|
||||||
|
css,
|
||||||
|
CSSResult,
|
||||||
|
customElement,
|
||||||
|
html,
|
||||||
|
internalProperty,
|
||||||
|
LitElement,
|
||||||
|
property,
|
||||||
|
PropertyValues,
|
||||||
|
TemplateResult,
|
||||||
|
} from "lit-element";
|
||||||
|
import { classMap } from "lit-html/directives/class-map";
|
||||||
|
import type { HassEntity } from "home-assistant-js-websocket";
|
||||||
|
|
||||||
|
import type { HomeAssistant } from "../types";
|
||||||
|
import { UNAVAILABLE } from "../data/entity";
|
||||||
|
import CoverEntity from "../util/cover-model";
|
||||||
|
|
||||||
|
import "./ha-icon-button";
|
||||||
|
import { computeCloseIcon, computeOpenIcon } from "../common/entity/cover_icon";
|
||||||
|
|
||||||
|
@customElement("ha-cover-controls")
|
||||||
|
class HaCoverControls extends LitElement {
|
||||||
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public stateObj!: HassEntity;
|
||||||
|
|
||||||
|
@internalProperty() private _entityObj?: CoverEntity;
|
||||||
|
|
||||||
|
protected updated(changedProperties: PropertyValues): void {
|
||||||
|
super.updated(changedProperties);
|
||||||
|
|
||||||
|
if (changedProperties.has("stateObj")) {
|
||||||
|
this._entityObj = new CoverEntity(this.hass, this.stateObj);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected render(): TemplateResult {
|
||||||
|
if (!this._entityObj) {
|
||||||
|
return html``;
|
||||||
|
}
|
||||||
|
|
||||||
|
return html`
|
||||||
|
<div class="state">
|
||||||
|
<ha-icon-button
|
||||||
|
class=${classMap({
|
||||||
|
hidden: !this._entityObj.supportsOpen,
|
||||||
|
})}
|
||||||
|
.label=${this.hass.localize(
|
||||||
|
"ui.dialogs.more_info_control.open_cover"
|
||||||
|
)}
|
||||||
|
.icon=${computeOpenIcon(this.stateObj)}
|
||||||
|
@click=${this._onOpenTap}
|
||||||
|
.disabled=${this._computeOpenDisabled()}
|
||||||
|
></ha-icon-button>
|
||||||
|
<ha-icon-button
|
||||||
|
class=${classMap({
|
||||||
|
hidden: !this._entityObj.supportsStop,
|
||||||
|
})}
|
||||||
|
.label=${this.hass.localize(
|
||||||
|
"ui.dialogs.more_info_control.stop_cover"
|
||||||
|
)}
|
||||||
|
icon="hass:stop"
|
||||||
|
@click=${this._onStopTap}
|
||||||
|
.disabled=${this.stateObj.state === UNAVAILABLE}
|
||||||
|
></ha-icon-button>
|
||||||
|
<ha-icon-button
|
||||||
|
class=${classMap({
|
||||||
|
hidden: !this._entityObj.supportsClose,
|
||||||
|
})}
|
||||||
|
.label=${this.hass.localize(
|
||||||
|
"ui.dialogs.more_info_control.close_cover"
|
||||||
|
)}
|
||||||
|
.icon=${computeCloseIcon(this.stateObj)}
|
||||||
|
@click=${this._onCloseTap}
|
||||||
|
.disabled=${this._computeClosedDisabled()}
|
||||||
|
></ha-icon-button>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _computeOpenDisabled(): boolean {
|
||||||
|
if (this.stateObj.state === UNAVAILABLE) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
const assumedState = this.stateObj.attributes.assumed_state === true;
|
||||||
|
return (
|
||||||
|
(this._entityObj.isFullyOpen || this._entityObj.isOpening) &&
|
||||||
|
!assumedState
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private _computeClosedDisabled(): boolean {
|
||||||
|
if (this.stateObj.state === UNAVAILABLE) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
const assumedState = this.stateObj.attributes.assumed_state === true;
|
||||||
|
return (
|
||||||
|
(this._entityObj.isFullyClosed || this._entityObj.isClosing) &&
|
||||||
|
!assumedState
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private _onOpenTap(ev): void {
|
||||||
|
ev.stopPropagation();
|
||||||
|
this._entityObj.openCover();
|
||||||
|
}
|
||||||
|
|
||||||
|
private _onCloseTap(ev): void {
|
||||||
|
ev.stopPropagation();
|
||||||
|
this._entityObj.closeCover();
|
||||||
|
}
|
||||||
|
|
||||||
|
private _onStopTap(ev): void {
|
||||||
|
ev.stopPropagation();
|
||||||
|
this._entityObj.stopCover();
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles(): CSSResult {
|
||||||
|
return css`
|
||||||
|
.state {
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
.hidden {
|
||||||
|
visibility: hidden !important;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"ha-cover-controls": HaCoverControls;
|
||||||
|
}
|
||||||
|
}
|
@ -41,7 +41,7 @@ class HaCoverTiltControls extends LitElement {
|
|||||||
|
|
||||||
return html` <ha-icon-button
|
return html` <ha-icon-button
|
||||||
class=${classMap({
|
class=${classMap({
|
||||||
invisible: !this._entityObj.supportsStop,
|
invisible: !this._entityObj.supportsOpenTilt,
|
||||||
})}
|
})}
|
||||||
label=${this.hass.localize(
|
label=${this.hass.localize(
|
||||||
"ui.dialogs.more_info_control.open_tilt_cover"
|
"ui.dialogs.more_info_control.open_tilt_cover"
|
||||||
@ -61,10 +61,10 @@ class HaCoverTiltControls extends LitElement {
|
|||||||
></ha-icon-button>
|
></ha-icon-button>
|
||||||
<ha-icon-button
|
<ha-icon-button
|
||||||
class=${classMap({
|
class=${classMap({
|
||||||
invisible: !this._entityObj.supportsStop,
|
invisible: !this._entityObj.supportsCloseTilt,
|
||||||
})}
|
})}
|
||||||
label=${this.hass.localize(
|
label=${this.hass.localize(
|
||||||
"ui.dialogs.more_info_control.open_tilt_cover"
|
"ui.dialogs.more_info_control.close_tilt_cover"
|
||||||
)}
|
)}
|
||||||
icon="hass:arrow-bottom-left"
|
icon="hass:arrow-bottom-left"
|
||||||
@click=${this._onCloseTiltTap}
|
@click=${this._onCloseTiltTap}
|
||||||
|
@ -76,7 +76,7 @@ class HaExpansionPanel extends LitElement {
|
|||||||
|
|
||||||
.summary {
|
.summary {
|
||||||
display: flex;
|
display: flex;
|
||||||
padding: 0px 16px;
|
padding: var(--expansion-panel-summary-padding, 0px 16px);
|
||||||
min-height: 48px;
|
min-height: 48px;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
@ -54,7 +54,7 @@ export interface HaFormSelectSchema extends HaFormBaseSchema {
|
|||||||
|
|
||||||
export interface HaFormMultiSelectSchema extends HaFormBaseSchema {
|
export interface HaFormMultiSelectSchema extends HaFormBaseSchema {
|
||||||
type: "multi_select";
|
type: "multi_select";
|
||||||
options?: { [key: string]: string } | string[] | Array<[string, string]>;
|
options?: Record<string, string> | string[] | Array<[string, string]>;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface HaFormFloatSchema extends HaFormBaseSchema {
|
export interface HaFormFloatSchema extends HaFormBaseSchema {
|
||||||
|
@ -12,6 +12,7 @@ import { afterNextRender } from "../common/util/render-status";
|
|||||||
import { ifDefined } from "lit-html/directives/if-defined";
|
import { ifDefined } from "lit-html/directives/if-defined";
|
||||||
|
|
||||||
import { getValueInPercentage, normalize } from "../util/calculate";
|
import { getValueInPercentage, normalize } from "../util/calculate";
|
||||||
|
import { formatNumber } from "../common/string/format_number";
|
||||||
|
|
||||||
const getAngle = (value: number, min: number, max: number) => {
|
const getAngle = (value: number, min: number, max: number) => {
|
||||||
const percentage = getValueInPercentage(normalize(value, min, max), min, max);
|
const percentage = getValueInPercentage(normalize(value, min, max), min, max);
|
||||||
@ -29,6 +30,8 @@ export class Gauge extends LitElement {
|
|||||||
|
|
||||||
@property({ type: Number }) public value = 0;
|
@property({ type: Number }) public value = 0;
|
||||||
|
|
||||||
|
@property({ type: String }) public language = "";
|
||||||
|
|
||||||
@property() public label = "";
|
@property() public label = "";
|
||||||
|
|
||||||
@internalProperty() private _angle = 0;
|
@internalProperty() private _angle = 0;
|
||||||
@ -88,7 +91,7 @@ export class Gauge extends LitElement {
|
|||||||
</svg>
|
</svg>
|
||||||
<svg class="text">
|
<svg class="text">
|
||||||
<text class="value-text">
|
<text class="value-text">
|
||||||
${this.value} ${this.label}
|
${formatNumber(this.value, this.language)} ${this.label}
|
||||||
</text>
|
</text>
|
||||||
</svg>`;
|
</svg>`;
|
||||||
}
|
}
|
||||||
|
@ -107,9 +107,7 @@ class HaHLSPlayer extends LitElement {
|
|||||||
const useExoPlayerPromise = this._getUseExoPlayer();
|
const useExoPlayerPromise = this._getUseExoPlayer();
|
||||||
const masterPlaylistPromise = fetch(this.url);
|
const masterPlaylistPromise = fetch(this.url);
|
||||||
|
|
||||||
const hls = ((await import(
|
const hls = ((await import("hls.js")) as any).default as HLSModule;
|
||||||
/* webpackChunkName: "hls.js" */ "hls.js"
|
|
||||||
)) as any).default as HLSModule;
|
|
||||||
let hlsSupported = hls.isSupported();
|
let hlsSupported = hls.isSupported();
|
||||||
|
|
||||||
if (!hlsSupported) {
|
if (!hlsSupported) {
|
||||||
|
@ -39,7 +39,7 @@ checkCacheVersion();
|
|||||||
|
|
||||||
const debouncedWriteCache = debounce(() => writeCache(chunks), 2000);
|
const debouncedWriteCache = debounce(() => writeCache(chunks), 2000);
|
||||||
|
|
||||||
const cachedIcons: { [key: string]: string } = {};
|
const cachedIcons: Record<string, string> = {};
|
||||||
|
|
||||||
@customElement("ha-icon")
|
@customElement("ha-icon")
|
||||||
export class HaIcon extends LitElement {
|
export class HaIcon extends LitElement {
|
||||||
|
@ -58,6 +58,7 @@ export class HaRelatedItems extends SubscribeMixin(LitElement) {
|
|||||||
getConfigEntries(this.hass).then((configEntries) => {
|
getConfigEntries(this.hass).then((configEntries) => {
|
||||||
this._entries = configEntries;
|
this._entries = configEntries;
|
||||||
});
|
});
|
||||||
|
this.hass.loadBackendTranslation("title");
|
||||||
}
|
}
|
||||||
|
|
||||||
protected updated(changedProps: PropertyValues) {
|
protected updated(changedProps: PropertyValues) {
|
||||||
|
@ -1,66 +0,0 @@
|
|||||||
import { dom } from "@polymer/polymer/lib/legacy/polymer.dom";
|
|
||||||
/* eslint-plugin-disable lit */
|
|
||||||
import { PolymerElement } from "@polymer/polymer/polymer-element";
|
|
||||||
import relativeTime from "../common/datetime/relative_time";
|
|
||||||
import LocalizeMixin from "../mixins/localize-mixin";
|
|
||||||
|
|
||||||
/*
|
|
||||||
* @appliesMixin LocalizeMixin
|
|
||||||
*/
|
|
||||||
class HaRelativeTime extends LocalizeMixin(PolymerElement) {
|
|
||||||
static get properties() {
|
|
||||||
return {
|
|
||||||
hass: Object,
|
|
||||||
datetime: {
|
|
||||||
type: String,
|
|
||||||
observer: "datetimeChanged",
|
|
||||||
},
|
|
||||||
|
|
||||||
datetimeObj: {
|
|
||||||
type: Object,
|
|
||||||
observer: "datetimeObjChanged",
|
|
||||||
},
|
|
||||||
|
|
||||||
parsedDateTime: Object,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
constructor() {
|
|
||||||
super();
|
|
||||||
this.updateRelative = this.updateRelative.bind(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
connectedCallback() {
|
|
||||||
super.connectedCallback();
|
|
||||||
// update every 60 seconds
|
|
||||||
this.updateInterval = setInterval(this.updateRelative, 60000);
|
|
||||||
}
|
|
||||||
|
|
||||||
disconnectedCallback() {
|
|
||||||
super.disconnectedCallback();
|
|
||||||
clearInterval(this.updateInterval);
|
|
||||||
}
|
|
||||||
|
|
||||||
datetimeChanged(newVal) {
|
|
||||||
this.parsedDateTime = newVal ? new Date(newVal) : null;
|
|
||||||
|
|
||||||
this.updateRelative();
|
|
||||||
}
|
|
||||||
|
|
||||||
datetimeObjChanged(newVal) {
|
|
||||||
this.parsedDateTime = newVal;
|
|
||||||
|
|
||||||
this.updateRelative();
|
|
||||||
}
|
|
||||||
|
|
||||||
updateRelative() {
|
|
||||||
const root = dom(this);
|
|
||||||
if (!this.parsedDateTime) {
|
|
||||||
root.innerHTML = this.localize("ui.components.relative_time.never");
|
|
||||||
} else {
|
|
||||||
root.innerHTML = relativeTime(this.parsedDateTime, this.localize);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
customElements.define("ha-relative-time", HaRelativeTime);
|
|
72
src/components/ha-relative-time.ts
Normal file
72
src/components/ha-relative-time.ts
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
import {
|
||||||
|
customElement,
|
||||||
|
UpdatingElement,
|
||||||
|
property,
|
||||||
|
PropertyValues,
|
||||||
|
} from "lit-element";
|
||||||
|
|
||||||
|
import relativeTime from "../common/datetime/relative_time";
|
||||||
|
|
||||||
|
import type { HomeAssistant } from "../types";
|
||||||
|
|
||||||
|
@customElement("ha-relative-time")
|
||||||
|
class HaRelativeTime extends UpdatingElement {
|
||||||
|
@property({ attribute: false }) public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property({ attribute: false }) public datetime?: string;
|
||||||
|
|
||||||
|
private _interval?: number;
|
||||||
|
|
||||||
|
public disconnectedCallback(): void {
|
||||||
|
super.disconnectedCallback();
|
||||||
|
this._clearInterval();
|
||||||
|
}
|
||||||
|
|
||||||
|
public connectedCallback(): void {
|
||||||
|
super.connectedCallback();
|
||||||
|
if (this.datetime) {
|
||||||
|
this._startInterval();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected firstUpdated(changedProps: PropertyValues) {
|
||||||
|
super.firstUpdated(changedProps);
|
||||||
|
this._updateRelative();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected update(changedProps: PropertyValues) {
|
||||||
|
super.update(changedProps);
|
||||||
|
this._updateRelative();
|
||||||
|
}
|
||||||
|
|
||||||
|
private _clearInterval(): void {
|
||||||
|
if (this._interval) {
|
||||||
|
window.clearInterval(this._interval);
|
||||||
|
this._interval = undefined;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private _startInterval(): void {
|
||||||
|
this._clearInterval();
|
||||||
|
|
||||||
|
// update every 60 seconds
|
||||||
|
this._interval = window.setInterval(() => this._updateRelative(), 60000);
|
||||||
|
}
|
||||||
|
|
||||||
|
private _updateRelative(): void {
|
||||||
|
if (!this.datetime) {
|
||||||
|
this.innerHTML = this.hass.localize("ui.components.relative_time.never");
|
||||||
|
} else {
|
||||||
|
this.innerHTML = relativeTime(
|
||||||
|
new Date(this.datetime),
|
||||||
|
this.hass.localize
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"ha-relative-time": HaRelativeTime;
|
||||||
|
}
|
||||||
|
}
|
30
src/components/ha-selector/ha-selector-area.ts
Normal file
30
src/components/ha-selector/ha-selector-area.ts
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
import { customElement, html, LitElement, property } from "lit-element";
|
||||||
|
import { HomeAssistant } from "../../types";
|
||||||
|
import { AreaSelector } from "../../data/selector";
|
||||||
|
import "../ha-area-picker";
|
||||||
|
|
||||||
|
@customElement("ha-selector-area")
|
||||||
|
export class HaAreaSelector extends LitElement {
|
||||||
|
@property() public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property() public selector!: AreaSelector;
|
||||||
|
|
||||||
|
@property() public value?: any;
|
||||||
|
|
||||||
|
@property() public label?: string;
|
||||||
|
|
||||||
|
protected render() {
|
||||||
|
return html`<ha-area-picker
|
||||||
|
.hass=${this.hass}
|
||||||
|
.value=${this.value}
|
||||||
|
.label=${this.label}
|
||||||
|
no-add
|
||||||
|
></ha-area-picker>`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"ha-selector-area": HaAreaSelector;
|
||||||
|
}
|
||||||
|
}
|
54
src/components/ha-selector/ha-selector-boolean.ts
Normal file
54
src/components/ha-selector/ha-selector-boolean.ts
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
import {
|
||||||
|
css,
|
||||||
|
CSSResult,
|
||||||
|
customElement,
|
||||||
|
html,
|
||||||
|
LitElement,
|
||||||
|
property,
|
||||||
|
} from "lit-element";
|
||||||
|
import { fireEvent } from "../../common/dom/fire_event";
|
||||||
|
import { HomeAssistant } from "../../types";
|
||||||
|
import "../ha-formfield";
|
||||||
|
import "../ha-switch";
|
||||||
|
|
||||||
|
@customElement("ha-selector-boolean")
|
||||||
|
export class HaBooleanSelector extends LitElement {
|
||||||
|
@property() public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property() public value?: number;
|
||||||
|
|
||||||
|
@property() public label?: string;
|
||||||
|
|
||||||
|
protected render() {
|
||||||
|
return html` <ha-formfield alignEnd spaceBetween .label=${this.label}>
|
||||||
|
<ha-switch
|
||||||
|
.checked=${this.value}
|
||||||
|
@change=${this._handleChange}
|
||||||
|
></ha-switch>
|
||||||
|
</ha-formfield>`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _handleChange(ev) {
|
||||||
|
const value = ev.target.checked;
|
||||||
|
if (this.value === value) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
fireEvent(this, "value-changed", { value });
|
||||||
|
}
|
||||||
|
|
||||||
|
static get styles(): CSSResult {
|
||||||
|
return css`
|
||||||
|
ha-formfield {
|
||||||
|
width: 100%;
|
||||||
|
margin: 16px 0;
|
||||||
|
--mdc-typography-body2-font-size: 1em;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"ha-selector-boolean": HaBooleanSelector;
|
||||||
|
}
|
||||||
|
}
|
87
src/components/ha-selector/ha-selector-device.ts
Normal file
87
src/components/ha-selector/ha-selector-device.ts
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
import {
|
||||||
|
customElement,
|
||||||
|
html,
|
||||||
|
internalProperty,
|
||||||
|
LitElement,
|
||||||
|
property,
|
||||||
|
} from "lit-element";
|
||||||
|
import { HomeAssistant } from "../../types";
|
||||||
|
import "../device/ha-device-picker";
|
||||||
|
import { DeviceRegistryEntry } from "../../data/device_registry";
|
||||||
|
import { ConfigEntry, getConfigEntries } from "../../data/config_entries";
|
||||||
|
import { DeviceSelector } from "../../data/selector";
|
||||||
|
|
||||||
|
@customElement("ha-selector-device")
|
||||||
|
export class HaDeviceSelector extends LitElement {
|
||||||
|
@property() public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property() public selector!: DeviceSelector;
|
||||||
|
|
||||||
|
@property() public value?: any;
|
||||||
|
|
||||||
|
@property() public label?: string;
|
||||||
|
|
||||||
|
@internalProperty() public _configEntries?: ConfigEntry[];
|
||||||
|
|
||||||
|
protected updated(changedProperties) {
|
||||||
|
if (changedProperties.has("selector")) {
|
||||||
|
const oldSelector = changedProperties.get("selector");
|
||||||
|
if (oldSelector !== this.selector && this.selector.device.integration) {
|
||||||
|
this._loadConfigEntries();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
protected render() {
|
||||||
|
return html`<ha-device-picker
|
||||||
|
.hass=${this.hass}
|
||||||
|
.value=${this.value}
|
||||||
|
.label=${this.label}
|
||||||
|
.deviceFilter=${(device) => this._filterDevices(device)}
|
||||||
|
.includeDeviceClasses=${this.selector.device.entity?.device_class
|
||||||
|
? [this.selector.device.entity.device_class]
|
||||||
|
: undefined}
|
||||||
|
.includeDomains=${this.selector.device.entity?.domain
|
||||||
|
? [this.selector.device.entity.domain]
|
||||||
|
: undefined}
|
||||||
|
allow-custom-entity
|
||||||
|
></ha-device-picker>`;
|
||||||
|
}
|
||||||
|
|
||||||
|
private _filterDevices(device: DeviceRegistryEntry): boolean {
|
||||||
|
if (
|
||||||
|
this.selector.device.manufacturer &&
|
||||||
|
device.manufacturer !== this.selector.device.manufacturer
|
||||||
|
) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
this.selector.device.model &&
|
||||||
|
device.model !== this.selector.device.model
|
||||||
|
) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (this.selector.device.integration) {
|
||||||
|
if (
|
||||||
|
!this._configEntries?.some((entry) =>
|
||||||
|
device.config_entries.includes(entry.entry_id)
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
private async _loadConfigEntries() {
|
||||||
|
this._configEntries = (await getConfigEntries(this.hass)).filter(
|
||||||
|
(entry) => entry.domain === this.selector.device.integration
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"ha-selector-device": HaDeviceSelector;
|
||||||
|
}
|
||||||
|
}
|
83
src/components/ha-selector/ha-selector-entity.ts
Normal file
83
src/components/ha-selector/ha-selector-entity.ts
Normal file
@ -0,0 +1,83 @@
|
|||||||
|
import {
|
||||||
|
customElement,
|
||||||
|
html,
|
||||||
|
internalProperty,
|
||||||
|
LitElement,
|
||||||
|
property,
|
||||||
|
} from "lit-element";
|
||||||
|
import { HomeAssistant } from "../../types";
|
||||||
|
import "../entity/ha-entity-picker";
|
||||||
|
import { HassEntity, UnsubscribeFunc } from "home-assistant-js-websocket";
|
||||||
|
import { computeStateDomain } from "../../common/entity/compute_state_domain";
|
||||||
|
import { subscribeEntityRegistry } from "../../data/entity_registry";
|
||||||
|
import { SubscribeMixin } from "../../mixins/subscribe-mixin";
|
||||||
|
import { EntitySelector } from "../../data/selector";
|
||||||
|
|
||||||
|
@customElement("ha-selector-entity")
|
||||||
|
export class HaEntitySelector extends SubscribeMixin(LitElement) {
|
||||||
|
@property() public hass!: HomeAssistant;
|
||||||
|
|
||||||
|
@property() public selector!: EntitySelector;
|
||||||
|
|
||||||
|
@internalProperty() private _entities?: Record<string, string>;
|
||||||
|
|
||||||
|
@property() public value?: any;
|
||||||
|
|
||||||
|
@property() public label?: string;
|
||||||
|
|
||||||
|
protected render() {
|
||||||
|
return html`<ha-entity-picker
|
||||||
|
.hass=${this.hass}
|
||||||
|
.value=${this.value}
|
||||||
|
.label=${this.label}
|
||||||
|
.entityFilter=${(entity) => this._filterEntities(entity)}
|
||||||
|
allow-custom-entity
|
||||||
|
></ha-entity-picker>`;
|
||||||
|
}
|
||||||
|
|
||||||
|
public hassSubscribe(): UnsubscribeFunc[] {
|
||||||
|
return [
|
||||||
|
subscribeEntityRegistry(this.hass.connection!, (entities) => {
|
||||||
|
const entityLookup = {};
|
||||||
|
for (const confEnt of entities) {
|
||||||
|
if (!confEnt.platform) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
entityLookup[confEnt.entity_id] = confEnt.platform;
|
||||||
|
}
|
||||||
|
this._entities = entityLookup;
|
||||||
|
}),
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
private _filterEntities(entity: HassEntity): boolean {
|
||||||
|
if (this.selector.entity.domain) {
|
||||||
|
if (computeStateDomain(entity) !== this.selector.entity.domain) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (this.selector.entity.device_class) {
|
||||||
|
if (
|
||||||
|
!entity.attributes.device_class ||
|
||||||
|
entity.attributes.device_class !== this.selector.entity.device_class
|
||||||
|
) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (this.selector.entity.integration) {
|
||||||
|
if (
|
||||||
|
!this._entities ||
|
||||||
|
this._entities[entity.entity_id] !== this.selector.entity.integration
|
||||||
|
) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
declare global {
|
||||||
|
interface HTMLElementTagNameMap {
|
||||||
|
"ha-selector-entity": HaEntitySelector;
|
||||||
|
}
|
||||||
|
}
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user