Custom HACS plugin that allows you to personalise the Home Assistant's sidebar per user or device basis
Default sidebar | Hide/add some items | Reorder items | Personalise |
---|---|---|---|
|
|
|
|
This is a refactor of custom-sidebar-v2 by @galloween, which unfortunately is now unmaintained and archived. In its beginning, @galloween's code was a refactor of the original Custom Sidebar plugin by @Villhellm (R.I.P.). This version refactored completely @galloween's code allowing to use a configuration in YAML
(as @Villhellm's one) or in JSON
(as @galloween's version), fixing several bugs, improving performance, and using home-assistant-query-selector to make it less likely to break with future Home Assistant front-end updates.
You can install the plugin manually or through HACS, not both. If you install the plugin using the two installations methods you could have issues or errors.
- Go to
HACS
dashboard - Search for
custom-sidebar
and click on it - On the plugin page, click on the
Download
yellow button in the bottom-right corner - Click on
Download
in the more-info dialog - When the plugin is already downloaded, add the url of the plugin as an extra_module_url in your
configuration.yaml
:
frontend:
extra_module_url:
- /hacsfiles/custom-sidebar/custom-sidebar-yaml.js
frontend:
extra_module_url:
- /hacsfiles/custom-sidebar/custom-sidebar-json.js
Important
Do not get creative with the path of the plugin, use the provided one. If you have HACS
installed the hacsfiles custom path will be there even if you cannot see it in your filesystem.
- Restart Home Assistant
- Download the latest custom-sidebar release
- Copy
custom-sidebar-yaml.js
orcustom-sidebar-json.js
into<config directory>/www/
(depending on the configuration format that you are going to use,YAML
orJSON
) - Add the url of the plugin as an extra_module_url in your
configuration.yaml
:
frontend:
extra_module_url:
- /local/custom-sidebar-yaml.js?v1.0.0
frontend:
extra_module_url:
- /local/custom-sidebar-json.js?v1.0.0
- Restart Home Assistant
Tip
It is recommendable that you add the plugin's version at the end of the URL as a parameter (e.g. ?v=1.0.0
) to force Home Assistant to load the new version instead of a version stored in cache. This step is not mandatory, but if you start suffering from caching issues, this would be one of the first steps to perform.
Depending on the file that you have added to extra_module_url, you will need to add your configuration in YAML
or JSON
format. If you used custom-sidebar-yaml.js
you need to provide the configuration in YAML
format. If you have used custom-sidebar-json.js
you need to provide the configuration in JSON
format.
Tip
In principle, using a YAML
or a JSON
configuration is a matter of taste and preference. If you don't plan to use templates, there is no advantage of one over the other, but if you plan to use templates, specially complex ones, YAML
is recommended because you can write code blocks in multiple lines. As JSON
doesn't allow line breaks, you would need to write the complete template in a single line.
Add a file named sidebar-config.yaml
or sidebar-config.json
into your <config directory>/www/
directory. It is recommendable that you copy the example sidebar-config.yaml or the example sidebar-config.json file, delete the id
parameter, and modify it to match your needs. It is recommended to start with the simplest configuration and start to extend it step by step, in that way if you suffer any issue, it is easy to identify with which code block it started.
Tip
If you want to check an extensive sidebar configuration which uses custom styles, variables and templates, check @Mariusthvdb's sidebar-config.yaml.
Property | Type | Required | Description |
---|---|---|---|
order | Array of items | no | List of items to process |
hide_all | Boolean | no | Hides all items of the sidebar by default, useful if one wants to hide everything and just show a few items. (This options doesn't make any effect in an item with the property new_item in true ) |
title* | String | no | Custom title to replace the Home Assistant title |
title_color* | String | no | Sets the color of the sidebar title |
subtitle* | String | no | Sets a custom subtitle below the Home Assistant title |
subtitle_color* | String | no | Sets the color of the sidebar subtitle |
sidebar_background* | String | no | Sets the background of the sidebar. It could be a color or a background declaration |
sidebar_border_color* | String | no | Sets the border-right color of the sidebar |
menu_background* | String | no | Sets the background of the sidebar menu area (the one containing the menu button and the title). If it is not set, the sidebar_background option will be used. It could be a color or a background declaration |
sidebar_editable* | Boolean or String | no | If it is set to false, long press on the sidebar title will be ignored and the button to edit the sidebar in the profile panel will be disabled. As a string it should be a JavaScript or a Jinja template that returns a boolean or a promise that resolves in a boolean |
sidebar_mode | String | no | Defines the default status of the sidebar when Home Assistant is loaded. It has three possible values: "hidden" to make the sidebar hidden, "narrow" to make the sidebar visible in narrow state and "extended" to make sidebar visible in extended state. This option will show or hide the sidebar ignoring if it is a desktop or a mobile device or if the Always hide the sidebar switch in the profile page in on or off (depending on the value of this option, this switch will be switched on or off automatically) |
sidebar_button_color* | String | no | Sets the color of the sidebar hamburger menu |
item_background* | String | no | Sets the background of the sidebar items. It could be a color or a background declaration |
item_background_hover* | String | no | Sets the background of the sidebar items in a hover state. It could be a color or a background declaration |
icon_color* | String | no | Sets the color of the sidebar icons |
icon_color_selected* | String | no | Sets the icon color of the selected sidebar item |
icon_color_hover* | String | no | Sets the icon color of the sidebar items in a hover state |
text_color* | String | no | Sets the text color of the sidebar items |
text_color_selected* | String | no | Sets the text color of the selected sidebar item |
text_color_hover* | String | no | Sets the text color of the sidebar items in a hover state |
selection_background* | String | no | Sets the background of the selected item. It could be a color or a background declaration. If it is not specified, the icon_color_selected will be used |
selection_opacity* | Number or String | no | Sets the opacity of the selected item background. It should be a number between 0 (fully transparent) and 1 (fully opaque). If it is not specified, the default 0.12 will be used |
info_color* | String | no | Sets the color of the info texts of the sidebar items |
info_color_selected* | String | no | Sets the color of the info text of the selected sidebar item |
info_color_hover* | String | no | Sets the color of the info texts in a hover state |
notification_color* | String | no | Sets the color of the sidebar notifications |
notification_color_selected* | String | no | Sets the color of the sidebar notification of the selected sidebar item |
notification_color_hover* | String | no | Sets the color of the sidebar notification when the item is in hover state |
notification_text_color* | String | no | Sets the color of the sidebar notifications texts |
notification_text_color_selected* | String | no | Sets the color of the sidebar notification texts of the selected sidebar item |
notification_text_color_hover* | String | no | Sets the color of the sidebar notifications texts when the item is in hover state |
divider_color* | String | no | Sets the color of both sidebar dividers |
divider_top_color* | String | no | Sets the color of the top sidebar divider. It overrides divider_color for this divider if it is set |
divider_bottom_color* | String | no | Sets the color of the bottom sidebar divider. It overrides divider_color for this divider if it is set |
scrollbar_thumb_color* | String | no | Sets the color of the sidebar scrollbar (This option uses non-baseline CSS styles and it could not work in some browsers) |
default_path | String | no | Defines the default URL path when the page loads. This path must start with / |
styles | String | no | Custom styles that will be added to the styles block of the plugin. Useful to override styles |
Tip
* These options allow JavaScript or Jinja templates.
Important
These options are intended for advanced users. They are not strictly necessary and you can use the plugin without making use of them. The purpose of these options is to reduce code repetition and share configurations. It is advisable that you do not use them if they result confusing to you or if you don't understand their usage. Check the advanced configuration options usage section
Property | Type | Required | Description |
---|---|---|---|
js_variables | Object | no | An object containing variales that will be used in JavaScript templates Consult the variables section for more info |
jinja_variables | Object | no | An object containing variales that will be used in Jinja templates. Consult the variables section for more info |
partials | Object | no | An object containing fragments of code that can be included in your templates. Consult the partials section for more info |
extendable_configs | Object | no | An object containing extendable configurations. Consult the extendable configurations section for more info |
extend_from | String or String[] | no | Indicates if the configuration should extend from extendable configrations. Consult the extendable configurations section for more info |
Property | Type | Required | Description |
---|---|---|---|
item | String | yes | This is a string that will be used to match each sidebar item by its text, its data-panel attribute or its href . If the exact property is not set, it is case insensitive and it can be a substring such as developer instead of Developer Tools or KITCHEN instead of kitchen-lights |
match | String | no | This property will define which string will be used to match the item property. It has three possible values "text" (default) to match the text content of the element, "data-panel" to match the data-panel attribute of the element, or "href", to match the href attribute of the element |
exact | Boolean | no | Specifies whether the item string match should be an exact match (true ) or not (false ). |
order | Number | no | Sets the order number of the sidebar item |
hide* | Boolean or String | no | Setting this property to true will hide the sidebar item and if the property hide_all from the main configuration is true , setting this property as false will avoid hiding the item |
name* | String | no | Changes the name of the sidebar item |
icon* | String | no | Specifies the icon of the sidebar item |
info* | String | no | Sets the content of the info text (a smaller secondary text below the main item text) |
notification* | String | no | Add a notification badge to the sidebar item |
item_background* | String | no | Sets the background of the items. It could be a color or a background declaration (it overrides the global item_background ) |
item_background_hover* | String | no | Sets the background of the items in a hover state. It could be a color or a background declaration (it overrides the global item_background_hover ) |
icon_color* | String | no | Sets the color of the icon (it overrides the global icon_color ) |
icon_color_selected* | String | no | Sets the icon color of the item when it is selected (it overrides the global icon_color_selected ) |
icon_color_hover* | String | no | Sets the item icon color in a hover state (it overrides the global icon_color_hover ) |
text_color* | String | no | Sets the text color of the item (it overrides the global text_color ) |
text_color_selected* | String | no | Sets the text color of the item when it is selected (it overrides the global text_color_selected ) |
text_color_hover* | String | no | Sets the item text color in a hover state (it overrides the global text_color_hover ) |
selection_background* | String | no | Sets the background of the item when it is selected. It could be a color or a background declaration. If it is not specified, the icon_color_selected will be used |
selection_opacity* | Number or String | no | Sets the opacity of the item background when it is selected. It should be a number between 0 (fully transparent) and 1 (fully opaque). If it is not specified, the default 0.12 will be used (it overrides the global selection_opacity ) |
info_color* | String | no | Sets the color of the info text (it overrides the global info_color ) |
info_color_selected* | String | no | Sets the color of the info text when the item is selected (it overrides the global info_color_selected ) |
info_color_hover* | String | no | Sets the color of the info text in a hover state (it overrides the global info_color_hover ) |
notification_color* | String | no | Sets the notification color (it overrides the global notification_color ) |
notification_color_selected* | String | no | Sets the color of the item notification when it is selected (it overrides the global notification_color_selected ) |
notification_color_hover* | String | no | Sets the color of the item notification in hover state (it overrides the global notification_color_hover ) |
notification_text_color* | String | no | Sets the color of the notification text (it overrides the global notification_text_color ) |
notification_text_color_selected* | String | no | Sets the color of the notification text when it is selected (it overrides the global notification_text_color_selected ) |
notification_text_color_hover* | String | no | Sets the color of the notifications text in hover state (it overrides the global notification_text_color_hover ) |
bottom | Boolean | no | Setting this property to true will group the item with the bottom items (Configuration, Developer Tools, etc) |
href | String | no | Specifies the href of the sidebar item |
target | String | no | Specifies the target property of the sidebar item |
on_click | OnClickAction | no | Specifies the onClick property of the sidebar item. It allows two types of actions, ServiceCallAction or JavaScriptAction . Take into account that setting this property will not stop the href option for working. If you want to avoid navigating to a page, you should set the href option as # (in new items you can just omit it).## service call action |
new_item | Boolean | no | Set this property to true to create a new item in the sidebar. Using this option makes icon a required property together with href or on_click |
Tip
* These item properties allow JavaScript or Jinja templates.
Short example in YAML
format:
title: My Home
icon_color_selected: var(--accent-color)
order:
- new_item: true
item: Google
href: https://mrdoob.com/projects/chromeexperiments/google-gravity/
icon: mdi:earth
target: _blank
order: 1
- item: overview
order: 2
- new_item: true
item: Automations
href: "/config/automation"
icon: mdi:robot
info: |
{{ states.automation | selectattr('state', 'eq', 'on') | list | count }} active
order: 3
Short example in JSON
format:
{
"title": "My Home",
"icon_color_selected": "var(--accent-color)",
"order": [
{
"new_item": true,
"item": "Google",
"href": "https://mrdoob.com/projects/chromeexperiments/google-gravity/",
"icon": "mdi:earth",
"target": "_blank",
"order": 1
},
{
"item": "overview",
"order": 2
},
{
"new_item": true,
"item": "Automations",
"href": "/config/automation",
"icon": "mdi:robot",
"info": "{{ states.automation | selectattr('state', 'eq', 'on') | list | count }} active",
"order": 3
}
]
}
Important
- All items in
config.order
should have a uniqueitem
property - Avoid an
item
property that could match multiple elements. If you find that an item property matches with multiple elements, try to use thematch
andexact
properties to match the specific item that you want to match. - The items will be ordered according to their
order
property OR in the order of appearance - If you use the
order
property in at least one item, make sure either all items (except hidden ones) have this property, or none of them (otherwise order may be messed up) - All the items placed in the bottom will be moved to the top by default. If you want to have some items in the bottom you need to add them to the
config.order
and specify theirbottom
property ontrue
. - Any items present in the Sidebar, but not in
config.order
, will be shown on the bottom of the top part of the list - Notifications and user account are not part of the main sidebar items so they cannot be targeted inside the
order
option to change their properties. On the other hand, global color properties will affect these items though. - The
default_path
option will change the default behaviour and every time that the page loads it will navigate to this path (either when the page loads for the first time or when it gets refreshed). If you don't want to have this behaviour and you would prefer to load Home Assistant in an specific path or refresh a specific page without being redirected to thedefault_path
, then you should not set this option. - The
style
option doesn't allow templates, it should be used only to override or correct some styles of the sidebar. - If you use
custom-sidebar
to set the order or to hide items from the sidebar, it is recommended that you don't use Home Assistant functionality to reorder/hide items, because it will conflict with the functionality of the plugin. - Take into account that using the
on_click
parameter in an item will not stop redirecting to a page if thehref
parameter has a value. If you want to stop navigating to a page when you click on an item, for new items you can just omit thehref
parameter, but for existing items you should set thehref
parameter as#
. - Take into account that the
code
parameter of anjavascript
action in theon_click
parameter doesn't need to be enclosed between three square brackets ([[[ ]]]
).
You can define user/device-specific options using the exceptions feature. Exceptions can be used if you would like to define different options for a specific user/device.
In an exception you can define almost all the options available in the main configuration options (excluding the advanced options js_variables
, jinja_variables
, partials
, and extendable_configs
). And on top of those options, the next ones will be available:
Property | Type | Required | Description |
---|---|---|---|
user | String or String[] | no | Home Assistant user name(s) you would like to display this order for. This option can be set alone or combined with device , not_device , is_admin , or is_owner |
not_user | String or String[] | no | Home Assistant user name(s) you wouldn't like to display this order for. This option can be set alone or combined with device , not_device , is_admin , or is_owner |
device | String or String[] | no | Device(s) you would like to display this order for. E.g. ipad, iphone, macintosh, windows, android (it uses the device's user-agent). This option can be set alone or combined with user , not_user , is_admin , or is_owner |
not_device | String or String[] | no | Device(s) you wouldn't like to display this order for. E.g. ipad, iphone, macintosh, windows, android (it uses the device's user-agent). This option can be set alone or combined with user , not_user is_admin , or is_owner |
is_admin | Boolean | no | Checks if the user is admin or not. This option can be set alone or combined with user , not_user , device , not_device , or is_owner |
is_owner | Boolean | no | Checks if the user is owner of the system. This option can be set alone or combined with user , not_user , device , not_device , or is_admin |
matchers_conditions | "OR" | "AND" | no | Defines if at least one of the matchers should match ("OR" which is the default value) or if all the matchers should match ("AND") |
Short example in YAML
format:
...
exceptions:
- user:
- Jim Hawkins
- Long John Silver
extend_from: base
title: My Home
order:
...
- not_user:
- John Doe
- Jack Sparrow
is_admin: true
matchers_conditions: AND
order:
...
- not_device: Android
order:
...
- is_admin: true
order:
...
Short example in JSON
format:
{
...
"exceptions": [
{
"user": ["Jim Hawkins", "Long John Silver"],
"extend_from": "base",
"title": "My Home",
"order": [
...
]
},
{
"not_user": ["John Doe", "Jack Sparrow"],
"is_admin": true,
"matchers_conditions": "AND",
"order": [
...
]
},
{
"not_device": "Android",
"order": [
...
]
},
{
"is_admin": true,
"order": [
...
]
}
]
}
Important
- If multiple exeptions match, their options will be merged from top to bottom and if multiple configurations have an
order
property, it will be merged too. Check the extendable configurations section for more info about how two configurations are merged. - If
matchers_conditions
is "OR" (default value), only one match from the matchers is needed for the exception to be picked. But if this option is "AND" then all of the matcher must match for the exception to be picked, in this case if any matcher does't match, then the exception will be discarded. - You cannot use
user
andnot_user
at the same time, doing so will end in an error - You cannot use
device
andnot_device
at the same time, doing so will end in an error - In exceptions it is possible to use the
extend_from
option with the valuebase
. If you use it with this value, the main configuration will be merged with the one in the exceptions. Consult the extendable configurations section to know how is the process of extending configurations.
Some config options and item properties admit templates. custom-sidebar
admits two templating systems, JavaScript templates or Jinja templates. JavaScript
templates are processed faster because the rendering is done in client side, Jinja
templates need to perform a websocket call to receive the template result, but in general you should not notice many differences between the two in terms of performance. The main differences between the two templating systems (apart from the syntax) are that JavaScript
can access client side data like DOM APIs, can return Promises and can work with reactive variables. Jinja
templates are mostly agnostic to the device in which Home Assistant
is being executed, cannot return Promises and do not support reactive variables.
This templating system IS NOT the same that Home Assistant implements. It is basically a JavaScript
code block in which you can use certain client-side objects, variables and methods. To set a property as a JavaScript
template block, include the code between three square brackets [[[ JavaScript code ]]]
. If you don‘t use the square brackets, the property will be interpreted as a regular string.
The JavaScript
code will be taken as something that you want to return, but if you have a more complex logic, you can create your own variables and return the desired result at the end.
The entities used in the templates will be stored, so if the state of an stored entity changes, all the templates that use this entity will be reevaluated snd rerendered. On top of this, if the variable panel_url
is used in a template, the template will be reevaluated every time that a new panel or a new view is loaded.
JavaScript
templates can make use of reactive variables. If you use a reactive variable in some templates and you modify it in the javascript
action of an on_click
parameter of an item, all the templates in which the variable was used will be reevaluated in the device in which the action is performed.
The next example will set the next options:
- Sets the title of the sidebar as "My Home" followed by the current time.
- Sets the background of the sidebar
red
when the panel config is selected andgreen
otherwise. - Adds the number of
HACS
updates as a notification in theHACS
item in the sidebar. In case that there are no updates, an empty string is returned and in these cases the notification will not be displayed. - Creates a new item that redirects to the
Home Assistant
info page with a dynamic text with the word "Info" followed by the installed Supervisor version between parentheses and the Operating System version in the info text.
title: '[[[ "My Home " + new Date(states("sensor.date_time_iso")).toLocaleTimeString().slice(0, 5) ]]]'
sidebar_background: |
[[[
return panel_url === '/config/dashboard'
? 'red'
: 'green';
]]]
order:
- item: hacs
notification: |
[[[
const outdatedHacsEntities = Object.values(entities.update).filter(
(entity) => entity.platform === 'hacs' && is_state(entity.entity_id, 'on')
);
return outdatedHacsEntities.length || '';
]]]
- new_item: true
item: info
name: '[[[ "Info (" + state_attr("update.home_assistant_supervisor_update", "latest_version") + ")" ]]]'
info: '[[[ return "OS " + state_attr("update.home_assistant_operating_system_update", "latest_version") ]]]'
href: '/config/info'
icon: mdi:information-outline
{
"title": "[[[ 'My Home ' + new Date(states('sensor.date_time_iso')).toLocaleTimeString().slice(0, 5) ]]]",
"sidebar_background": "[[[ return panel_url === '/config/dashboard' ? 'red' : 'green' ]]]",
"order": [
{
"item": "hacs",
"notification": "[[[ return Object.values(entities.update).filter((entity) => entity.platform === 'hacs' && is_state(entity.entity_id, 'on')).length || '' ]]]"
},
{
"new_item": true,
"item": "info",
"name": "[[[ 'Info (' + state_attr('update.home_assistant_supervisor_update', 'latest_version') + ')' ]]]",
"info": "[[[ return 'OS ' + state_attr('update.home_assistant_operating_system_update', 'latest_version') ]]]",
"href": "/config/info",
"icon": "mdi:information-outline"
}
]
}
Tip
Custom Sidebar
uses Home Assistant Javascript Templates for the JavaScript
templating system. To know all the objects, variables and methods available in the JavaScript
templates, consult the proper section in the repository.
This templating system is the same that Home Assistant implements. You can use the majority of the template methods and objects. The entire template will be processed, rendered and the result will be used as the desired option. If you don‘t want to display anything in certain scenarios, you should return an empty string in those cases. The next client side varianles will be available in Jinja templates
:
user_name
: String with the logged user's nameuser_is_admin
: Bolean value than indicates if the logged user is admin or notuser_is_owner
: Bolean value than indicates if the logged user is the owner or notuser_agent
: User agent of the browser in which Home Assistant is being executed
When the entities used in a templates change their state, it will trigger an update and the updated result of the template will be rendered.
The next example will set the next options:
- Sets the title of the sidebar as "My Home" followed by the current time.
- Adds the number of
HACS
updates as a notification in theHACS
item in the sidebar. In case that there are no updates, an empty string is returned and in these cases the notification will not be displayed. - Creates a new item that redirects to the
Home Assistant
info page with a dynamic text with the word "Info" followed by the installed Supervisor version between parentheses and the Operating System version in the info text.
title: 'My Home {{ as_timestamp(states("sensor.date_time_iso")) | timestamp_custom("%H:%M") }}'
order:
- item: hacs
notification: |
{{
expand(states.update)
| selectattr('state', 'eq', 'on')
| map(attribute='entity_id')
| map('device_attr', 'identifiers')
| map('contains', 'hacs')
| list
| length or ""
}}
- new_item: true
item: info
name: 'Info ({{ state_attr("update.home_assistant_supervisor_update", "latest_version") }})'
info: 'OS {{ state_attr("update.home_assistant_operating_system_update", "latest_version") }}'
href: '/config/info'
icon: mdi:information-outline
{
"title": "My Home {{ as_timestamp(states('sensor.date_time_iso')) | timestamp_custom('%H:%M') }}",
"order": [
{
"item": "hacs",
"notification": "{{ expand(states.update) | selectattr('state', 'eq', 'on') | map(attribute='entity_id') | map('device_attr', 'identifiers') | map('contains', 'hacs') | list | length or '' }}"
},
{
"new_item": true,
"item": "info",
"name": "Info ({{ state_attr('update.home_assistant_supervisor_update', 'latest_version') }})",
"info": "OS {{ state_attr('update.home_assistant_operating_system_update', 'latest_version') }}",
"href": "/config/info",
"icon": "mdi:information-outline"
}
]
}
Important
Custom Sidebar
's advanced configurations options are intended for advanced users. They are not strictly necessary and you can use the plugin without making use of them. The purpose of these options is to reduce code repetition and share configurations. It is advisable that you do not use them if they result confusing to you or if you don't understand their usage.
js_variables
and jinja_variables
are an objects to declare variables intended to be used inside JavaScript or Jinja templates respectively. These objects can store string
, number
, and boolean
variables as well as dictionaries
(or objects
) and lists
(or arrays
) and in the case of js_templates
they can also declare reactive variables (setting a string with the variable wrapped in a ref()
method). These variables will be available in any partial or in any template.
Important
jinja_variables
only allowsstring
,number
andboolean
variables as well asdictionaries
andlists
containing otherdictionaries
orlists
or the aforementioned primitives. In the case ofjs_variables
, they allow the same type of variables as well as reactive variables. Trying to send other kind of variables will end in an error.js_variables
andjinja_variables
don't compile any template string. So if you set a variable as a template string, it will be interpreted as a string and a warning will be thrown in the console.
The next examples using js_variables
and jinja_variables
will set the same title ("Title 80 dog"):
js_variables:
my_string: Title
my_number: 100
my_boolean: true
my_object:
prop1: 4
prop2: 5
my_array:
- cat
- dog
- bird
title: |
[[[
if (my_boolean) {
return `${my_string} ${(my_number * my_object.prop1) / my_object.prop2} ${my_array[1]}`;
}
return 'Home Assistant';
]]]
jinja_variables:
my_string: Title
my_number: 100
my_boolean: true
my_dictionary:
prop1: 4
prop2: 5
my_list:
- cat
- dog
- bird
title: |
{% if my_boolean %}
{{ my_string }} {{ ((my_number * my_dictionary.prop1) / my_dictionary.prop2) | int }} {{ my_list.1 }}
{% else %}
Home Assistant
{% endif %}
The next example will change the sidebar title every time that the item "Change Title" is clicked. It will set the title with the string "Title" followed by a number (it will start in 0
and it will be incrementing every time that the item is clicked).
js_variables:
title_number: ref(0)
title: |
[[[ return 'Title ' + ref('title_number').value; ]]]
order:
- new_item: true
item: Change Title
icon: mdi:gesture-tap
on_click:
action: javascript
code: |
ref('title_number').value++;
Important
- To make the templates detect that a reactive variable has been mutated, one needs to assign a new value to the reactive variable. For example, changing the items or an array using
push
orpop
will not make the remplates using that variable to be reevaluated. You need to assign a new array to the value of the reactive variable to make the change been detected. - To make the template aware that it contains a reactive variable, the
value
property of the variable should be accesed when the template is rendered. If the code accesing thevalue
property is not executed when the template renders, then the reactive variable will not be tracked. That is why is recomendable to access thevalue
property of the reactive variable outside any condition and build the logic using the retrieved value. In this way the template will track that the reactive variable is being used and any time that the variable changes, the template will get re-evaluated.
Partials are fragments of code that can be included in your templates. They can be inserted in JavaScript, Jinja templates or inside another partial. Any entity used in them will make the template in which the partial is inserted to be reevaluated when the entity changes its state, so it is not recommended to use a bloated partial using multiple entities that have no context with each other because that will provoke that the templates in which that partial is used get reevaluated when anyone of the entities used in it change.
Tip
Partials will automatically use the variables set in the js_variables
or jinja_variables
(depending on the kind of template in which they are inserted).
js_variables:
supervisor_update: update.home_assistant_supervisor_update
os_update: update.home_assistant_operating_system_update
partials:
supervisor_version: |
const supervisorVersion = state_attr(supervisor_update, "latest_version");
updates: |
@partial supervisor_version
const osVersion = state_attr(os_update, "latest_version");
order:
- new_item: true
item: info
name: |
[[[
@partial updates
return `Info ${supervisorVersion}`;
]]]
info: |
[[[
@partial updates
return `OS ${ osVersion }`
]]]
href: '/config/info'
icon: mdi:information-outline
jinja_variables:
supervisor_update: update.home_assistant_supervisor_update
os_update: update.home_assistant_operating_system_update
partials:
supervisor_version: |
{% set supervisorVersion = state_attr(supervisor_update, "latest_version") %}
updates: |
@partial supervisor_version
{% set osVersion = state_attr(os_update, "latest_version") %}
order:
- new_item: true
item: info
name: |
@partial updates
Info {{ supervisorVersion }}
info: |
@partial updates
OS {{ osVersion }}
href: '/config/info'
icon: mdi:information-outline
Extendable configurations (extendable_configs
) is an object containing different configurations options that could be extended from the main configuration, from the exceptions or from another extendable configuration, making them a very flexible option to share configuration blocks. To specify that a configuration should extend from an extendable configuration, the extend_from
option should be used specifying the extendable configuration name(s).
Extending from a configuration basically means "import what I don't already have", so if a configuration already has an option, it will prevail and it will not be overridden if the configuration is extended. For example, the next configuration has a main configuration extending from an extendable configuration named example
, let's analyse what will be the result of that extend.
title: Custom Title
order:
- item: overview
name: Dashboard
order: 3
- new_item: true
item: Integrations
href: "/config/integrations"
icon: mdi:puzzle
order: 2
extend_from: example
extendable_configs:
example:
title: My Home
subtitle: Assistant
order:
- item: overview
icon: mdi:monitor-dashboard
order: 0
- new_item: true
item: Google
href: https://mrdoob.com/projects/chromeexperiments/google-gravity/
icon: mdi:earth
target: _blank
order: 1
- As the
title
option is defined in the main configuration, it will not get thetitle
option from the extendable configuration. - As the
subtitle
option is not defined in the main configuration, it will be get from the extendable configuration. - As the main configuration and the extendable configuration both have an
order
option, it will be merged:- Both orders have an
overview
item, so it will be merged. As the main config order-item has also anorder
property, it will not be extended, but as the extendable order-item has anicon
property that doesn't exist in the main config order-item, it will be extended. - As the extendable order-item doesn't have a
name
property, it will remain there. - The
Integrations
doesn't exist in the extendable order so it will remain as it is. - The
Google
extendable item doesn't exist in the main config, so it will be extended.
- Both orders have an
The resulted main config after the extending process will be:
title: Custom Title
subtitle: Assistant
order:
- new_item: true
item: Google
href: https://mrdoob.com/projects/chromeexperiments/google-gravity/
icon: mdi:earth
target: _blank
order: 1
- new_item: true
item: Integrations
href: "/config/integrations"
icon: mdi:puzzle
order: 2
- item: overview
name: Dashboard
icon: mdi:monitor-dashboard
order: 3
It is possible to extend from multiple configurations and they will be extended in order, as shown in the next example:
extend_from:
- colors
- titles
extendable_configs:
colors:
icon_color: red
text_color: red
titles:
title: Custom Title
subtitle: Custom Subtitle
As already mentioned, an extendable configuration can extend from other extendable configurations:
title: Custom Title
extend_from: example
extendable_configs:
colorful:
title_color: red,
subtitle_color: blue
example:
subtitle: Assistant
extend_from: colorful
The resulted main config will be:
title: Custom Title
subtitle: Assistant
title_color: red,
subtitle_color: blue
In the case of exceptions, they can also extend from the main configuration if base
is used in the extend_from
option:
title: Custom Title
extend_from: example
extendable_configs:
colorful:
title_color: red,
subtitle_color: blue
example:
subtitle: Assistant
extend_from: colorful
exceptions:
- user:
- ElChiniNet
order:
- item: overview
name: Dashboard
icon: mdi:monitor-dashboard
order: 3
extend_from: base
So, the configuration for the user ElChiniNet
will be the same previous main config, plus an order with an order-item.
The next example is a more complex one extending from multiple configurations:
title: My Home
extend_from: admin_config
order:
- new_item: true
item: Google
href: https://mrdoob.com/projects/chromeexperiments/google-gravity/
icon: mdi:earth
target: _blank
order: 1
extendable_configs:
multicolor:
icon_color: red
icon_color_selected: blue
icon_color_hover: green
text_color: red
text_color_selected: blue
text_color_hover: green
admin_config:
order:
- new_item: true
item: Integrations
href: "/config/integrations"
icon: mdi:puzzle
order: 2
- new_item: true
item: Entities
href: "/config/entities"
icon: mdi:hexagon-multiple
order: 3
user_config:
extend_from: multicolor
hide_all: true
order:
- item: overview
hide: false
exceptions:
- is_admin: true
extend_from:
- multicolor
- admin_config
order:
- item: config
bottom: true
- user:
- ElChiniNet
- Palaus
etend_from: base
title: HA
- user:
- Jim Hawkins
- Long John Silver
extend_from: user_config
order:
- item: overview
name: Dashboard
Important
- You need to be careful of circular dependencies when extending configurations, if this is detected an error will be thrown
- You can only use
base
insideextend_from
if you are in an exception, trying to use it in the main config or in an extendable configuration will throw and error
Check out Home Assistant's native sidebar tools, maybe it will be enough for your needs.
- You can use Home Assistant's
panel_custom
integration to add internal links to the sidebar. Take a look at this tutorial. Official docs. - You can use Home Assistant's
Webpage dashboard
feature to add external URLs to Home Assistant's sidebar. Official docs. If you use Home Assistant'sWebpage dashboard
feature to add some wepage dashboards to your sidebar, then you can modify them as the regular ones adding them to theorder
property. - You can click and hold the Home Assistant header on top of the sidebar and then it will allow you to add/remove and re-order some of the items (but not add new custom ones). This feature is also accessible from your profile settings (if you click on your username in the bottom left corner). The drawback of this feature is that it only works per device/session, so you need to log-in in the specific device and do these changes, but if you log-out, the changes will be lost.
- Do you have an idea or a question? Open a discussion
- Do you have an issue or have you encountered a bug? Open an issue
- Do you need to discuss or to show and tell? visit the
custom-sidebar
entry in the Home Assistant Forum - Video tutorial about how to install the plugin (in Spanish) by @hectorzin
- Video tutorial about how to install the plugin (in German) by @Senbei123