A Lovelace custom card for showing images stored in Media Source and, optionally, toggle an entity when clicked.
The cards included in Home Assistant Lovelace for showing images only allow you to show images that are publicly accessible (i.e: url of an image from the internet). This is a privacy concern to me as I want to show pictures of my house and garden and even members of my family.
This cards allows you to store your images in a the media source module of Home Assistant and show them in Lovelace dashboards without having to make them public.
It can also toggle an entity if provided in configuration, like a light, switch, etc.
- Download the media-source-image-card
- Place the file in your config/wwwfolder
- Include the card as a resource through the UI (panels page) or using some code like this:
title: Home
resources:
  - url: /local/button-card.js
    type: module- If not already there, upload an image going to the Media Browser page in Home Assistant, inside the "My media" folder.
- Include the card code dashboard by selecting the card type itself or just a manual card:
  type: custom:media-source-image-card
  image: media-source://media_source/local/my_image.jpg # put your image name here
  entity_id: switch.my_light_switch  # this entity is optionalThis card is available as a HACS custom repository, you can follow HACS guide for custom repos, basically these are the steps to follow:
- Go to the frontend section in your Home Assistant's HACS page.
- Click on the 3 dots in the top right corner.
- Select "Custom repositories"
- Add the URL of this repo.
- Select the correct category.
- Click the "ADD" button.
This card is quite simple so there are only a few options:
| Name | Type | Default | Description | 
|---|---|---|---|
| type | string | custom:media-source-image-card | REQUIRED | 
| image | string | REQUIRED The path to the image in media source format. i.e: media-source://media_source/local/my_image.jpg | |
| aspect_ratio | number | '1.9' | Image aspect ratio | 
| object_fit | string | 'contain' | The CSS object-fit property for resizing and fitting the image. Possible values are: cover,contain,fill, etc. | 
| object_position | string | '50% 50%' | The CSS object-position property for placing the image. | 
| entity_id | string | The entity you want to toggle when card is clicked | |
| apply_grayscale | boolean | If trueapplies a grayscale on the image when entity isoff | |
| forced_refresh_interval | integer | Number of seconds to force an image refresh | |
| tap_action | string | toggle | Action to perform on click. One of none,toggleorcall-service. See actions below | 
By default, when the card is clicked, it would toggle the entity. You can customize these action to be one of this:
- none: nothing happens
- toggle: the entity is toggled (default behaviour)
- call-service: a service is called with a custom configuration
This is a simple call service example:
type: custom:media-source-image-card
image: media-source://media_source/local/banana.png
entity_id: input_boolean.boolean_test
apply_grayscale: true
tap_action:
  action: call-service
  service: input_boolean.turn_on
  target:
    entity_id: input_boolean.boolean_testIf no target is provided, the entity_id field is used.
You can use templates in the image field. In fact, you can use jinja2 and javascript templates.
The main difference is that jinja templates are rendered in the server and tend to hit perfomance overall while javascript templates run locally in the browser and need no additional messaging with the server.
Personally, I prefer javascript templates :)
You can use jinja templates like this:
type: custom:media-source-image-card
image: |
  media-source://media_source/local/{{ states('input_text.image') }}
entity_id: input_boolean.boolean_test
apply_grayscale: true
tap_action:
  action: call-service
  service: input_boolean.turn_on
  target:
    entity_id: input_boolean.boolean_testYou can also provide a javascript template like this:
type: custom:media-source-image-card
image: |
  [[[
    return `media-source://media_source/local/${ hass.states['input_text.texting'].state }`;
  ]]]
entity_id: input_boolean.boolean_test
apply_grayscale: true
tap_action:
  action: call-service
  service: input_boolean.turn_on
  target:
    entity_id: input_boolean.boolean_testYou need to enclose your function between triple brackets ('[[[' and ']]]'). The variables available in the function are:
| Variable | Description | 
|---|---|
| hass | The whole hass object where you can access everything | 
| states | The states object inside hass | 
| user | The object with information about the logged user | 
| config | The card config itself | 
This example only shows an image:
  type: custom:media-source-image-card
  image: media-source://media_source/local/20230919_145924.jpgShows and image. When clicked and toggles a switch on/off:
type: custom:media-source-image-card
image: media-source://media_source/local/20230919_145924.jpg
entity_id: switch.light_switchShows and image. When clicked, toggles a switch on/off and applies grayscale when it's off:
type: custom:media-source-image-card
image: media-source://media_source/local/20230919_145924.jpg
entity_id: switch.light_switch
apply_grayscale: truetype: custom:media-source-image-card
image: media-source://media_source/local/banana.png
entity_id: input_boolean.boolean_test
apply_grayscale: true
tap_action:
  action: call-service
  service: input_boolean.turn_on
  target:
    entity_id: input_boolean.boolean_testThis will refresh the image every 10 seconds, resolving the template again and showing a random image in a X.png format, where X is a number.
type: custom:media-source-image-card
image: |
  media-source://media_source/local/{{ '%02d' % range(1, 20) | random }}.png
entity_id: input_boolean.boolean_test
apply_grayscale: true
forced_refresh_interval: 10
tap_action:
  action: call-service
  service: input_boolean.turn_on
  target:
    entity_id: input_boolean.boolean_test

