Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Update all development dependencies and run project linting #324

Merged
merged 2 commits into from
Jan 6, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
14 changes: 0 additions & 14 deletions .eslintrc.cjs

This file was deleted.

3 changes: 1 addition & 2 deletions .github/FUNDING.yml
Original file line number Diff line number Diff line change
@@ -1,6 +1,6 @@
# These are supported funding model platforms

github: # Replace with up to 4 GitHub Sponsors-enabled usernames e.g., [user1, user2]
github: [cgiesche]
patreon: # Replace with a single Patreon username
open_collective: # Replace with a single Open Collective username
ko_fi: # Replace with a single Ko-fi username
Expand All @@ -9,5 +9,4 @@ community_bridge: # Replace with a single Community Bridge project-name e.g., cl
liberapay: # Replace with a single Liberapay username
issuehunt: # Replace with a single IssueHunt username
otechie: # Replace with a single Otechie username
github: [cgiesche]
custom: ['paypal.me/cgidev']
12 changes: 6 additions & 6 deletions .github/workflows/main.yml
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
name: CI

# Controls when the action will run.
# Controls when the action will run.
on:
# Triggers the workflow on push or pull request events but only for the master branch
push:
branches: [ master ]
branches: [master]
pull_request:
branches: [ master ]
branches: [master]

# Allows you to run this workflow manually from the Actions tab
workflow_dispatch:
Expand All @@ -22,13 +22,13 @@ jobs:
steps:
# Checks-out your repository under $GITHUB_WORKSPACE, so your job can access it
- uses: actions/checkout@v2

- name: Setup Node.js environment
uses: actions/[email protected]

- name: NPM Install
run: npm install

- name: Build
run: npm run build

Expand Down
13 changes: 11 additions & 2 deletions .prettierrc.json
Original file line number Diff line number Diff line change
Expand Up @@ -4,5 +4,14 @@
"tabWidth": 2,
"singleQuote": true,
"printWidth": 100,
"trailingComma": "none"
}
"trailingComma": "none",
"endOfLine": "auto",
"overrides": [
{
"files": "*.vue",
"options": {
"printWidth": 120
}
}
]
}
168 changes: 92 additions & 76 deletions README.md
Original file line number Diff line number Diff line change
@@ -1,124 +1,140 @@
# StreamDeck Home-Assistant Plugin

The aim of this project is to allow owners of an elgato StreamDeck to control their entities or display sensor data via
their StreamDeck. As the code is very generic, nearly every sensor should work out of the box as well as every
service that does not need any more information but the entity id.

![img.png](doc/example.png)

## Features

### Generic functions for any Home Assistant entity
* Display state and additional attributes via customizable templates
* Call services when a StreamDeck button is pressed (or hold for some time)

- Display state and additional attributes via customizable templates
- Call services when a StreamDeck button is pressed (or hold for some time)

### Entities with background-icons
* Switches
* Lights (same icon as switches)
* Input Boolean (same icon as switches)
* Binary Sensors
* Plugs
* Temperature
* Humidity
* Pressure
* Power
* Voltage
* Battery-Level
* Weather
* Vacuum robots (experimental, tested with Roborock S5)

- Switches
- Lights (same icon as switches)
- Input Boolean (same icon as switches)
- Binary Sensors
- Plugs
- Temperature
- Humidity
- Pressure
- Power
- Voltage
- Battery-Level
- Weather
- Vacuum robots (experimental, tested with Roborock S5)

# Installation

## Prerequisites
* Installed Stream-Deck application
* Connected Stream-Deck

- Installed Stream-Deck application
- Connected Stream-Deck

## Installation

### Via the official Stream Deck store

