ha-core/homeassistant/helpers/icon.py

194 lines
5.9 KiB
Python

"""Icon helper methods."""
from __future__ import annotations
import asyncio
from collections.abc import Iterable
from functools import lru_cache
import logging
from typing import Any
from homeassistant.core import HomeAssistant, callback
from homeassistant.loader import Integration, async_get_integrations
from homeassistant.util.json import load_json_object
from .translation import build_resources
ICON_CACHE = "icon_cache"
_LOGGER = logging.getLogger(__name__)
@callback
def _component_icons_path(component: str, integration: Integration) -> str | None:
"""Return the icons json file location for a component.
Ex: components/hue/icons.json
If component is just a single file, will return None.
"""
domain = component.rpartition(".")[-1]
# If it's a component that is just one file, we don't support icons
# Example custom_components/my_component.py
if integration.file_path.name != domain:
return None
return str(integration.file_path / "icons.json")
def _load_icons_files(icons_files: dict[str, str]) -> dict[str, dict[str, Any]]:
"""Load and parse icons.json files."""
return {
component: load_json_object(icons_file)
for component, icons_file in icons_files.items()
}
async def _async_get_component_icons(
hass: HomeAssistant,
components: set[str],
integrations: dict[str, Integration],
) -> dict[str, Any]:
"""Load icons."""
icons: dict[str, Any] = {}
# Determine files to load
files_to_load = {}
for loaded in components:
domain = loaded.rpartition(".")[-1]
if (path := _component_icons_path(loaded, integrations[domain])) is None:
icons[loaded] = {}
else:
files_to_load[loaded] = path
# Load files
if files_to_load and (
load_icons_job := hass.async_add_executor_job(_load_icons_files, files_to_load)
):
icons |= await load_icons_job
return icons
class _IconsCache:
"""Cache for icons."""
__slots__ = ("_hass", "_loaded", "_cache", "_lock")
def __init__(self, hass: HomeAssistant) -> None:
"""Initialize the cache."""
self._hass = hass
self._loaded: set[str] = set()
self._cache: dict[str, dict[str, Any]] = {}
self._lock = asyncio.Lock()
async def async_fetch(
self,
category: str,
components: set[str],
) -> dict[str, dict[str, Any]]:
"""Load resources into the cache."""
if components_to_load := components - self._loaded:
# Icons are never unloaded so if there are no components to load
# we can skip the lock which reduces contention
async with self._lock:
# Check components to load again, as another task might have loaded
# them while we were waiting for the lock.
if components_to_load := components - self._loaded:
await self._async_load(components_to_load)
return {
component: result
for component in components
if (result := self._cache.get(category, {}).get(component))
}
async def _async_load(self, components: set[str]) -> None:
"""Populate the cache for a given set of components."""
_LOGGER.debug("Cache miss for: %s", components)
integrations: dict[str, Integration] = {}
domains = {loaded.rpartition(".")[-1] for loaded in components}
ints_or_excs = await async_get_integrations(self._hass, domains)
for domain, int_or_exc in ints_or_excs.items():
if isinstance(int_or_exc, Exception):
raise int_or_exc
integrations[domain] = int_or_exc
icons = await _async_get_component_icons(self._hass, components, integrations)
self._build_category_cache(components, icons)
self._loaded.update(components)
@callback
def _build_category_cache(
self,
components: set[str],
icons: dict[str, dict[str, Any]],
) -> None:
"""Extract resources into the cache."""
categories: set[str] = set()
for resource in icons.values():
categories.update(resource)
for category in categories:
self._cache.setdefault(category, {}).update(
build_resources(icons, components, category)
)
async def async_get_icons(
hass: HomeAssistant,
category: str,
integrations: Iterable[str] | None = None,
) -> dict[str, Any]:
"""Return all icons of integrations.
If integration specified, load it for that one; otherwise default to loaded
integrations.
"""
if integrations:
components = set(integrations)
else:
components = {
component for component in hass.config.components if "." not in component
}
if ICON_CACHE in hass.data:
cache: _IconsCache = hass.data[ICON_CACHE]
else:
cache = hass.data[ICON_CACHE] = _IconsCache(hass)
return await cache.async_fetch(category, components)
@lru_cache
def icon_for_battery_level(
battery_level: int | None = None, charging: bool = False
) -> str:
"""Return a battery icon valid identifier."""
icon = "mdi:battery"
if battery_level is None:
return f"{icon}-unknown"
if charging and battery_level > 10:
icon += f"-charging-{int(round(battery_level / 20 - 0.01)) * 20}"
elif charging:
icon += "-outline"
elif battery_level <= 5:
icon += "-alert"
elif 5 < battery_level < 95:
icon += f"-{int(round(battery_level / 10 - 0.01)) * 10}"
return icon
def icon_for_signal_level(signal_level: int | None = None) -> str:
"""Return a signal icon valid identifier."""
if signal_level is None or signal_level == 0:
return "mdi:signal-cellular-outline"
if signal_level > 70:
return "mdi:signal-cellular-3"
if signal_level > 30:
return "mdi:signal-cellular-2"
return "mdi:signal-cellular-1"