From 27dee2511367013c9dab4cc464b801b521ec2dd2 Mon Sep 17 00:00:00 2001 From: Johan Frick Date: Wed, 3 Jan 2024 08:31:43 +0100 Subject: [PATCH] feature: show audio input format always show in volumes section and optionally in player section --- README.md | 1 + src/components/player-controls.ts | 38 ++++++++++++++++++++++++++++++- src/editor/advanced-editor.ts | 4 ++++ src/sections/volumes.ts | 2 +- src/services/hass-service.ts | 4 ++-- src/types.ts | 1 + 6 files changed, 46 insertions(+), 4 deletions(-) diff --git a/README.md b/README.md index 01b6cb73..8cd39fe2 100644 --- a/README.md +++ b/README.md @@ -137,6 +137,7 @@ customSources: # Main use case is probably to set tv media player to play TV sou thumbnail: https://cdn-icons-png.flaticon.com/512/716/716429.png dynamicVolumeSlider: true # default is false. See more in section further down. artworkHostname: http://192.168.0.59:8123 #default is ''. Usually not needed, but depending on your setup your device might not be able to access the artwork on the default host. One example where it could be needed is if you cast the dashboard with Google Cast. +showAudioInputFormat: true # default is false. Will show the audio input format (e.g. Dolby Digital) in the player section if available. By default, it will only show if the input format in the volumes section. # media browser specific mediaBrowserItemsPerRow: 4 # default is 1. Use this to show items as icons. diff --git a/src/components/player-controls.ts b/src/components/player-controls.ts index 95c52ec1..b4df513a 100755 --- a/src/components/player-controls.ts +++ b/src/components/player-controls.ts @@ -6,11 +6,12 @@ import { CardConfig, MediaPlayerEntityFeature } from '../types'; import { mdiVolumeMinus, mdiVolumePlus } from '@mdi/js'; import { MediaPlayer } from '../model/media-player'; import { when } from 'lit/directives/when.js'; +import { until } from 'lit-html/directives/until.js'; const { SHUFFLE_SET, REPEAT_SET, PLAY, PAUSE, NEXT_TRACK, PREVIOUS_TRACK } = MediaPlayerEntityFeature; class PlayerControls extends LitElement { - @property({attribute: false}) store!: Store; + @property({ attribute: false }) store!: Store; private config!: CardConfig; private activePlayer!: MediaPlayer; private mediaControlService!: MediaControlService; @@ -27,11 +28,15 @@ class PlayerControls extends LitElement { this.activePlayer.state !== 'idle', () => html`
+
+
+ ${this.config.showAudioInputFormat && until(this.getAudioInputFormat())} +
`, )} @@ -42,6 +47,13 @@ class PlayerControls extends LitElement { private volDown = async () => await this.mediaControlService.volumeDown(this.activePlayer); private volUp = async () => await this.mediaControlService.volumeUp(this.activePlayer); + private async getAudioInputFormat() { + const sensors = await this.store.hassService.getRelatedEntities(this.activePlayer, 'sensor'); + const audioInputFormat = sensors.find((sensor) => sensor.entity_id.includes('audio_input_format')); + return audioInputFormat && audioInputFormat.state && audioInputFormat.state !== 'No audio' + ? html`
${audioInputFormat.state}
` + : ''; + } static get styles() { return css` .main { @@ -61,6 +73,30 @@ class PlayerControls extends LitElement { --mdc-icon-button-size: 5rem; --mdc-icon-size: 5rem; } + .audio-input-format { + flex: 1 0 0; + margin-bottom: 10px; + text-align: center; + align-self: stretch; + position: relative; + } + .audio-input-format > div { + color: var(--card-background-color); + background: var(--disabled-text-color); + text-overflow: ellipsis; + overflow: hidden; + white-space: nowrap; + position: absolute; + bottom: 0; + right: 0; + max-width: 100%; + font-size: smaller; + line-height: normal; + padding: 3px; + } + .flex-1 { + flex: 1; + } `; } } diff --git a/src/editor/advanced-editor.ts b/src/editor/advanced-editor.ts index 3ed7e9d0..e0edff25 100644 --- a/src/editor/advanced-editor.ts +++ b/src/editor/advanced-editor.ts @@ -40,6 +40,10 @@ export const ADVANCED_SCHEMA = [ name: 'numberOfFavoritesToShow', valueMin: 1, }, + { + name: 'showAudioInputFormat', + selector: { boolean: {} }, + }, ]; class AdvancedEditor extends BaseEditor { diff --git a/src/sections/volumes.ts b/src/sections/volumes.ts index 406824fe..e4ebf52a 100755 --- a/src/sections/volumes.ts +++ b/src/sections/volumes.ts @@ -74,7 +74,7 @@ class Volumes extends LitElement { if (hide) { return; } - const relatedEntities = await this.hassService.getRelatedEntities(player); + const relatedEntities = await this.hassService.getRelatedEntities(player, 'switch', 'number', 'sensor'); return relatedEntities.map((relatedEntity: HassEntity) => { relatedEntity.attributes.friendly_name = relatedEntity.attributes.friendly_name?.replaceAll(player.name, '')?.trim() ?? ''; diff --git a/src/services/hass-service.ts b/src/services/hass-service.ts index 5ac99249..ea935c55 100644 --- a/src/services/hass-service.ts +++ b/src/services/hass-service.ts @@ -37,7 +37,7 @@ export default class HassService { return mediaPlayerItem; } - async getRelatedEntities(player: MediaPlayer) { + async getRelatedEntities(player: MediaPlayer, ...entityTypes: string[]) { return new Promise(async (resolve, reject) => { const subscribeMessage = { type: 'render_template', @@ -48,7 +48,7 @@ export default class HassService { unsubscribe(); resolve( response.result - .filter((item: string) => item.includes('switch') || item.includes('number')) + .filter((item: string) => entityTypes.some((type) => item.includes(type))) .map((item) => this.hass.states[item]), ); }, subscribeMessage); diff --git a/src/types.ts b/src/types.ts index c341b711..8b9722d6 100644 --- a/src/types.ts +++ b/src/types.ts @@ -44,6 +44,7 @@ export interface CardConfig extends LovelaceCardConfig { topFavorites?: string[]; numberOfFavoritesToShow?: number; hideBrowseMediaButton?: boolean; + showAudioInputFormat?: boolean; } export interface MediaArtworkOverride {