Reduce media selector size (#26033)

This commit is contained in:
Paul Bottein 2025-07-02 18:08:31 +02:00 committed by Bram Kragten
parent 33ea02208a
commit 45e9c51525

View File

@ -24,6 +24,10 @@ const MANUAL_SCHEMA = [
{ name: "media_content_type", required: false, selector: { text: {} } }, { name: "media_content_type", required: false, selector: { text: {} } },
] as const; ] as const;
const INCLUDE_DOMAINS = ["media_player"];
const EMPTY_FORM = {};
@customElement("ha-selector-media") @customElement("ha-selector-media")
export class HaMediaSelector extends LitElement { export class HaMediaSelector extends LitElement {
@property({ attribute: false }) public hass!: HomeAssistant; @property({ attribute: false }) public hass!: HomeAssistant;
@ -84,7 +88,7 @@ export class HaMediaSelector extends LitElement {
(stateObj && (stateObj &&
supportsFeature(stateObj, MediaPlayerEntityFeature.BROWSE_MEDIA)); supportsFeature(stateObj, MediaPlayerEntityFeature.BROWSE_MEDIA));
const hasAccept = this.selector.media?.accept?.length; const hasAccept = this.selector?.media?.accept?.length;
return html` return html`
${hasAccept ${hasAccept
@ -100,7 +104,7 @@ export class HaMediaSelector extends LitElement {
.disabled=${this.disabled} .disabled=${this.disabled}
.helper=${this.helper} .helper=${this.helper}
.required=${this.required} .required=${this.required}
include-domains='["media_player"]' .includeDomains=${INCLUDE_DOMAINS}
allow-custom-entity allow-custom-entity
@value-changed=${this._entityChanged} @value-changed=${this._entityChanged}
></ha-entity-picker> ></ha-entity-picker>
@ -114,7 +118,7 @@ export class HaMediaSelector extends LitElement {
</ha-alert> </ha-alert>
<ha-form <ha-form
.hass=${this.hass} .hass=${this.hass}
.data=${this.value} .data=${this.value || EMPTY_FORM}
.schema=${MANUAL_SCHEMA} .schema=${MANUAL_SCHEMA}
.computeLabel=${this._computeLabelCallback} .computeLabel=${this._computeLabelCallback}
></ha-form> ></ha-form>
@ -122,21 +126,19 @@ export class HaMediaSelector extends LitElement {
: html` : html`
<ha-card <ha-card
outlined outlined
tabindex="0"
role="button"
aria-label=${!this.value?.media_content_id
? this.hass.localize("ui.components.selectors.media.pick_media")
: this.value.metadata?.title || this.value.media_content_id}
@click=${this._pickMedia} @click=${this._pickMedia}
@keydown=${this._handleKeyDown}
class=${this.disabled || (!this.value?.entity_id && !hasAccept) class=${this.disabled || (!this.value?.entity_id && !hasAccept)
? "disabled" ? "disabled"
: ""} : ""}
> >
<div <div class="content-container">
class="thumbnail ${classMap({ <div class="thumbnail">
portrait:
!!this.value?.metadata?.media_class &&
MediaClassBrowserSettings[
this.value.metadata.children_media_class ||
this.value.metadata.media_class
].thumbnail_ratio === "portrait",
})}"
>
${this.value?.metadata?.thumbnail ${this.value?.metadata?.thumbnail
? html` ? html`
<div <div
@ -180,6 +182,7 @@ export class HaMediaSelector extends LitElement {
) )
: this.value.metadata?.title || this.value.media_content_id} : this.value.metadata?.title || this.value.media_content_id}
</div> </div>
</div>
</ha-card> </ha-card>
`} `}
`; `;
@ -229,6 +232,13 @@ export class HaMediaSelector extends LitElement {
}); });
} }
private _handleKeyDown(ev: KeyboardEvent) {
if (ev.key === "Enter" || ev.key === " ") {
ev.preventDefault();
this._pickMedia();
}
}
static styles = css` static styles = css`
ha-entity-picker { ha-entity-picker {
display: block; display: block;
@ -243,41 +253,52 @@ export class HaMediaSelector extends LitElement {
} }
ha-card { ha-card {
position: relative; position: relative;
width: 200px; width: 100%;
box-sizing: border-box; box-sizing: border-box;
cursor: pointer; cursor: pointer;
transition: background-color 180ms ease-in-out;
min-height: 56px;
}
ha-card:hover:not(.disabled),
ha-card:focus:not(.disabled) {
background-color: var(--state-icon-hover-color, rgba(0, 0, 0, 0.04));
}
ha-card:focus {
outline: none;
} }
ha-card.disabled { ha-card.disabled {
pointer-events: none; pointer-events: none;
color: var(--disabled-text-color); color: var(--disabled-text-color);
} }
.content-container {
display: flex;
align-items: center;
padding: 8px;
gap: 12px;
}
ha-card .thumbnail { ha-card .thumbnail {
width: 100%; width: 40px;
height: 40px;
flex-shrink: 0;
position: relative; position: relative;
box-sizing: border-box; box-sizing: border-box;
transition: padding-bottom 0.1s ease-out; border-radius: 8px;
padding-bottom: 100%; overflow: hidden;
}
ha-card .thumbnail.portrait {
padding-bottom: 150%;
} }
ha-card .image { ha-card .image {
border-radius: 3px 3px 0 0; border-radius: 8px;
} }
.folder { .folder {
--mdc-icon-size: calc(var(--media-browse-item-size, 175px) * 0.4); --mdc-icon-size: 24px;
} }
.title { .title {
font-size: var(--ha-font-size-l); font-size: var(--ha-font-size-m);
padding-top: 16px;
overflow: hidden; overflow: hidden;
text-overflow: ellipsis; text-overflow: ellipsis;
margin-bottom: 16px;
padding-left: 16px;
padding-right: 4px;
padding-inline-start: 16px;
padding-inline-end: 4px;
white-space: nowrap; white-space: nowrap;
line-height: 1.4;
flex: 1;
min-width: 0;
} }
.image { .image {
position: absolute; position: absolute;
@ -290,13 +311,15 @@ export class HaMediaSelector extends LitElement {
background-position: center; background-position: center;
} }
.centered-image { .centered-image {
margin: 0 8px; margin: 4px;
background-size: contain; background-size: contain;
} }
.icon-holder { .icon-holder {
display: flex; display: flex;
justify-content: center; justify-content: center;
align-items: center; align-items: center;
width: 100%;
height: 100%;
} }
`; `;
} }