UNPKG

zigbee-herdsman-converters

Version:

Collection of device converters to be used with zigbee-herdsman

212 lines • 9.83 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 () { var ownKeys = function(o) { ownKeys = Object.getOwnPropertyNames || function (o) { var ar = []; for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k; return ar; }; return ownKeys(o); }; return function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]); __setModuleDefault(result, mod); return result; }; })(); Object.defineProperty(exports, "__esModule", { value: true }); exports.definitions = void 0; const zigbee_herdsman_1 = require("zigbee-herdsman"); const fz = __importStar(require("../converters/fromZigbee")); const tz = __importStar(require("../converters/toZigbee")); const constants = __importStar(require("../lib/constants")); const exposes = __importStar(require("../lib/exposes")); const m = __importStar(require("../lib/modernExtend")); const reporting = __importStar(require("../lib/reporting")); const globalStore = __importStar(require("../lib/store")); const utils = __importStar(require("../lib/utils")); const e = exposes.presets; const ea = exposes.access; const kmpcilOptions = { presence_timeout_dc: () => { return e .numeric("presence_timeout_dc", ea.STATE) .withValueMin(60) .withDescription("Time in seconds after which presence is cleared after detecting it (default 60 seconds) while in DC."); }, presence_timeout_battery: () => { return e .numeric("presence_timeout_battery", ea.STATE) .withValueMin(120) .withDescription("Time in seconds after which presence is cleared after detecting it (default 420 seconds) while in Battery."); }, }; function handleKmpcilPresence(model, msg, publish, options, meta) { const useOptionsTimeoutBattery = options?.presence_timeout_battery != null; const timeoutBattery = useOptionsTimeoutBattery ? options.presence_timeout_battery : 420; // 100 seconds by default const useOptionsTimeoutDc = options?.presence_timeout_dc != null; const timeoutDc = useOptionsTimeoutDc ? options.presence_timeout_dc : 60; const mode = meta.state ? meta.state.power_state : false; const timeout = Number(mode ? timeoutDc : timeoutBattery); // Stop existing timer because motion is detected and set a new one. clearTimeout(globalStore.getValue(msg.endpoint, "timer")); const timer = setTimeout(() => publish({ presence: false }), timeout * 1000); globalStore.putValue(msg.endpoint, "timer", timer); return { presence: true }; } const kmpcilConverters = { presence_binary_input: { cluster: "genBinaryInput", type: ["attributeReport", "readResponse"], convert: (model, msg, publish, options, meta) => { const payload = handleKmpcilPresence(model, msg, publish, options, meta); if (msg.data.presentValue !== undefined) { const presentValue = msg.data.presentValue; payload.power_state = (presentValue & 0x01) > 0; payload.occupancy = (presentValue & 0x04) > 0; payload.vibration = (presentValue & 0x02) > 0; } return payload; }, }, presence_power: { cluster: "genPowerCfg", type: ["attributeReport", "readResponse"], options: [kmpcilOptions.presence_timeout_dc(), kmpcilOptions.presence_timeout_battery()], convert: (model, msg, publish, options, meta) => { const payload = handleKmpcilPresence(model, msg, publish, options, meta); if (msg.data.batteryVoltage !== undefined) { payload.voltage = msg.data.batteryVoltage * 100; if (model.meta?.battery?.voltageToPercentage) { // @ts-expect-error ignore payload.battery = utils.batteryVoltageToPercentage(payload.voltage, model.meta.battery.voltageToPercentage); } } return payload; }, }, }; exports.definitions = [ { zigbeeModel: ["RES005"], model: "KMPCIL_RES005", vendor: "KMPCIL", description: "Environment sensor", exposes: [e.battery(), e.temperature(), e.humidity(), e.pressure(), e.occupancy(), e.switch()], fromZigbee: [fz.battery, fz.temperature, fz.humidity, fz.pressure, fz.kmpcil_res005_occupancy, fz.kmpcil_res005_on_off], toZigbee: [tz.kmpcil_res005_on_off], configure: async (device, coordinatorEndpoint) => { const endpoint = device.getEndpoint(8); const binds = [ "genPowerCfg", "msTemperatureMeasurement", "msRelativeHumidity", "msPressureMeasurement", "genBinaryInput", "genBinaryOutput", ]; await reporting.bind(endpoint, coordinatorEndpoint, binds); await reporting.temperature(endpoint); await reporting.humidity(endpoint); const payloadBattery = [ { attribute: "batteryPercentageRemaining", minimumReportInterval: 1, maximumReportInterval: 120, reportableChange: 1, }, ]; await endpoint.configureReporting("genPowerCfg", payloadBattery); const payloadPressure = [ { // 0 = measuredValue, override dataType from int16 to uint16 // https://github.com/Koenkk/zigbee-herdsman/pull/191/files?file-filters%5B%5D=.ts#r456569398 attribute: { ID: 0, type: zigbee_herdsman_1.Zcl.DataType.UINT16 }, minimumReportInterval: 2, maximumReportInterval: constants.repInterval.HOUR, reportableChange: 3, }, ]; await endpoint.configureReporting("msPressureMeasurement", payloadPressure); const options = { disableDefaultResponse: true }; await endpoint.write("genBinaryInput", { 81: { value: 0x01, type: 0x10 } }, options); await endpoint.write("genBinaryInput", { 257: { value: 25, type: 0x23 } }, options); const payloadBinaryInput = [ { attribute: "presentValue", minimumReportInterval: 0, maximumReportInterval: 30, reportableChange: 1, }, ]; await endpoint.configureReporting("genBinaryInput", payloadBinaryInput); await endpoint.write("genBinaryOutput", { 81: { value: 0x01, type: 0x10 } }, options); const payloadBinaryOutput = [ { attribute: "presentValue", minimumReportInterval: 0, maximumReportInterval: 30, reportableChange: 1, }, ]; await endpoint.configureReporting("genBinaryOutput", payloadBinaryOutput); }, extend: [m.illuminance()], }, { zigbeeModel: ["tagv1"], model: "KMPCIL-tag-001", vendor: "KMPCIL", description: "Arrival sensor", fromZigbee: [kmpcilConverters.presence_binary_input, kmpcilConverters.presence_power, fz.temperature], exposes: [ e.battery(), e.presence(), e.binary("power_state", exposes.access.STATE, true, false), e.occupancy(), e.vibration(), e.temperature(), ], toZigbee: [], meta: { battery: { voltageToPercentage: "3V_1500_2800" } }, configure: async (device, coordinatorEndpoint) => { const endpoint = device.getEndpoint(1); for (const cluster of ["msTemperatureMeasurement", "genPowerCfg", "genBinaryInput"]) { // This sleep here(and the sleep) after is to allow the command to be // fully sent to coordinator. In case repeater involved and the repeater // is litted in resources, we may want to give some time so that the sequence of // commands does not overwhelm the repeater. await utils.sleep(2000); await endpoint.bind(cluster, coordinatorEndpoint); } await utils.sleep(1000); const p = reporting.payload("batteryVoltage", 0, 10, 1); await endpoint.configureReporting("genPowerCfg", p); await utils.sleep(1000); const p2 = reporting.payload("presentValue", 0, 300, 1); await endpoint.configureReporting("genBinaryInput", p2); await utils.sleep(1000); await reporting.temperature(endpoint); await endpoint.read("genBinaryInput", ["presentValue"]); }, }, ]; //# sourceMappingURL=kmpcil.js.map