UNPKG

zigbee-herdsman-converters

Version:

Collection of device converters to be used with zigbee-herdsman

573 lines 34 kB
"use strict"; var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; var __importDefault = (this && this.__importDefault) || function (mod) { return (mod && mod.__esModule) ? mod : { "default": mod }; }; Object.defineProperty(exports, "__esModule", { value: true }); const exposes = __importStar(require("../lib/exposes")); const fromZigbee_1 = __importDefault(require("../converters/fromZigbee")); const toZigbee_1 = __importDefault(require("../converters/toZigbee")); const legacy = __importStar(require("../lib/legacy")); const reporting = __importStar(require("../lib/reporting")); const e = exposes.presets; const ea = exposes.access; const tuya = __importStar(require("../lib/tuya")); const globalStore = __importStar(require("../lib/store")); const ota = __importStar(require("../lib/ota")); const utils = __importStar(require("../lib/utils")); const valueConverterLocal = { wateringState: { from: (value, meta, options, publish) => { const result = { state: value ? 'ON' : 'OFF', ...(value ? {} : { // ensure time_left is set to zero when it's OFF time_left: 0, }), }; // prepare the time reporting for water scheduler // indications when the watering was triggered by scheduler: // - scheduling is enabled // - current state is on // - time_left wasn't reported before and is 0 // - current hour & minute matches scheduling period if (meta.state.schedule_mode !== 'OFF' && result.state === 'ON' && meta.state.time_left === 0 && !globalStore.hasValue(meta.device, 'watering_timer_active_time_slot')) { const now = new Date(); const timeslot = [1, 2, 3, 4, 5, 6] .map((slotNumber) => utils.getObjectProperty(meta.state, `schedule_slot_${slotNumber}`, {})) // @ts-expect-error .find((ts) => ts.state === 'ON' && ts.start_hour === now.getHours() && ts.start_minute === now.getMinutes() && ts.timer > 0); if (timeslot) { // @ts-expect-error const iterationDuration = timeslot.timer + timeslot.pause; // automatic watering detected globalStore.putValue(meta.device, 'watering_timer_active_time_slot', { timeslot_start_timestamp: now.getTime(), // end of last watering excluding last pause // @ts-expect-error timeslot_end_timestamp: now.getTime() + (timeslot.iterations * iterationDuration - timeslot.pause) * 60 * 1000, // @ts-expect-error timer: timeslot.timer, iteration_inverval: null, // will be set in the next step iteration_start_timestamp: 0, // will be set in the next step }); } } // setup time reporting for water scheduler when necessary if (globalStore.hasValue(meta.device, 'watering_timer_active_time_slot')) { const ts = globalStore.getValue(meta.device, 'watering_timer_active_time_slot'); if ( // time slot execution is already completed (Date.now() > (ts.timeslot_end_timestamp - 5000)) || // scheduling was interrupted by turning watering on manually // @ts-expect-error (result.state === 'ON' && result.state != meta.state.state && meta.state.time_left > 0)) { // reporting is no longer necessary clearInterval(ts.iteration_inverval); globalStore.clearValue(meta.device, 'watering_timer_active_time_slot'); } else if (result.state === 'OFF' && result.state !== meta.state.state) { // turned off --> disable reporting for this iteration only clearInterval(ts.iteration_inverval); ts.iteration_inverval = null; } else if (result.state === 'ON' && result.state !== meta.state.state && meta.state.time_left === 0) { // automatic scheduling detected (reported as ON, but without any info about duration) ts.iteration_report = true; ts.iteration_start_timestamp = Date.now(); if (ts.timer > 1) { // report every minute const interval = ts.iteration_inverval = setInterval(() => { const now = Date.now(); const wateringEndTime = ts.iteration_start_timestamp + ts.timer * 60 * 1000; const timeLeftInMinutes = Math.round((wateringEndTime - now) / 1000 / 60); if (timeLeftInMinutes > 0) { if (timeLeftInMinutes === 1) { clearInterval(interval); } publish({ time_left: timeLeftInMinutes, }); } }, 60 * 1000); } // initial reporting result.time_left = ts.timer; } } return result; }, }, wateringScheduleMode: { from: (value) => { const [scheduleMode, scheduleValue] = value; const isWeekday = scheduleMode === 0; return { schedule_mode: scheduleValue === 0 ? 'OFF' : isWeekday ? 'WEEKDAY' : 'PERIODIC', schedule_periodic: !isWeekday ? scheduleValue : 0, schedule_weekday: ['monday', 'tuesday', 'wednesday', 'thursday', 'friday', 'saturday', 'sunday'] .reduce((scheduleMap, dayName, index) => ({ ...scheduleMap, [dayName]: isWeekday && (scheduleValue & (1 << index)) > 0 ? 'ON' : 'OFF', }), {}), }; }, }, wateringSchedulePeriodic: { to: (value) => { if (!utils.isInRange(0, 7, value)) throw new Error(`Invalid value: ${value} (expected ${0} to ${7})`); // Note: mode value of 0 switches to disabled weekday scheduler const scheduleMode = value > 0 ? 1 : 0; return [scheduleMode, value]; }, }, wateringScheduleWeekday: { to: (value, meta) => { // map each day to ON/OFF and use current state as default to allow partial updates const dayValues = ['monday', 'tuesday', 'wednesday', 'thursday', 'friday', 'saturday', 'sunday'] // @ts-expect-error .map((dayName) => utils.getObjectProperty(value, dayName, utils.getObjectProperty(meta.state.schedule_weekday, dayName, 'OFF'))); const scheduleValue = dayValues.reduce((dayConfig, value, index) => { // @ts-expect-error return dayConfig | (value === 'ON' ? 1 << index : 0); }, 0); // value of 0 switches to weekday scheduler const scheduleMode = 0; return [scheduleMode, scheduleValue]; }, }, wateringScheduleSlot: (timeSlotNumber) => ({ from: (buffer) => { return { state: buffer.readUInt8(0) === 1 ? 'ON' : 'OFF', start_hour: utils.numberWithinRange(buffer.readUInt8(1), 0, 23), // device reports non-valid value 255 initially start_minute: utils.numberWithinRange(buffer.readUInt8(2), 0, 59), // device reports non-valid value 255 initially timer: utils.numberWithinRange(buffer.readUInt8(3) * 60 + buffer.readUInt8(4), 1, 599), // device reports non-valid value 0 initially pause: utils.numberWithinRange(buffer.readUInt8(6) * 60 + buffer.readUInt8(7), 0, 599), iterations: utils.numberWithinRange(buffer.readUInt8(9), 1, 9), // device reports non-valid value 0 initially }; }, to: (value, meta) => { // use default values from current config to allow partial updates const timeslot = utils.getObjectProperty(meta.state, `schedule_slot_${timeSlotNumber}`, {}); const state = utils.getObjectProperty(value, 'state', timeslot.state ?? false); const startHour = utils.getObjectProperty(value, 'start_hour', timeslot.start_hour ?? 23); const startMinute = utils.getObjectProperty(value, 'start_minute', timeslot.start_minute ?? 59); const duratonInMin = utils.getObjectProperty(value, 'timer', timeslot.timer ?? 1); const iterations = utils.getObjectProperty(value, 'iterations', timeslot.iterations ?? 1); const pauseInMin = utils.getObjectProperty(value, 'pause', timeslot.pause ?? 0); if (!utils.isInRange(0, 23, startHour)) throw new Error(`Invalid start hour value ${startHour} (expected ${0} to ${23})`); if (!utils.isInRange(0, 59, startMinute)) throw new Error(`Invalid start minute value: ${startMinute} (expected ${0} to ${59})`); if (!utils.isInRange(1, 599, duratonInMin)) throw new Error(`Invalid timer value: ${duratonInMin} (expected ${1} to ${599})`); if (!utils.isInRange(1, 9, iterations)) throw new Error(`Invalid iterations value: ${iterations} (expected ${1} to ${9})`); if (!utils.isInRange(0, 599, pauseInMin)) throw new Error(`Invalid pause value: ${pauseInMin} (expected ${0} to ${599})`); if (iterations > 1 && pauseInMin === 0) throw new Error(`Pause value must be at least 1 minute when using multiple iterations`); return [ // @ts-expect-error state === 'ON' ? 1 : 0, // time slot enabled or not startHour, // start hour startMinute, // start minute Math.floor(duratonInMin / 60), // duration for n hours duratonInMin % 60, // duration + n minutes 0, // what's this? -> was always reported as 0 Math.floor(pauseInMin / 60), // pause in hours pauseInMin % 60, // pause + n minutes 0, // what's this? -> was always reported as 0 iterations, // iterations ]; }, }), }; const definitions = [ { fingerprint: [ { manufacturerName: '_TZ3000_kdi2o9m6' }, // EU { modelID: 'TS011F', manufacturerName: '_TZ3000_plyvnuf5' }, // CH { modelID: 'TS011F', manufacturerName: '_TZ3000_wamqdr3f' }, // FR { modelID: 'TS011F', manufacturerName: '_TZ3000_00mk2xzy' }, // BS { modelID: 'TS011F', manufacturerName: '_TZ3000_upjrsxh1' }, // DK { manufacturerName: '_TZ3000_00mk2xzy' }, // BS ], model: 'HG06337', vendor: 'Lidl', description: 'Silvercrest smart plug (EU, CH, FR, BS, DK)', extend: [tuya.modernExtend.tuyaOnOff({ indicatorMode: true })], configure: async (device, coordinatorEndpoint) => { const endpoint = device.getEndpoint(11); await reporting.bind(endpoint, coordinatorEndpoint, ['genOnOff']); await reporting.onOff(endpoint); }, }, { fingerprint: tuya.fingerprint('TS011F', ['_TZ3000_j1v25l17', '_TZ3000_ynmowqk2', '_TZ3000_3uimvkn6']), model: 'HG08673', vendor: 'Lidl', description: 'Silvercrest smart plug with power monitoring (EU, FR)', ota: ota.zigbeeOTA, extend: [tuya.modernExtend.tuyaOnOff({ electricalMeasurements: true, powerOutageMemory: true, indicatorMode: true, childLock: true })], configure: async (device, coordinatorEndpoint) => { const endpoint = device.getEndpoint(1); await tuya.configureMagicPacket(device, coordinatorEndpoint); await reporting.bind(endpoint, coordinatorEndpoint, ['genOnOff', 'haElectricalMeasurement']); await reporting.rmsVoltage(endpoint, { change: 5 }); await reporting.rmsCurrent(endpoint, { change: 50 }); await reporting.activePower(endpoint, { change: 10 }); // Energy reporting (currentSummDelivered) doesn't work; requires polling: https://github.com/Koenkk/zigbee2mqtt/issues/14356 endpoint.saveClusterAttributeKeyValue('haElectricalMeasurement', { acCurrentDivisor: 1000, acCurrentMultiplier: 1 }); endpoint.saveClusterAttributeKeyValue('seMetering', { divisor: 100, multiplier: 1 }); device.save(); }, options: [exposes.options.measurement_poll_interval().withDescription('Only the energy value is polled for this device.')], onEvent: (type, data, device, options) => tuya.onEventMeasurementPoll(type, data, device, options, false, true), whiteLabel: [ tuya.whitelabel('Lidl', 'HG08673-BS', 'Silvercrest smart plug with power monitoring (BS)', ['_TZ3000_3uimvkn6']), ], }, { fingerprint: [{ modelID: 'TS004F', manufacturerName: '_TZ3000_rco1yzb1' }], model: 'HG08164', vendor: 'Lidl', description: 'Silvercrest smart button', fromZigbee: [fromZigbee_1.default.command_on, fromZigbee_1.default.command_off, fromZigbee_1.default.command_step, fromZigbee_1.default.command_stop, fromZigbee_1.default.battery, tuya.fz.on_off_action], toZigbee: [], configure: async (device, coordinatorEndpoint) => { const endpoint = device.getEndpoint(1); await endpoint.read('genBasic', [0x0004, 0x000, 0x0001, 0x0005, 0x0007, 0xfffe]); await endpoint.read('genOnOff', ['tuyaOperationMode']); try { await endpoint.read(0xE001, [0xD011]); } catch (err) { /* do nothing */ } await endpoint.read('genPowerCfg', ['batteryVoltage', 'batteryPercentageRemaining']); await reporting.bind(endpoint, coordinatorEndpoint, ['genPowerCfg']); await reporting.bind(endpoint, coordinatorEndpoint, ['genOnOff']); await reporting.batteryPercentageRemaining(endpoint); }, exposes: [e.action(['on', 'off', 'brightness_stop', 'brightness_step_up', 'brightness_step_down', 'single', 'double']), e.battery()], }, { fingerprint: [{ modelID: 'TS0211', manufacturerName: '_TZ1800_ladpngdx' }], model: 'HG06668', vendor: 'Lidl', description: 'Silvercrest smart wireless door bell button', fromZigbee: [fromZigbee_1.default.battery, fromZigbee_1.default.tuya_doorbell_button, fromZigbee_1.default.ignore_basic_report], toZigbee: [], configure: async (device, coordinatorEndpoint) => { const endpoint = device.getEndpoint(1); await reporting.bind(endpoint, coordinatorEndpoint, ['genPowerCfg']); await reporting.batteryPercentageRemaining(endpoint); }, exposes: [e.battery(), e.action(['pressed']), e.battery_low(), e.tamper()], }, { fingerprint: [{ modelID: 'TY0202', manufacturerName: '_TZ1800_fcdjzz3s' }], model: 'HG06335/HG07310', vendor: 'Lidl', description: 'Silvercrest smart motion sensor', fromZigbee: [fromZigbee_1.default.ias_occupancy_alarm_1, fromZigbee_1.default.battery], toZigbee: [], exposes: [e.occupancy(), e.battery_low(), e.tamper(), e.battery()], configure: async (device, coordinatorEndpoint) => { const endpoint = device.getEndpoint(1); await reporting.bind(endpoint, coordinatorEndpoint, ['genPowerCfg']); await reporting.batteryVoltage(endpoint); await reporting.batteryPercentageRemaining(endpoint); }, }, { fingerprint: [ { modelID: 'TY0203', manufacturerName: '_TZ1800_ejwkn2h2' }, { modelID: 'TY0203', manufacturerName: '_TZ1800_ho6i0zk9' }, ], model: 'HG06336', vendor: 'Lidl', description: 'Silvercrest smart window and door sensor', fromZigbee: [fromZigbee_1.default.ias_contact_alarm_1, fromZigbee_1.default.ias_contact_alarm_1_report, fromZigbee_1.default.battery], toZigbee: [], exposes: [e.contact(), e.battery_low(), e.tamper(), e.battery()], configure: async (device, coordinatorEndpoint) => { const endpoint = device.getEndpoint(1); await reporting.bind(endpoint, coordinatorEndpoint, ['genPowerCfg']); await reporting.batteryPercentageRemaining(endpoint); }, }, { fingerprint: [{ modelID: 'TS1001', manufacturerName: '_TYZB01_bngwdjsr' }], model: 'FB20-002', vendor: 'Lidl', description: 'Livarno Lux switch and dimming light remote control', exposes: [e.action(['on', 'off', 'brightness_stop', 'brightness_step_up', 'brightness_step_down', 'brightness_move_up', 'brightness_move_down'])], fromZigbee: [fromZigbee_1.default.command_on, fromZigbee_1.default.command_off, fromZigbee_1.default.command_step, fromZigbee_1.default.command_move, fromZigbee_1.default.command_stop], toZigbee: [], }, { fingerprint: [{ modelID: 'TS1001', manufacturerName: '_TYZB01_hww2py6b' }], model: 'FB21-001', vendor: 'Lidl', description: 'Livarno Lux switch and dimming light remote control', exposes: [e.action(['on', 'off', 'brightness_stop', 'brightness_step_up', 'brightness_step_down', 'brightness_move_up', 'brightness_move_down', 'switch_scene'])], fromZigbee: [fromZigbee_1.default.command_on, fromZigbee_1.default.command_off, fromZigbee_1.default.command_step, fromZigbee_1.default.command_move, fromZigbee_1.default.command_stop, fromZigbee_1.default.tuya_switch_scene], toZigbee: [], }, { fingerprint: [ { modelID: 'TS011F', manufacturerName: '_TZ3000_wzauvbcs' }, // EU { modelID: 'TS011F', manufacturerName: '_TZ3000_oznonj5q' }, { modelID: 'TS011F', manufacturerName: '_TZ3000_1obwwnmq' }, { modelID: 'TS011F', manufacturerName: '_TZ3000_4uf3d0ax' }, // FR { modelID: 'TS011F', manufacturerName: '_TZ3000_vzopcetz' }, // CZ { modelID: 'TS011F', manufacturerName: '_TZ3000_vmpbygs5' }, // BS ], model: 'HG06338', vendor: 'Lidl', description: 'Silvercrest 3 gang switch, with 4 USB (EU, FR, CZ, BS)', extend: [tuya.modernExtend.tuyaOnOff({ endpoints: ['l1', 'l2', 'l3'] })], meta: { multiEndpoint: true }, configure: async (device, coordinatorEndpoint) => { await tuya.configureMagicPacket(device, coordinatorEndpoint); for (const ID of [1, 2, 3]) { await reporting.bind(device.getEndpoint(ID), coordinatorEndpoint, ['genOnOff']); } }, endpoint: (device) => { return { 'l1': 1, 'l2': 2, 'l3': 3 }; }, }, { fingerprint: [{ modelID: 'TS0601', manufacturerName: '_TZE200_s8gkrkxk' }], model: 'HG06467', vendor: 'Lidl', description: 'Melinera smart LED string lights', toZigbee: [toZigbee_1.default.on_off, legacy.tz.silvercrest_smart_led_string], fromZigbee: [fromZigbee_1.default.on_off, legacy.fz.silvercrest_smart_led_string], exposes: [e.light_brightness_colorhs().setAccess('brightness', ea.STATE_SET).setAccess('color_hs', ea.STATE_SET)], }, { fingerprint: [{ modelID: 'TS0504B', manufacturerName: '_TZ3210_sroezl0s' }], model: '14153806L', vendor: 'Lidl', description: 'Livarno smart LED ceiling light', extend: [tuya.modernExtend.tuyaLight({ colorTemp: { range: [153, 500] }, color: true })], configure: async (device, coordinatorEndpoint) => { device.getEndpoint(1).saveClusterAttributeKeyValue('lightingColorCtrl', { colorCapabilities: 29 }); }, }, { fingerprint: [{ modelID: 'TS0601', manufacturerName: '_TZE200_htnnfasr' }], model: 'PSBZS A1', vendor: 'Lidl', description: 'Parkside smart watering timer', fromZigbee: [fromZigbee_1.default.ignore_basic_report, fromZigbee_1.default.ignore_tuya_set_time, fromZigbee_1.default.ignore_onoff_report, tuya.fz.datapoints], toZigbee: [tuya.tz.datapoints], onEvent: async (type, data, device) => { // @ts-expect-error await tuya.onEventSetLocalTime(type, data, device); // @ts-expect-error if (type === 'deviceInterview' && data.status === 'successful') { // dirty hack: reset frost guard & frost alarm to get the initial state // wait 10 seconds to ensure configure is done await utils.sleep(10000); const endpoint = device.getEndpoint(1); try { await tuya.sendDataPointBool(endpoint, 109, false); await tuya.sendDataPointBool(endpoint, 108, false); } catch (e) { // ignore, just prevent any crashes } } }, configure: async (device, coordinatorEndpoint) => { await tuya.configureMagicPacket(device, coordinatorEndpoint); await reporting.bind(device.getEndpoint(1), coordinatorEndpoint, ['genOnOff']); // set reporting interval of genOnOff to max to "disable" it // background: genOnOff reporting does not respect timer or button, that makes the on/off reporting pretty useless // the device is reporting it's state change anyway via tuya DPs await reporting.onOff(device.getEndpoint(1), { max: 0xffff }); }, exposes: [ e.battery(), tuya.exposes.switch(), e.numeric('timer', ea.STATE_SET).withValueMin(1).withValueMax(599).withUnit('min') .withDescription('Auto off after specific time for manual watering.'), e.numeric('time_left', ea.STATE).withUnit('min') .withDescription('Remaining time until the watering turns off.'), e.binary('frost_lock', ea.STATE, 'ON', 'OFF') .withDescription('Indicates if the frost guard is currently active. ' + 'If the temperature drops below 5° C, device activates frost guard and disables irrigation. ' + 'You need to reset the frost guard to activate irrigation again. Note: There is no way to enable frost guard manually.'), e.enum('reset_frost_lock', ea.SET, ['RESET']).withDescription('Resets frost lock to make the device workable again.'), e.enum('schedule_mode', ea.STATE, ['OFF', 'WEEKDAY', 'PERIODIC']) .withDescription('Scheduling mode that is currently in use.'), e.numeric('schedule_periodic', ea.STATE_SET).withValueMin(0).withValueMax(7).withUnit('day') .withDescription('Watering by periodic interval: Irrigate every n days'), e.composite('schedule_weekday', 'schedule_weekday', ea.STATE_SET) .withDescription('Watering by weekday: Irrigate individually for each day.') .withFeature(e.binary('monday', ea.STATE_SET, 'ON', 'OFF')) .withFeature(e.binary('tuesday', ea.STATE_SET, 'ON', 'OFF')) .withFeature(e.binary('wednesday', ea.STATE_SET, 'ON', 'OFF')) .withFeature(e.binary('thursday', ea.STATE_SET, 'ON', 'OFF')) .withFeature(e.binary('friday', ea.STATE_SET, 'ON', 'OFF')) .withFeature(e.binary('saturday', ea.STATE_SET, 'ON', 'OFF')) .withFeature(e.binary('sunday', ea.STATE_SET, 'ON', 'OFF')), ...[1, 2, 3, 4, 5, 6].map((timeSlotNumber) => e.composite(`schedule_slot_${timeSlotNumber}`, `schedule_slot_${timeSlotNumber}`, ea.STATE_SET) .withDescription(`Watering time slot ${timeSlotNumber}`) .withFeature(e.binary('state', ea.STATE_SET, 'ON', 'OFF').withDescription('On/off state of the time slot')) .withFeature(e.numeric('start_hour', ea.STATE_SET).withUnit('h').withValueMin(0).withValueMax(23) .withDescription('Starting time (hour)')) .withFeature(e.numeric('start_minute', ea.STATE_SET).withUnit('min').withValueMin(0).withValueMax(59) .withDescription('Starting time (minute)')) .withFeature(e.numeric('timer', ea.STATE_SET).withUnit('min').withValueMin(1).withValueMax(599) .withDescription('Auto off after specific time for scheduled watering.')) .withFeature(e.numeric('pause', ea.STATE_SET).withUnit('min').withValueMin(0).withValueMax(599) .withDescription('Pause after each iteration.')) .withFeature(e.numeric('iterations', ea.STATE_SET).withValueMin(1).withValueMax(9) .withDescription('Number of watering iterations. Works only if there is a pause.'))), ], meta: { tuyaDatapoints: [ [1, null, valueConverterLocal.wateringState], // disable optimistic state reporting (device may not turn on when battery is low) [1, 'state', tuya.valueConverter.onOff, { optimistic: false }], [5, 'timer', tuya.valueConverter.raw], [6, 'time_left', tuya.valueConverter.raw], [11, 'battery', tuya.valueConverter.raw], [108, 'frost_lock', tuya.valueConverter.onOff], // there is no state reporting for reset [109, 'reset_frost_lock', tuya.valueConverterBasic.lookup({ 'RESET': tuya.enum(0) }), { optimistic: false }], [107, null, valueConverterLocal.wateringScheduleMode], [107, 'schedule_periodic', valueConverterLocal.wateringSchedulePeriodic], [107, 'schedule_weekday', valueConverterLocal.wateringScheduleWeekday], [101, 'schedule_slot_1', valueConverterLocal.wateringScheduleSlot(1)], [102, 'schedule_slot_2', valueConverterLocal.wateringScheduleSlot(2)], [103, 'schedule_slot_3', valueConverterLocal.wateringScheduleSlot(3)], [104, 'schedule_slot_4', valueConverterLocal.wateringScheduleSlot(4)], [105, 'schedule_slot_5', valueConverterLocal.wateringScheduleSlot(5)], [106, 'schedule_slot_6', valueConverterLocal.wateringScheduleSlot(6)], ], }, }, { fingerprint: [{ modelID: 'TS0101', manufacturerName: '_TZ3000_br3laukf' }], model: 'HG06620', vendor: 'Lidl', description: 'Silvercrest garden spike with 2 sockets', extend: [tuya.modernExtend.tuyaOnOff()], configure: async (device, coordinatorEndpoint) => { const endpoint = device.getEndpoint(1); await reporting.bind(endpoint, coordinatorEndpoint, ['genOnOff']); await reporting.onOff(endpoint); }, }, { fingerprint: [{ modelID: 'TS0101', manufacturerName: '_TZ3000_pnzfdr9y' }], model: 'HG06619', vendor: 'Lidl', description: 'Silvercrest outdoor plug', extend: [tuya.modernExtend.tuyaOnOff()], configure: async (device, coordinatorEndpoint) => { const endpoint = device.getEndpoint(1); await reporting.bind(endpoint, coordinatorEndpoint, ['genOnOff']); await reporting.onOff(endpoint); }, }, { fingerprint: [{ modelID: 'TS0505B', manufacturerName: '_TZ3000_lxw3zcdk' }], model: 'HG08633', vendor: 'Lidl', description: 'Livarno gardenspot RGB', extend: [tuya.modernExtend.tuyaLight({ colorTemp: { range: [153, 500] }, color: { modes: ['hs', 'xy'] } })], }, { fingerprint: [{ modelID: 'TS0601', manufacturerName: '_TZE200_chyvmhay' }], model: '368308_2010', vendor: 'Lidl', description: 'Silvercrest radiator valve with thermostat', fromZigbee: [fromZigbee_1.default.ignore_tuya_set_time, legacy.fromZigbee.zs_thermostat], toZigbee: [legacy.toZigbee.zs_thermostat_current_heating_setpoint, legacy.toZigbee.zs_thermostat_child_lock, legacy.toZigbee.zs_thermostat_comfort_temp, legacy.toZigbee.zs_thermostat_eco_temp, legacy.toZigbee.zs_thermostat_preset_mode, legacy.toZigbee.zs_thermostat_system_mode, legacy.toZigbee.zs_thermostat_local_temperature_calibration, legacy.toZigbee.zs_thermostat_current_heating_setpoint_auto, legacy.toZigbee.zs_thermostat_openwindow_time, legacy.toZigbee.zs_thermostat_openwindow_temp, legacy.toZigbee.zs_thermostat_binary_one, legacy.toZigbee.zs_thermostat_binary_two, legacy.toZigbee.zs_thermostat_away_setting, legacy.toZigbee.zs_thermostat_local_schedule], onEvent: tuya.onEventSetLocalTime, configure: async (device, coordinatorEndpoint) => { const endpoint = device.getEndpoint(1); await reporting.bind(endpoint, coordinatorEndpoint, ['genBasic']); }, exposes: [ e.child_lock(), e.comfort_temperature(), e.eco_temperature(), e.battery_voltage(), e.numeric('current_heating_setpoint_auto', ea.STATE_SET).withValueMin(0.5).withValueMax(29.5) .withValueStep(0.5).withUnit('°C').withDescription('Temperature setpoint automatic'), e.climate().withSetpoint('current_heating_setpoint', 0.5, 29.5, 0.5, ea.STATE_SET) .withLocalTemperature(ea.STATE).withLocalTemperatureCalibration(-12.5, 5.5, 0.1, ea.STATE_SET) .withSystemMode(['off', 'heat', 'auto'], ea.STATE_SET) .withPreset(['schedule', 'manual', 'holiday', 'boost']), e.numeric('detectwindow_temperature', ea.STATE_SET).withUnit('°C').withDescription('Open window detection temperature') .withValueMin(-10).withValueMax(35), e.numeric('detectwindow_timeminute', ea.STATE_SET).withUnit('min').withDescription('Open window time in minute') .withValueMin(0).withValueMax(1000), e.binary('binary_one', ea.STATE_SET, 'ON', 'OFF').withDescription('Unknown binary one'), e.binary('binary_two', ea.STATE_SET, 'ON', 'OFF').withDescription('Unknown binary two'), e.binary('away_mode', ea.STATE, 'ON', 'OFF').withDescription('Away mode'), e.composite('away_setting', 'away_setting', ea.STATE_SET) .withFeature(e.away_preset_days()).setAccess('away_preset_days', ea.ALL) .withFeature(e.away_preset_temperature()).setAccess('away_preset_temperature', ea.ALL) .withFeature(e.numeric('away_preset_year', ea.ALL).withUnit('year').withDescription('Start away year 20xx')) .withFeature(e.numeric('away_preset_month', ea.ALL).withUnit('month').withDescription('Start away month')) .withFeature(e.numeric('away_preset_day', ea.ALL).withUnit('day').withDescription('Start away day')) .withFeature(e.numeric('away_preset_hour', ea.ALL).withUnit('hour').withDescription('Start away hours')) .withFeature(e.numeric('away_preset_minute', ea.ALL).withUnit('min').withDescription('Start away minutes')), ...['monday', 'tuesday', 'wednesday', 'thursday', 'friday', 'saturday', 'sunday'].map((day) => { const expose = e.composite(day, day, ea.STATE_SET); [1, 2, 3, 4, 5, 6, 7, 8, 9].forEach((i) => { expose.withFeature(e.numeric(`${day}_temp_${i}`, ea.ALL).withValueMin(0.5) .withValueMax(29.5).withValueStep(0.5).withUnit('°C').withDescription(`Temperature ${i}`)); expose.withFeature(e.enum(`${day}_hour_${i}`, ea.STATE_SET, ['00', '01', '02', '03', '04', '05', '06', '07', '08', '09', '10', '11', '12', '13', '14', '15', '16', '17', '18', '19', '20', '21', '22', '23', '24']).withDescription(`Hour TO for temp ${i}`)); expose.withFeature(e.enum(`${day}_minute_${i}`, ea.STATE_SET, ['00', '15', '30', '45']) .withDescription(`Minute TO for temp ${i}`)); }); return expose; }), ], }, ]; exports.default = definitions; module.exports = definitions; //# sourceMappingURL=lidl.js.map