2019-04-22 21:56:30 +02:00
|
|
|
import esphome.codegen as cg
|
|
|
|
import esphome.config_validation as cv
|
2019-04-17 12:06:00 +02:00
|
|
|
from esphome import pins
|
|
|
|
from esphome.components import sensor
|
|
|
|
from esphome.const import CONF_COUNT_MODE, CONF_FALLING_EDGE, CONF_ID, CONF_INTERNAL_FILTER, \
|
2019-04-22 21:56:30 +02:00
|
|
|
CONF_PIN, CONF_RISING_EDGE, CONF_UPDATE_INTERVAL, CONF_NUMBER, \
|
|
|
|
ICON_PULSE, UNIT_PULSES_PER_MINUTE
|
2019-04-17 12:06:00 +02:00
|
|
|
from esphome.core import CORE
|
|
|
|
|
|
|
|
pulse_counter_ns = cg.esphome_ns.namespace('pulse_counter')
|
|
|
|
PulseCounterCountMode = pulse_counter_ns.enum('PulseCounterCountMode')
|
|
|
|
COUNT_MODES = {
|
|
|
|
'DISABLE': PulseCounterCountMode.PULSE_COUNTER_DISABLE,
|
|
|
|
'INCREMENT': PulseCounterCountMode.PULSE_COUNTER_INCREMENT,
|
|
|
|
'DECREMENT': PulseCounterCountMode.PULSE_COUNTER_DECREMENT,
|
|
|
|
}
|
|
|
|
|
2019-04-22 21:56:30 +02:00
|
|
|
COUNT_MODE_SCHEMA = cv.enum(COUNT_MODES, upper=True)
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
PulseCounterSensor = pulse_counter_ns.class_('PulseCounterSensor',
|
2019-05-08 10:17:08 +02:00
|
|
|
sensor.Sensor, cg.PollingComponent)
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
def validate_internal_filter(value):
|
|
|
|
value = cv.positive_time_period_microseconds(value)
|
|
|
|
if CORE.is_esp32:
|
|
|
|
if value.total_microseconds > 13:
|
|
|
|
raise cv.Invalid("Maximum internal filter value for ESP32 is 13us")
|
|
|
|
return value
|
|
|
|
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
|
|
|
def validate_pulse_counter_pin(value):
|
|
|
|
value = pins.internal_gpio_input_pin_schema(value)
|
|
|
|
if CORE.is_esp8266 and value[CONF_NUMBER] >= 16:
|
|
|
|
raise cv.Invalid("Pins GPIO16 and GPIO17 cannot be used as pulse counters on ESP8266.")
|
|
|
|
return value
|
|
|
|
|
|
|
|
|
2019-04-22 21:56:30 +02:00
|
|
|
CONFIG_SCHEMA = sensor.sensor_schema(UNIT_PULSES_PER_MINUTE, ICON_PULSE, 2).extend({
|
|
|
|
cv.GenerateID(): cv.declare_id(PulseCounterSensor),
|
2019-04-17 12:06:00 +02:00
|
|
|
cv.Required(CONF_PIN): validate_pulse_counter_pin,
|
|
|
|
cv.Optional(CONF_COUNT_MODE, default={
|
|
|
|
CONF_RISING_EDGE: 'INCREMENT',
|
|
|
|
CONF_FALLING_EDGE: 'DISABLE',
|
|
|
|
}): cv.Schema({
|
|
|
|
cv.Required(CONF_RISING_EDGE): COUNT_MODE_SCHEMA,
|
|
|
|
cv.Required(CONF_FALLING_EDGE): COUNT_MODE_SCHEMA,
|
|
|
|
}),
|
|
|
|
cv.Optional(CONF_INTERNAL_FILTER, default='13us'): validate_internal_filter,
|
|
|
|
cv.Optional(CONF_UPDATE_INTERVAL, default='60s'): cv.update_interval,
|
2019-04-22 21:56:30 +02:00
|
|
|
}).extend(cv.polling_component_schema('60s'))
|
2019-04-17 12:06:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
def to_code(config):
|
2019-04-22 21:56:30 +02:00
|
|
|
var = cg.new_Pvariable(config[CONF_ID])
|
2019-04-17 12:06:00 +02:00
|
|
|
yield cg.register_component(var, config)
|
|
|
|
yield sensor.register_sensor(var, config)
|
2019-04-22 21:56:30 +02:00
|
|
|
|
|
|
|
pin = yield cg.gpio_pin_expression(config[CONF_PIN])
|
|
|
|
cg.add(var.set_pin(pin))
|
|
|
|
count = config[CONF_COUNT_MODE]
|
|
|
|
cg.add(var.set_rising_edge_mode(count[CONF_RISING_EDGE]))
|
|
|
|
cg.add(var.set_falling_edge_mode(count[CONF_FALLING_EDGE]))
|
|
|
|
cg.add(var.set_filter_us(config[CONF_INTERNAL_FILTER]))
|