You can find and install this plugin from the [official Stream Deck Store](https://apps.elgato.com/plugins/de.perdoctus.streamdeck.homeassistant)

### Manual installation (not recommended)
* Download the latest plugin release [here][https://github.com/cgiesche/streamdeck-homeassistant/releases]
* Open the downloaded .sdplugin file. It will be automatically installed into your Stream-Deck application
* Note for MacOS users: [manual installation steps][https://www.reddit.com/r/homeassistant/comments/laq2g4/homeassistant_streamdeck_plugin_dynamic_not_just/glu0zep/?utm_source=share&utm_medium=web2x&context=3]

- Download the latest plugin release [here][https://github.com/cgiesche/streamdeck-homeassistant/releases]
- Open the downloaded .sdplugin file. It will be automatically installed into your Stream-Deck application
- Note for MacOS users: [manual installation steps][https://www.reddit.com/r/homeassistant/comments/laq2g4/homeassistant_streamdeck_plugin_dynamic_not_just/glu0zep/?utm_source=share&utm_medium=web2x&context=3]

# Configuration

There are two sections on the plugin's configuration panel:
* Home Assistant Settings
Contains global settings for your Home Assistant installation. Once saved, the settings are used for all Stream-Deck buttons.
* Entity Settings
Contains settings for an individual button.

- Home Assistant Settings
Contains global settings for your Home Assistant installation. Once saved, the settings are used for all Stream-Deck buttons.
- Entity Settings
Contains settings for an individual button.

## Home Assistant Settings
* Server URL: `ws://your-homeassistant-ip-or-hostname:your-homeassistant-port/api/websocket`, for example (local network) `ws://192.126.0.5:8123/api/websocket` or (public, with ssl enabled on default port 443) `wss://my-secure-homeassistant.com/api/websocket`.
* Access Token: An long-lived access-token (with admin right) obtained from your Home Assistant. The plugin requires the admin rights, because it uses "execute-script" command that is restricted to admin users.
_Long-lived access tokens can be created using the "Long-Lived Access Tokens" section at the bottom of a user's Home Assistant profile page._ (Quote from https://developers.home-assistant.io/docs/auth_api/#long-lived-access-token)


- Server URL: `ws://your-homeassistant-ip-or-hostname:your-homeassistant-port/api/websocket`, for example (local network) `ws://192.126.0.5:8123/api/websocket` or (public, with ssl enabled on default port 443) `wss://my-secure-homeassistant.com/api/websocket`.
- Access Token: An long-lived access-token (with admin right) obtained from your Home Assistant. The plugin requires the admin rights, because it uses "execute-script" command that is restricted to admin users.
_Long-lived access tokens can be created using the "Long-Lived Access Tokens" section at the bottom of a user's Home Assistant profile page._ (Quote from https://developers.home-assistant.io/docs/auth_api/#long-lived-access-token)

After you save your Home Assistant Settings, the plugin will automatically try to connect to your Home Assistant installation. If the connection was successful, the Entity Settings section should allow you to see and configure your entities.

![img_1.png](doc/ha_settings.png)

## Entity Settings

### Basic configuration
* Domain: Home Assistant entities are grouped by domains. Select the domain (for example "switch") of an entity, you want to configure.
* Entity: This is the actual entity you are going to configure (for example "Kitchen Light")
* Service: The service will be called when you press the StreamDeck button (green marker at the top right corner).
* Service (long press): The service that will be called every time you press and hold the StreamDeck button for more than 300ms (blue marker at the top right corner).
* Service Data JSON: JSON formatted data that is sent with your service call when you press a button.
Example:
```
{
"brightness": 120,
"rgb_color": [255, 0, 0]
}
```
If not specified in the JSON, the attribute `entity_id` is being added automatically containing the selected entity's id. You can also specify the target entity_id manually, for example to switch multiple lights at once:
```
{
"entity_id": [
"switch.osram_plug_01_bac00b0a_on_off",
"switch.osram_plug_01_87c00a0a_on_off"
],
"brightness": 120,
"rgb_color": [255, 0, 0]
}
```
You can also use [Jinja2 templates](https://www.home-assistant.io/docs/configuration/templating/) to dynamically process data based on the states or attributes of your Home Assistant entities. See [Using Jinja2 Templates in Service Data JSON](#Advanced-configuration) for more information.

- Domain: Home Assistant entities are grouped by domains. Select the domain (for example "switch") of an entity, you want to configure.
- Entity: This is the actual entity you are going to configure (for example "Kitchen Light")
- Service: The service will be called when you press the StreamDeck button (green marker at the top right corner).
- Service (long press): The service that will be called every time you press and hold the StreamDeck button for more than 300ms (blue marker at the top right corner).
- Service Data JSON: JSON formatted data that is sent with your service call when you press a button.
Example:
```
{
"brightness": 120,
"rgb_color": [255, 0, 0]
}
```
If not specified in the JSON, the attribute `entity_id` is being added automatically containing the selected entity's id. You can also specify the target entity_id manually, for example to switch multiple lights at once:
```
{
"entity_id": [
"switch.osram_plug_01_bac00b0a_on_off",
"switch.osram_plug_01_87c00a0a_on_off"
],
"brightness": 120,
"rgb_color": [255, 0, 0]
}
```
You can also use [Jinja2 templates](https://www.home-assistant.io/docs/configuration/templating/) to dynamically process data based on the states or attributes of your Home Assistant entities. See [Using Jinja2 Templates in Service Data JSON](#Advanced-configuration) for more information.

![img.png](doc/entity_settings.png)

### Advanced configuration
* **Custom Title:** Enable to override the main Title of this button. **You have to clear the main Title field on top to make this work!**
* Title Template: A [nunjucks template](https://mozilla.github.io/nunjucks/templating.html) that will be used as the button's title. You can use any of the variables (depending on the selected entity) that are shown below the text-field. For example `{{temperature}}°C` or `{{friendly_name}}` or (this won't fit the button, but you get the idea) `The pressure is {{pressure}} and the wind speed is {{wind_speed}}.`
* The variable `{{state}}` always contains the "main state" of an entity (for example "on" and "off" for buttons or "12.4" for temperature sensors)
* The variable `{{unit_of_measurement}}` often contains the ... unit of measurement ... of a sensor's state (for example "°C" for a temperature sensor)

* **Custom Labels:** Every button can display up to 4 lines of information
* Each line in the text-box represents one line on the button
* Depending on if there is an icon or a title for the entity, you may need to leave blank lines in order to not mess up the layout :)
* You can use [nunjucks template](https://mozilla.github.io/nunjucks/templating.html) for dynamic content (see above).
- **Custom Title:** Enable to override the main Title of this button. **You have to clear the main Title field on top to make this work!**

- Title Template: A [nunjucks template](https://mozilla.github.io/nunjucks/templating.html) that will be used as the button's title. You can use any of the variables (depending on the selected entity) that are shown below the text-field. For example `{{temperature}}°C` or `{{friendly_name}}` or (this won't fit the button, but you get the idea) `The pressure is {{pressure}} and the wind speed is {{wind_speed}}.`
- The variable `{{state}}` always contains the "main state" of an entity (for example "on" and "off" for buttons or "12.4" for temperature sensors)
- The variable `{{unit_of_measurement}}` often contains the ... unit of measurement ... of a sensor's state (for example "°C" for a temperature sensor)

- **Custom Labels:** Every button can display up to 4 lines of information

- Each line in the text-box represents one line on the button
- Depending on if there is an icon or a title for the entity, you may need to leave blank lines in order to not mess up the layout :)
- You can use [nunjucks template](https://mozilla.github.io/nunjucks/templating.html) for dynamic content (see above).

After you hit the save button, the button should immediately show the new configuration.

![img.png](doc/custom_labels.png)

* **Using Jinja2 Templates in Service Data JSON**
* **Jinja2 Template Integration:** You can incorporate Jinja2 templates within the Service Data JSON to dynamically
process data based on the states or attributes of your Home Assistant entities.
* **Encapsulation with Raw Tags:** It's crucial to enclose Jinja2 templates within `{% raw %}` and `{% endraw %}`
tags. This encapsulation ensures that the StreamDeck plugin processes these templates as Jinja2, distinct from any
Nunjucks templates you might use elsewhere in your configurations.
* **Example of Jinja2 Template Usage:**
- **Using Jinja2 Templates in Service Data JSON**

- **Jinja2 Template Integration:** You can incorporate Jinja2 templates within the Service Data JSON to dynamically
process data based on the states or attributes of your Home Assistant entities.
- **Encapsulation with Raw Tags:** It's crucial to enclose Jinja2 templates within `{% raw %}` and `{% endraw %}`
tags. This encapsulation ensures that the StreamDeck plugin processes these templates as Jinja2, distinct from any
Nunjucks templates you might use elsewhere in your configurations.
- **Example of Jinja2 Template Usage:**

```json
{
"temperature": "{% raw %}{{ state_attr('climate.ff_office_heating','temperature') + 0.5 }}{% endraw %}"
}
```
```json
{
"temperature": "{% raw %}{{ state_attr('climate.ff_office_heating','temperature') + 0.5 }}{% endraw %}"
}
```

# Happy? Consider donating me a coffee :)
[![buy me a coffee](https://www.paypalobjects.com/en_US/i/btn/btn_donate_SM.gif)](https://www.paypal.com/donate?hosted_button_id=3UKRJEJVWV9H4)

[![buy me a coffee](https://www.paypalobjects.com/en_US/i/btn/btn_donate_SM.gif)](https://www.paypal.com/donate?hosted_button_id=3UKRJEJVWV9H4)

[https://www.reddit.com/r/homeassistant/comments/laq2g4/homeassistant_streamdeck_plugin_dynamic_not_just/glu0zep/?utm_source=share&utm_medium=web2x&context=3]: https://www.reddit.com/r/homeassistant/comments/laq2g4/homeassistant_streamdeck_plugin_dynamic_not_just/glu0zep/?utm_source=share&utm_medium=web2x&context=3

[ff]: https://www.reddit.com/r/homeassistant/comments/laq2g4/homeassistant_streamdeck_plugin_dynamic_not_just/glu0zep/?utm_source=share&utm_medium=web2x&context=3

[https://github.com/cgiesche/streamdeck-homeassistant/releases]: https://github.com/cgiesche/streamdeck-homeassistant/releases
15 changes: 3 additions & 12 deletions doc/services.json
Original file line number Diff line number Diff line change
Expand Up @@ -479,10 +479,7 @@
"snapshot_entities": {
"name": "Snapshot entities",
"description": "The entities of which a snapshot is to be taken",
"example": [
"light.ceiling",
"light.kitchen"
],
"example": ["light.ceiling", "light.kitchen"],
"selector": {
"object": null
}
Expand Down Expand Up @@ -2180,11 +2177,7 @@
"default": 0,
"selector": {
"select": {
"options": [
"0",
"1",
"2"
]
"options": ["0", "1", "2"]
}
}
},
Expand Down Expand Up @@ -3875,9 +3868,7 @@
"default": "hls",
"selector": {
"select": {
"options": [
"hls"
]
"options": ["hls"]
}
}
}
Expand Down
23 changes: 23 additions & 0 deletions eslint.config.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,23 @@
import js from '@eslint/js'
import pluginVue from 'eslint-plugin-vue'
import path from 'node:path'
import skipFormattingConfig from '@vue/eslint-config-prettier/skip-formatting'
import eslintConfigPrettier from 'eslint-config-prettier'

import { includeIgnoreFile } from '@eslint/compat'
import { fileURLToPath } from 'node:url'

const __filename = fileURLToPath(import.meta.url)
const __dirname = path.dirname(__filename)
const gitignorePath = path.resolve(__dirname, '.gitignore')

export default [
js.configs.recommended,
...pluginVue.configs['flat/essential'],
includeIgnoreFile(gitignorePath),
eslintConfigPrettier,
skipFormattingConfig,
{
files: ['**/*.vue', '**/*.js', '**/*.jsx', '**/*.cjs', '**/*.mjs']
}
]
Loading
Loading