2019-02-15 13:35:26 +00:00
|
|
|
"""Battery Charge and Range Support for the Nissan Leaf."""
|
2021-12-28 07:56:07 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2019-02-15 13:35:26 +00:00
|
|
|
import logging
|
|
|
|
|
2021-12-29 11:23:54 +00:00
|
|
|
from pycarwings2.pycarwings2 import Leaf
|
|
|
|
from voluptuous.validators import Number
|
|
|
|
|
2021-12-16 21:44:50 +00:00
|
|
|
from homeassistant.components.sensor import SensorDeviceClass, SensorEntity
|
2022-01-10 16:10:46 +00:00
|
|
|
from homeassistant.const import LENGTH_KILOMETERS, LENGTH_MILES, PERCENTAGE
|
2021-12-28 07:56:07 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2019-02-15 13:35:26 +00:00
|
|
|
from homeassistant.helpers.icon import icon_for_battery_level
|
2021-12-28 07:56:07 +00:00
|
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
2019-02-15 13:35:26 +00:00
|
|
|
from homeassistant.util.unit_system import IMPERIAL_SYSTEM, METRIC_SYSTEM
|
|
|
|
|
2022-01-02 17:48:33 +00:00
|
|
|
from . import LeafEntity
|
|
|
|
from .const import (
|
2019-07-31 19:25:30 +00:00
|
|
|
DATA_BATTERY,
|
|
|
|
DATA_CHARGING,
|
|
|
|
DATA_LEAF,
|
|
|
|
DATA_RANGE_AC,
|
|
|
|
DATA_RANGE_AC_OFF,
|
|
|
|
)
|
2019-03-21 05:56:46 +00:00
|
|
|
|
2019-02-15 13:35:26 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
ICON_RANGE = "mdi:speedometer"
|
2019-02-15 13:35:26 +00:00
|
|
|
|
|
|
|
|
2021-12-28 07:56:07 +00:00
|
|
|
def setup_platform(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config: ConfigType,
|
2022-01-07 16:54:37 +00:00
|
|
|
add_entities: AddEntitiesCallback,
|
2021-12-28 07:56:07 +00:00
|
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
|
|
) -> None:
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Sensors setup."""
|
2019-02-22 16:34:23 +00:00
|
|
|
if discovery_info is None:
|
|
|
|
return
|
2019-02-15 13:35:26 +00:00
|
|
|
|
2022-01-07 16:54:37 +00:00
|
|
|
entities: list[LeafEntity] = []
|
2019-02-22 16:34:23 +00:00
|
|
|
for vin, datastore in hass.data[DATA_LEAF].items():
|
|
|
|
_LOGGER.debug("Adding sensors for vin=%s", vin)
|
2022-01-07 16:54:37 +00:00
|
|
|
entities.append(LeafBatterySensor(datastore))
|
|
|
|
entities.append(LeafRangeSensor(datastore, True))
|
|
|
|
entities.append(LeafRangeSensor(datastore, False))
|
2019-02-15 13:35:26 +00:00
|
|
|
|
2022-01-07 16:54:37 +00:00
|
|
|
add_entities(entities, True)
|
2019-02-15 13:35:26 +00:00
|
|
|
|
|
|
|
|
2021-03-22 18:46:46 +00:00
|
|
|
class LeafBatterySensor(LeafEntity, SensorEntity):
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Nissan Leaf Battery Sensor."""
|
|
|
|
|
2021-12-29 15:43:10 +00:00
|
|
|
def __init__(self, car: Leaf) -> None:
|
|
|
|
"""Set up battery sensor."""
|
|
|
|
super().__init__(car)
|
|
|
|
self._attr_unique_id = f"{self.car.leaf.vin.lower()}_soc"
|
|
|
|
|
2019-02-15 13:35:26 +00:00
|
|
|
@property
|
2021-12-29 11:23:54 +00:00
|
|
|
def name(self) -> str:
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Sensor Name."""
|
2020-04-04 18:17:11 +00:00
|
|
|
return f"{self.car.leaf.nickname} Charge"
|
2019-02-15 13:35:26 +00:00
|
|
|
|
|
|
|
@property
|
2021-12-29 11:23:54 +00:00
|
|
|
def device_class(self) -> str:
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Return the device class of the sensor."""
|
2021-12-16 21:44:50 +00:00
|
|
|
return SensorDeviceClass.BATTERY
|
2019-02-15 13:35:26 +00:00
|
|
|
|
|
|
|
@property
|
2021-12-29 11:23:54 +00:00
|
|
|
def native_value(self) -> Number | None:
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Battery state percentage."""
|
2021-11-18 14:18:25 +00:00
|
|
|
if self.car.data[DATA_BATTERY] is None:
|
|
|
|
return None
|
2019-02-15 13:35:26 +00:00
|
|
|
return round(self.car.data[DATA_BATTERY])
|
|
|
|
|
|
|
|
@property
|
2021-12-29 11:23:54 +00:00
|
|
|
def native_unit_of_measurement(self) -> str:
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Battery state measured in percentage."""
|
2020-09-05 19:09:14 +00:00
|
|
|
return PERCENTAGE
|
2019-02-15 13:35:26 +00:00
|
|
|
|
|
|
|
@property
|
2021-12-29 11:23:54 +00:00
|
|
|
def icon(self) -> str:
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Battery state icon handling."""
|
|
|
|
chargestate = self.car.data[DATA_CHARGING]
|
2019-07-31 19:25:30 +00:00
|
|
|
return icon_for_battery_level(battery_level=self.state, charging=chargestate)
|
2019-02-15 13:35:26 +00:00
|
|
|
|
|
|
|
|
2021-03-22 18:46:46 +00:00
|
|
|
class LeafRangeSensor(LeafEntity, SensorEntity):
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Nissan Leaf Range Sensor."""
|
|
|
|
|
2021-12-29 11:23:54 +00:00
|
|
|
def __init__(self, car: Leaf, ac_on: bool) -> None:
|
2019-12-10 22:25:06 +00:00
|
|
|
"""Set up range sensor. Store if AC on."""
|
2019-02-15 13:35:26 +00:00
|
|
|
self._ac_on = ac_on
|
|
|
|
super().__init__(car)
|
2021-12-29 15:43:10 +00:00
|
|
|
if ac_on:
|
|
|
|
self._attr_unique_id = f"{self.car.leaf.vin.lower()}_range_ac"
|
|
|
|
else:
|
|
|
|
self._attr_unique_id = f"{self.car.leaf.vin.lower()}_range"
|
2019-02-15 13:35:26 +00:00
|
|
|
|
|
|
|
@property
|
2021-12-29 11:23:54 +00:00
|
|
|
def name(self) -> str:
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Update sensor name depending on AC."""
|
|
|
|
if self._ac_on is True:
|
2020-04-04 18:17:11 +00:00
|
|
|
return f"{self.car.leaf.nickname} Range (AC)"
|
|
|
|
return f"{self.car.leaf.nickname} Range"
|
2019-02-15 13:35:26 +00:00
|
|
|
|
2021-12-29 11:23:54 +00:00
|
|
|
def log_registration(self) -> None:
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Log registration."""
|
|
|
|
_LOGGER.debug(
|
2020-01-05 12:09:17 +00:00
|
|
|
"Registered LeafRangeSensor integration with Home Assistant for VIN %s",
|
2019-07-31 19:25:30 +00:00
|
|
|
self.car.leaf.vin,
|
|
|
|
)
|
2019-02-15 13:35:26 +00:00
|
|
|
|
|
|
|
@property
|
2021-12-29 11:23:54 +00:00
|
|
|
def native_value(self) -> float | None:
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Battery range in miles or kms."""
|
|
|
|
if self._ac_on:
|
|
|
|
ret = self.car.data[DATA_RANGE_AC]
|
|
|
|
else:
|
|
|
|
ret = self.car.data[DATA_RANGE_AC_OFF]
|
|
|
|
|
2021-11-18 14:18:25 +00:00
|
|
|
if ret is None:
|
|
|
|
return None
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
if not self.car.hass.config.units.is_metric or self.car.force_miles:
|
2019-02-15 13:35:26 +00:00
|
|
|
ret = IMPERIAL_SYSTEM.length(ret, METRIC_SYSTEM.length_unit)
|
|
|
|
|
|
|
|
return round(ret)
|
|
|
|
|
|
|
|
@property
|
2021-12-29 11:23:54 +00:00
|
|
|
def native_unit_of_measurement(self) -> str:
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Battery range unit."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if not self.car.hass.config.units.is_metric or self.car.force_miles:
|
2019-02-15 13:35:26 +00:00
|
|
|
return LENGTH_MILES
|
|
|
|
return LENGTH_KILOMETERS
|
|
|
|
|
|
|
|
@property
|
2021-12-29 11:23:54 +00:00
|
|
|
def icon(self) -> str:
|
2019-02-15 13:35:26 +00:00
|
|
|
"""Nice icon for range."""
|
|
|
|
return ICON_RANGE
|