mirror of
https://github.com/esphome/esphome.git
synced 2025-09-16 10:12:21 +01:00
175 lines
6.1 KiB
Python
175 lines
6.1 KiB
Python
import esphome.codegen as cg
|
|
from esphome.components import esp32_ble_tracker, sensor
|
|
import esphome.config_validation as cv
|
|
from esphome.const import (
|
|
CONF_BATTERY_LEVEL,
|
|
CONF_DISTANCE,
|
|
CONF_ID,
|
|
CONF_LEVEL,
|
|
CONF_MAC_ADDRESS,
|
|
CONF_TEMPERATURE,
|
|
DEVICE_CLASS_BATTERY,
|
|
DEVICE_CLASS_TEMPERATURE,
|
|
ENTITY_CATEGORY_DIAGNOSTIC,
|
|
ICON_COUNTER,
|
|
ICON_RULER,
|
|
ICON_SIGNAL,
|
|
ICON_THERMOMETER,
|
|
STATE_CLASS_MEASUREMENT,
|
|
UNIT_CELSIUS,
|
|
UNIT_EMPTY,
|
|
UNIT_MILLIMETER,
|
|
UNIT_PERCENT,
|
|
)
|
|
|
|
CONF_TANK_TYPE = "tank_type"
|
|
CONF_CUSTOM_DISTANCE_FULL = "custom_distance_full"
|
|
CONF_CUSTOM_DISTANCE_EMPTY = "custom_distance_empty"
|
|
CONF_SIGNAL_QUALITY = "signal_quality"
|
|
CONF_MINIMUM_SIGNAL_QUALITY = "minimum_signal_quality"
|
|
CONF_IGNORED_READS = "ignored_reads"
|
|
|
|
ICON_PROPANE_TANK = "mdi:propane-tank"
|
|
|
|
TANK_TYPE_CUSTOM = "CUSTOM"
|
|
|
|
|
|
def small_distance(value):
|
|
"""small_distance is stored in mm"""
|
|
meters = cv.distance(value)
|
|
return meters * 1000
|
|
|
|
|
|
#
|
|
# Map of standard tank types to their
|
|
# empty and full distance values.
|
|
# Format is - tank name: (empty distance in mm, full distance in mm)
|
|
#
|
|
CONF_SUPPORTED_TANKS_MAP = {
|
|
TANK_TYPE_CUSTOM: (0, 100),
|
|
"20LB_V": (38, 254), # empty/full readings for 20lb US tank
|
|
"30LB_V": (38, 381),
|
|
"40LB_V": (38, 508),
|
|
"EUROPE_6KG": (38, 336),
|
|
"EUROPE_11KG": (38, 366),
|
|
"EUROPE_14KG": (38, 467),
|
|
}
|
|
|
|
CODEOWNERS = ["@spbrogan"]
|
|
DEPENDENCIES = ["esp32_ble_tracker"]
|
|
|
|
mopeka_pro_check_ns = cg.esphome_ns.namespace("mopeka_pro_check")
|
|
MopekaProCheck = mopeka_pro_check_ns.class_(
|
|
"MopekaProCheck", esp32_ble_tracker.ESPBTDeviceListener, cg.Component
|
|
)
|
|
|
|
SensorReadQuality = mopeka_pro_check_ns.enum("SensorReadQuality")
|
|
SIGNAL_QUALITIES = {
|
|
"ZERO": SensorReadQuality.QUALITY_ZERO,
|
|
"LOW": SensorReadQuality.QUALITY_LOW,
|
|
"MEDIUM": SensorReadQuality.QUALITY_MED,
|
|
"HIGH": SensorReadQuality.QUALITY_HIGH,
|
|
}
|
|
|
|
CONFIG_SCHEMA = (
|
|
cv.Schema(
|
|
{
|
|
cv.GenerateID(): cv.declare_id(MopekaProCheck),
|
|
cv.Required(CONF_MAC_ADDRESS): cv.mac_address,
|
|
cv.Optional(CONF_CUSTOM_DISTANCE_FULL): small_distance,
|
|
cv.Optional(CONF_CUSTOM_DISTANCE_EMPTY): small_distance,
|
|
cv.Required(CONF_TANK_TYPE): cv.enum(CONF_SUPPORTED_TANKS_MAP, upper=True),
|
|
cv.Optional(CONF_TEMPERATURE): sensor.sensor_schema(
|
|
unit_of_measurement=UNIT_CELSIUS,
|
|
icon=ICON_THERMOMETER,
|
|
accuracy_decimals=0,
|
|
device_class=DEVICE_CLASS_TEMPERATURE,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
cv.Optional(CONF_LEVEL): sensor.sensor_schema(
|
|
unit_of_measurement=UNIT_PERCENT,
|
|
icon=ICON_PROPANE_TANK,
|
|
accuracy_decimals=0,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
cv.Optional(CONF_DISTANCE): sensor.sensor_schema(
|
|
unit_of_measurement=UNIT_MILLIMETER,
|
|
icon=ICON_RULER,
|
|
accuracy_decimals=0,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
cv.Optional(CONF_BATTERY_LEVEL): sensor.sensor_schema(
|
|
unit_of_measurement=UNIT_PERCENT,
|
|
accuracy_decimals=0,
|
|
device_class=DEVICE_CLASS_BATTERY,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
cv.Optional(CONF_SIGNAL_QUALITY): sensor.sensor_schema(
|
|
unit_of_measurement=UNIT_EMPTY,
|
|
icon=ICON_SIGNAL,
|
|
accuracy_decimals=0,
|
|
state_class=STATE_CLASS_MEASUREMENT,
|
|
),
|
|
cv.Optional(CONF_IGNORED_READS): sensor.sensor_schema(
|
|
unit_of_measurement=UNIT_EMPTY,
|
|
icon=ICON_COUNTER,
|
|
accuracy_decimals=0,
|
|
entity_category=ENTITY_CATEGORY_DIAGNOSTIC,
|
|
),
|
|
cv.Optional(CONF_MINIMUM_SIGNAL_QUALITY, default="MEDIUM"): cv.enum(
|
|
SIGNAL_QUALITIES, upper=True
|
|
),
|
|
}
|
|
)
|
|
.extend(esp32_ble_tracker.ESP_BLE_DEVICE_SCHEMA)
|
|
.extend(cv.COMPONENT_SCHEMA)
|
|
)
|
|
|
|
|
|
async def to_code(config):
|
|
var = cg.new_Pvariable(config[CONF_ID])
|
|
await cg.register_component(var, config)
|
|
await esp32_ble_tracker.register_ble_device(var, config)
|
|
|
|
cg.add(var.set_address(config[CONF_MAC_ADDRESS].as_hex))
|
|
|
|
if config[CONF_TANK_TYPE] == TANK_TYPE_CUSTOM:
|
|
# Support custom tank min/max
|
|
if CONF_CUSTOM_DISTANCE_EMPTY in config:
|
|
cg.add(var.set_tank_empty(config[CONF_CUSTOM_DISTANCE_EMPTY]))
|
|
else:
|
|
cg.add(var.set_tank_empty(CONF_SUPPORTED_TANKS_MAP[TANK_TYPE_CUSTOM][0]))
|
|
if CONF_CUSTOM_DISTANCE_FULL in config:
|
|
cg.add(var.set_tank_full(config[CONF_CUSTOM_DISTANCE_FULL]))
|
|
else:
|
|
cg.add(var.set_tank_full(CONF_SUPPORTED_TANKS_MAP[TANK_TYPE_CUSTOM][1]))
|
|
else:
|
|
# Set the Tank empty and full based on map - User is requesting standard tank
|
|
t = config[CONF_TANK_TYPE]
|
|
cg.add(var.set_tank_empty(CONF_SUPPORTED_TANKS_MAP[t][0]))
|
|
cg.add(var.set_tank_full(CONF_SUPPORTED_TANKS_MAP[t][1]))
|
|
|
|
if (
|
|
minimum_signal_quality := config.get(CONF_MINIMUM_SIGNAL_QUALITY, None)
|
|
) is not None:
|
|
cg.add(var.set_min_signal_quality(minimum_signal_quality))
|
|
|
|
if CONF_TEMPERATURE in config:
|
|
sens = await sensor.new_sensor(config[CONF_TEMPERATURE])
|
|
cg.add(var.set_temperature(sens))
|
|
if CONF_LEVEL in config:
|
|
sens = await sensor.new_sensor(config[CONF_LEVEL])
|
|
cg.add(var.set_level(sens))
|
|
if CONF_DISTANCE in config:
|
|
sens = await sensor.new_sensor(config[CONF_DISTANCE])
|
|
cg.add(var.set_distance(sens))
|
|
if CONF_BATTERY_LEVEL in config:
|
|
sens = await sensor.new_sensor(config[CONF_BATTERY_LEVEL])
|
|
cg.add(var.set_battery_level(sens))
|
|
if CONF_SIGNAL_QUALITY in config:
|
|
sens = await sensor.new_sensor(config[CONF_SIGNAL_QUALITY])
|
|
cg.add(var.set_signal_quality(sens))
|
|
if CONF_IGNORED_READS in config:
|
|
sens = await sensor.new_sensor(config[CONF_IGNORED_READS])
|
|
cg.add(var.set_ignored_reads(sens))
|