def getDriverDate() { return "2023-04-11" } // **** DATE OF THE DEVICE DRIVER **** // // ^^^^^^^^^^ UPDATE THIS DATE IF YOU MAKE ANY CHANGES ^^^^^^^^^^ /** * Inovelli VZM31-SN Blue Series Zigbee 2-in-1 Dimmer * * Author: Eric Maycock (erocm123) * Contributor: Mark Amber (marka75160) * Platform: Hubitat * * Copyright 2023 Eric Maycock / Inovelli * Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except * in compliance with the License. You may obtain a copy of the License at: * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software distributed under the License is distributed * on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License * for the specific language governing permissions and limitations under the License. * * 2021-12-15(EM) Initial release. * 2021-12-16(EM) Adding configuration options and working on code * 2021-12-20(EM) Adding additional parameters * 2021-12-21(EM) Adding configuration options * 2021-12-22(EM) Cleaning up and consolidating code * 2021-12-23(EM) Adding min & max level parameters * 2021-12-27(EM) Starting to standardize logging * 2021-12-30(EM) Adding options for power type and switch type * 2022-01-03(EM) Change wording and make settings dynamic based on which mode is chosen Fix for parameter 9 not working correctly. More text changes to LED parameters Fixing Set Level when you enter a value > 99 * 2022-01-04(MA) Fix 'div() on null object' error when Param9 is blank. Fix attribute size is 'bits' not 'bytes' * 2022-01-05(MA) Fix ranges for params 1-4. These are slightly different for zigbee(blue) than they were for zwave(red) * 2022-01-06(EM) Updating parse section of code. Requesting firmware version and date. * 2022-01-11(EM) Adding code for firmware update. * 2022-01-20(EM) Changes to make the driver compatible with firmware v5. * 2022-01-20(EM) Some config parameter fixes. Energy measurement was changed to the simple metering cluster. * 2022-01-20(EM) Fix scene reports not working since firmware v5. Need to "Save Preferences" to configure the reporting. * 2022-01-21(MA) Fix typo's and cleanup some of the new parameter descriptions * 2022-01-22(MA) More cleanup of the new parameter descriptions * 2022-01-23(MA) Fix range on Active Energy Report (parameter20) * 2022-01-24(MA) Add Custom Color override for the LED Indicator to allow any color from a standard hue color wheel * 2022-01-25(MA) Parameter259 (On/Off LED mode) should only be visible when in On/Off Mode * 2022-01-26(MA) Restore formatting on custom hue value - its needed to avoid div() errors on null values * Fix issue using dropdown color after clearing custom color * 2022-01-27(MA) Fix setLevel so it uses separate Dim Up / Dim Down rates (parameter1 and parameter5) * 2022-02-01(MA) Lots of tweaks and enhancements to support v6 firmware update * 2022-02-02(EM) Changing speed parameters to match functionality. Updated setLevel method to use firmware speed options (param 1-8) * 2022-02-03(MA) Fix LEDeffect. Clean up some text/spelling/formatting. More enhancements to logging * 2022-02-04(MA) Fix startLevelChange to use dimming params in seconds. Fix 0-99 vs 0-255 scaling on Default Levels (param 13-14) * 2022-02-08(EM) Reverse change that broke LEDeffect * 2022-02-09(MA) Enhance the 0-255 to 0-99 conversion formulas. Fix rounding down 1 to 0. ZigBee LEVEL range is 0x01-0xfe * Fix levelChange UP so it now turns light on if off, levelChange DOWN now stops at 1% not 0%. This matches Red Series dimmers * Removed extra 'rattr' commands from level changing events since the firmware automatically sends them anyway. Doing both was affecting performance * Add lastActivity, lastEvent, and lastRan features to match Red Series dimmers. * Add lastCommand as a feature enhancement over the Red Series. I can easily remove if its not desired * 2022-02-10(MA) Add bind() method to support the Zigbee Bindings app. * 2022-02-11(MA) Add range checking for LEDeffect * 2022-02-14(MA) Arrange procedures in alphabetical order to help match edits between HE and ST. No other code changes * 2022-02-15(MA) Merged most (not all) changes into the ST driver. No changes to this HE driver *** placeholder only * 2022-02-16(MA) Fix button released event. Add Config button Held and Released events. Add digital button support for testing scenes (un-comment in the metadata section to enable) * 2022-02-17(MA) Clean up tab/space and other formatting to simplify diff comparisons between HE and ST groovy drivers * 2022-02-19(MA) Patches for bugs still in v7 firmware. (mostly for parameters 13-14) * 2022-02-20(MA) New 'Reset Parameters' command to reset ALL parameters and not just the ones it thinks have changed. * This is needed when settings in the device don't match settings in the hub (typical after a factory reset and some firmware updates). * It has the option to reset all parameters to their Current Settings on the hub or reset all parameters to their Default values. * 2022-02-21(MA) Excluding Parmeter258 (Switch/Dimmer Output Mode) from Reset All command since it creates confusion with different parameter sets. * Extended delay time between bulk parameter changes to try and avoid lockups * 2022-02-22(MA) Hotfix to remove spaces that broke multi-tap button events * 2022-02-24(MA) Replace recently added 'Reset Parameters' command with new options for the existing Config command. Options to reset all settings to Default or force All current settings to device. * Add 'switchMode' attribute so the current Operating Mode (Dimmer or On/Off) is diplayed under Current States. * Adjusted the 0-99 ranges to 0-100 for consistency. * Update local settings variable whenever a device report is received and the device value is different than the local setting * Refresh command now includes 'get all attributes' so a refresh will ensure all state variables match whats in the device * Created new 'calculateSize' common method to use wherever a bitsize needs to be converted to a hex DataType * Add temporary hack to prevent parameter 22 from getting set to the same value as this causes freeze in v7 firmware. Will remove hack when fixed in future firmware * 2022-02-27(MA) Add default delay to Refresh commands so we're not waiting 2 seconds on every attribute * 2022-02-28(EM) Adding individual LED notifications, modifying parameters for firmware v8, & fixing issue that was preventing "initialize()" commands from being sent. * 2022-03-01(MA) Rename with official model name in preparation for production release and standardize across other drivers (like the VZM35 Fan Switch) * Add Tertiary colors to LED Indicator options. More tweaks for v8 firmware. Hotfix: percent conversion fix and p22 hack removal * 2022-03_02(MA) More detailed parsing of Zigbee Description Map reports * Display Level values as 0-100% instead of 0-255. * Allow default values for LEDeffects - easier to enable/disable with just one or two clicks * Move "Save Preferences" code from config() to updated(), move bindings from initialize() to configure() * Created Refresh-ALL and simplified Config-ALL & Config-DEFAULT commands * CLICKING ON CONFIGURE WILL RE-ESTABLISH ZIGBEE REPORT BINDINGS AFTER FACTORY RESET OF SWITCH * 2022-03-03(MA) Hotfix: in some edge cases settings were not getting sent. * 2022-03-04(MA) Parameter21 auto-senses Neutral and is read-only. Add powerSource attribute to display what the switch detected instead of what the user selected. * 2022-03-06(MA) change hubitat hexutils to zigbee hexutils for compatibility with ST driver (which doesn't have the hubitat libraries) * 2022-03-07(MA) Even more detailed parsing. Stub in some code for possible future "Preset Level" command * 2022-03-08(MA) Hotfix: found another case where some settings were not getting sent to device, and remove extra level report from StopLevelChange * 2022-03-09(MA) various tweaks and code optimizations to help merge with Fan driver. Remove lastRan carryover from Red Series as its not used with the Blue Series * 2022-03-10(MA) add null check before sending attribute * 2022-03-12(MA) move switch config options to the top. Add doubleTapped event for Config button. fix another rounding error * 2022-03-16(MA) remove doubleTap and add full 5-tap capability for Config button. * 2022-03-21(MA) replace empty ENUM values with " " and fix basic on/off via rules and dashboard * 2022-03-26(MA) updated for v9 firmware * 2022-03-28(MA) added Alexa clusters to fingerprint ID, cleaned up a little code to match a little better with ST driver * 2022-04-11(MA) sync up with changes to Zigbee Fan driver for consistency. No functional changes to dimmer * 2022-04-21(MA) change default LED intensity (params 97-98 ) to match PRD (33%/1%). Also change switch mode (param 258) default to On/Off * 2022-04-25(MA) updated for v10 (0x0A) firmware * 2022-05-03(MA) some additional support for zigbee binding app and a couple small tweaks to text and logging * 2022-05-04(MA) added logging for Alexa Cluster * 2022-05-30(MA) merge with changes to zigbee fan v4 * 2022-05-31(MA) fix bug with dimRate in StartLevelChange * 2022-06-04(MA) added parsing and logging for binding clusters - still under development * 2022-06-06(MA) more updates to stay in sync with Fan v4 firmware * 2022-06-07(MA) fix bug with null level on initial pairing * 2022-06-08(MA) lots of cleanup and minor bug fixes; add some code to detect model and more merges with VZM35 * 2022-06-10(MA) merge with changes to 2022-06-10 VZM31-SN * 2022-06-18(MA) fix startLevelChange to accept a duration value; fix setPrivateCluster and setZigbeeAttribute; standardize all variables to camelCase * 2022-06-19(MA) minor adjustment to percent conversions to be consistent with Fan v4 firmware * 2022-06-20(MA) fix condition where user enters decimal string (e.g. "12.3") for a parameter * 2022-06-21(MA) enhanced logic for Fan speed changes when using setLevel() * 2022-06-23(MA) add weblink to Hue Color Wheel for Custom LED color; add color to some log entries * 2022-06-27(MA) param 95-98 titles change color to match selection; Add common Led groupings for the Led Effect One notification * 2022-07-03(MA) new Refresh User option to only refresh User-changed settings; enhanced support for custom LED bar colors * 2022-07-08(MA) add param#261, add Aurora Effect, and other updates for firmware v1.11 * 2022-07-15(MA) cleanup and remove some unneeded debug code; add grey background to white text; remove unused report bindings * 2022-07-22(MA) don't request ramp rate for fan; more cleanup for production * 2022-07-27(MA) updates for v1.12 firmware; remove details from last.command to keep it simple (details are in log.info) * 2022-07-29(MA) add Quick Start to parsing/reporting; add driverDate variable so it can be seen on the Device page in Hubitat * 2022-07-30(MA) fix params 9-10 so they send the full 0-255 to the switch; don't log Config() calls if info logging is off * 2022-08-02(MA) fix fan high speed not working with neutral; don't display min-level and max-level when in on/off mode; fix p9-p10 so they only get written if changed * 2022-08-09(MA) added Trace logging; setCluster/setAttribute commands will Get current value if you leave Value blank * 2022-08-14(MA) emulate QuickStart for dimmer(can be disabled); add presetLevel command to use in Rule Machine - can also be done with setPrivateCluster custom command * 2022-11-02(EM) added warning for firmware update and requirement for double click. Enabled maximum level setting in on/off mode for "problem load" troubleshooting in 3-way dumb mode * 2022-11-03(MA) updates for fw2.05: addeded param 262; added additional LED effects rising,falling,fast/slow, etc. * 2022-11-04(MA) fix 'siren' fast/slow effects (18/19) were backwards * 2022-11-05(MA) fix selection of multiple individual leds with ledEffectOne (e.g.1357 to select the odd leds and 246 to select the even leds) * 2022-11-17(MA) more fixes for when user enters decimal (floating) values for an integer parameter * 2022-11-18(MA) fix startLevelChange with null duration * 2022-11-24(MA) improvements to quickStartEmulation * 2022-11-26(MA) fix Config Default not defaulting all parameters * 2022-12-12(MA) add presetLevel command; allow param15 to be set in on/off mode; workaround for setLevel bug in firmware (firmware ignores off-on duration) * 2022-12-26(MA) sync with updates to Fan driver * 2022-12-28(MA) add Identify command; add device names to trace logging; add remoteControl command to allow re-enabling if Remote Protection (P257) is disabled * 2022-12-30(MA) add more context to LED Effect dropdowns; add more detail to state.lastCommand * 2022-12-31(MA) more changes to quickStart emulation. * 2023-01-01(MA) enhanced Unknown Cluster/Attribute logging; fix typo in remoteControl command * 2023-01-04(MA) P257 is read-only, must be changed with remoteControl command * 2023-01-08(MA) HOTFIX for ledEffect Integer/String error when called from Rule Machine * 2023-01-10(MA) improved ledEffect reporting in log and state variables * 2023-01-11(MA) cleanup sendEvent doesn't use "displayed:false" on Hubitat * 2023-01-12(MA) change QuickStart description to experimental * 2023-01-18(MA) updates for Dimmer v2.10 firmware * 2023-01-22(MA) fix ledEffect sendEvent * 2023-01-24(MA) decrease defaultDelay slightly and increase longDelay slightly; small tweak to setLevel * 2023-02-21(MA) add setParameter/getParameter; add Aux Unique Scenes option; add state.dimmingMethod (leading/trailing) * 2023-02-23(MA) fix Leading/Trailing error in non-neutral; misc code cleanup; more standardization between the different VMark devices * 2023-02-26(MA) fix missing preferences; fix state.auxType; enhance parsing of Unknown Command and Unknown Attribute * 2023-03-01(MA) synchronize all changes up to this point between VZM31, VZM35, and VZW31; includes all current firmware changes * 2023-03-12(MA) add params 55,56; fix minor bugs and typos; prep for production firmware release. * 2023-03-23(MA) rename bindInitiator/bindTarget to bindSlave/bindSource to reduce confusion * 2023-03-31(MA) display effect name instead of number in ledEffect attribute * 2023-04-03(MA) add parameters 25,100,125 (for Dimmer v2.14+ firmware) * 2023-04-07(EM) Adding group binding support. See: https://community.inovelli.com/t/how-to-s-setup-zigbee-group-binding-hubitat/13909/1 * 2023-04-09(MA) added range/error checking to group binding; enhanced cluster name reporting; remove "beta" designation for v2.14 public release * 2023-04-10(MA) Hotfix P25 is 1-bit boolean, not 1-byte integer * 2023-04-11(MA) Hotfix Add parsing for P125; fix P55-56 not scaling to percent * * !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! * !! !! * !! DON'T FORGET TO UPDATE THE DRIVER DATE AT THE TOP OF THIS PAGE !! * !! !! * !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! **/ import groovy.json.JsonSlurper import groovy.json.JsonOutput import groovy.transform.Field import hubitat.helper.ColorUtils //import hubitat.helper.HexUtils import java.security.MessageDigest metadata { definition (name: "Inovelli Dimmer 2-in-1 Blue Series VZM31-SN", namespace: "InovelliUSA", author: "E.Maycock/M.Amber", filename: "Inovelli-zigbee-2-in-1-dimmer", importUrl:"https://raw.githubusercontent.com/InovelliUSA/Hubitat/master/Drivers/inovelli-dimmer-blue-series-vzm31-sn.src/inovelli-dimmer-blue-series-vzm31-sn.groovy" ) { capability "Actuator" //device can "do" something (has commands) capability "Sensor" //device can "report" something (has attributes) capability "ChangeLevel" capability "Configuration" capability "EnergyMeter" //capability "FanControl" capability "HoldableButton" capability "LevelPreset" capability "PowerMeter" capability "PushableButton" capability "Refresh" capability "ReleasableButton" //capability "SignalStrength" //placeholder for future testing to see if this can be implemented capability "Switch" capability "SwitchLevel" attribute "lastButton", "String" //last button event attribute "ledEffect", "String" //LED effect that was requested attribute "numberOfBindings", "String" //(read only) attribute "smartBulb", "String" //Smart Bulb mode enabled or disabled //attribute "smartFan", "String" //Smart Fan mode enabled or disabled attribute "switchMode", "String" //Dimmer or On/Off only // Uncomment these lines if you would like to test your scenes with digital button presses. /** command "pressUpX1" command "pressDownX1" command "pressUpX2" command "pressDownX2" command "pressUpX3" command "pressDownX3" command "pressUpX4" command "pressDownX4" command "pressUpX5" command "pressDownX5" command "holdUp" command "holdDown" command "releaseUp" command "releaseDown" command "pressConfigX1" command "pressConfigX2" command "pressConfigX3" command "pressConfigX4" command "pressConfigX5" command "holdConfig" command "releaseConfig" **/ command "bind", [[name: "Command String", type: "STRING", description: "passthru for Binding Apps but may be used to manually enter ZDO Bind/Unbind commands"]] command "bindSlave", [[name: "click on Slave switch to initiate binding with Source switch"]] command "bindSource", [[name: "click on Source switch to complete binding with Slave switch"]] command "configure", [[name: "Option", type: "ENUM", description: "blank=current states only, User=user changed settings only, All=configure all settings, Default=set all settings to default", constraints: [" ","User","All","Default"]]] command "identify", [[name: "Seconds", type: "NUMBER", description: "number of seconds to blink the LED bar so it can be identified (leave blank to see remaining seconds in the logs)"]] command "initialize", [[name: "clear state variables, clear LED notifications, refresh current states"]] command "ledEffectAll", [[name: "Effect*",type:"ENUM", description: "255=Stop, 1=Solid, 2=Fast Blink, 3=Slow Blink, 4=Pulse, 5=Chase, 6=Open/Close, 7=Small-to-Big, 8=Aurora, 9=Slow Falling, 10=Medium Falling, 11=Fast Falling, 12=Slow Rising, 13=Medium Rising, 14=Fast Rising, 15=Medium Blink, 16=Slow Chase, 17=Fast Chase, 18=Fast Siren, 19=Slow Siren, 0=LEDs off", constraints: ["255=Stop","1=Solid","2=Fast Blink","3=Slow Blink","4=Pulse","5=Chase","6=Open/Close","7=Small-to-Big","8=Aurora","9=Slow Falling","10=Medium Falling","11=Fast Falling","12=Slow Rising","13=Medium Rising","14=Fast Rising","15=Medium Blink","16=Slow Chase","17=Fast Chase","18=Fast Siren","19=Slow Siren","0=LEDs off"]], [name: "Color", type:"NUMBER", description: "0-254=Hue Color, 255=White, default=Red"], [name: "Level", type:"NUMBER", description: "0-100=LED Intensity, default=100"], [name: "Duration", type:"NUMBER", description: "1-60=seconds, 61-120=1-120 minutes, 121-254=1-134 hours, 255=Indefinitely, default=60"]] command "ledEffectOne", [[name: "LEDnum*",type:"ENUM", description: "LED 1-7", constraints: ["7 (top)","6","5","4 (middle)","3","2","1 (bottom)","123 (bottom half)","567 (top half)","12 (bottom 3rd)","345 (middle 3rd)","67 (top 3rd)","147 (bottom-middle-top)","1357 (odd)","246 (even)"]], [name: "Effect*",type:"ENUM", description: "255=Stop, 1=Solid, 2=Fast Blink, 3=Slow Blink, 4=Pulse, 5=Chase, 6=Falling, 7=Rising, 8=Aurora, 0=LED off", constraints: ["255=Stop","1=Solid","2=Fast Blink","3=Slow Blink","4=Pulse","5=Chase","6=Falling","7=Rising","8=Aurora","0=LED off"]], [name: "Color", type:"NUMBER", description: "0-254=Hue Color, 255=White, default=Red"], [name: "Level", type:"NUMBER", description: "0-100=LED Intensity, default=100"], [name: "Duration", type:"NUMBER", description: "1-60=seconds, 61-120=1-120 minutes, 121-254=1-134 hours, 255=Indefinitely, default=60"]] command "presetLevel", [[name: "Level", type: "NUMBER", description: "Level to preset (1 to 101)"]] command "refresh", [[name: "Option", type: "ENUM", description: "blank=current states only, User=user changed settings only, All=refresh all settings", constraints: [" ","User","All"]]] command "remoteControl", [[name: "Option*", type: "ENUM", description: "change the setting of Remote Protection (P257)", constraints: [" ","Enabled","Disabled"]]] //Fan does not support power/energy reporting but Dimmer does command "resetEnergyMeter" command "setParameter", [[name: "Parameter*",type:"NUMBER", description: "Parameter number"], [name: "Raw Value", type:"NUMBER", description: "Value for the parameter (leave blank to get current value)"], [name: "Enter the internal raw value. Percentages and Color Hues are entered as 0-255. Leave blank to get current value"]] //uncomment this command if you need it for backward compatibility //command "setPrivateCluster", [[name: "Number*",type:"NUMBER", description: "setPrivateCluster is DEPRECIATED. Use setParameter instead"], // [name: "Value*", type:"NUMBER", description: "setPrivateCluster is DEPRECIATED. Use setParameter instead"], // [name: "Size*", type:"ENUM", description: "setPrivateCluster is DEPRECIATED. Use setParameter instead", constraints: ["8", "16","1"]], // [name: "DEPRECIATED", description: "This command is depreciated. Use setParameter instead"]] //Dimmer does not support setSpeed commands but Fan does //command "setSpeed", [[name: "FanSpeed*", type: "ENUM", constraints: ["off","low","medium-low","medium","medium-high","high","up","down"]]] command "setZigbeeAttribute", [[name: "Cluster*", type:"NUMBER", description: "Cluster (in decimal) ex. Inovelli Private Cluster=0xFC31 input 64561"], [name: "Attribute*",type:"NUMBER", description: "Attribute (in decimal) ex. 0x0100 input 256"], [name: "Value", type:"NUMBER", description: "Enter the value (in decimal, ex. 0x0F input 15) Leave blank to get current value without changing it"], [name: "Size", type:"ENUM", description: "8=uint8, 16=uint16, 32=unint32, 1=bool",constraints: ["8", "16","32","1"]]] command "startLevelChange", [[name: "Direction*",type:"ENUM", description: "Direction for level change", constraints: ["up","down"]], [name: "Duration", type:"NUMBER", description: "Transition duration in seconds"]] command "toggle" command "updateFirmware", [[name: "Firmware in this channel may be \"beta\" quality. Double-click \"Update Firmware\" to proceed"]] fingerprint profileId:"0104", endpointId:"01", inClusters:"0000,0003,0004,0005,0006,0008,0702,0B04,0B05,FC57,FC31", outClusters:"0003,0019", model:"VZM31-SN", manufacturer:"Inovelli" // fingerprint profileId:"0104", endpointId:"02", inClusters:"0000,0003", outClusters:"0003,0019,0006,0008", model:"VZM31-SN", manufacturer:"Inovelli" // fingerprint profileId:"0104", endpointId:"01", inClusters:"0000,0003,0004,0005,0006,0008,0702,0B04,FC31", outClusters:"0003,0019", model:"VZM31-SN", manufacturer:"Inovelli" } preferences { getParameterNumbers().each{ i -> switch(configParams["parameter${i.toString().padLeft(3,"0")}"].type){ case "number": switch(i){ case 23: //special case for Quick Start is below break case 51: //Device Bind Number input "parameter${i}", "number", title: "${i}. " + green(bold(configParams["parameter${i.toString().padLeft(3,"0")}"].name)), description: italic(configParams["parameter${i.toString().padLeft(3,"0")}"].description + "
Range=" + configParams["parameter${i.toString().padLeft(3,"0")}"].range), //defaultValue: configParams["parameter${i.toString().padLeft(3,"0")}"].default, range: configParams["parameter${i.toString().padLeft(3,"0")}"].range break default: input "parameter${i}", "number", title: "${i}. " + bold(configParams["parameter${i.toString().padLeft(3,"0")}"].name), description: italic(configParams["parameter${i.toString().padLeft(3,"0")}"].description + "
Range=" + configParams["parameter${i.toString().padLeft(3,"0")}"].range + " Default=" + configParams["parameter${i.toString().padLeft(3,"0")}"].default), //defaultValue: configParams["parameter${i.toString().padLeft(3,"0")}"].default, range: configParams["parameter${i.toString().padLeft(3,"0")}"].range break } break case "enum": switch(i){ case 23: //special case for Quick Start is below break case 21: //Power Source case 157: //Remote Protection case 257: //Remote Protection input "parameter${i}", "enum", title: "${i}. " + green(bold(configParams["parameter${i.toString().padLeft(3,"0")}"].name)), description: italic(configParams["parameter${i.toString().padLeft(3,"0")}"].description), //defaultValue: configParams["parameter${i.toString().padLeft(3,"0")}"].default, options: configParams["parameter${i.toString().padLeft(3,"0")}"].range break case 22: //Aux Type case 52: //Smart Bulb Mode case 158: //Switch Mode case 258: //Switch Mode input "parameter${i}", "enum", title: "${i}. " + crimson(bold(configParams["parameter${i.toString().padLeft(3,"0")}"].name)), description: italic(configParams["parameter${i.toString().padLeft(3,"0")}"].description), //defaultValue: configParams["parameter${i.toString().padLeft(3,"0")}"].default, options: configParams["parameter${i.toString().padLeft(3,"0")}"].range break case 95: case 96: //special case for custom color is below break default: input "parameter${i}", "enum", title: "${i}. " + bold(configParams["parameter${i.toString().padLeft(3,"0")}"].name), description: italic(configParams["parameter${i.toString().padLeft(3,"0")}"].description), //defaultValue: configParams["parameter${i.toString().padLeft(3,"0")}"].default, options: configParams["parameter${i.toString().padLeft(3,"0")}"].range break } break } if (i==23) { //quickStart is implemented in firmware for the fan, emulated in this driver for 2-in-1 Dimmer if (state.model?.substring(0,5)!="VZM35") { input "parameter${i}", "number", title: "${i}. " + orangeRed(bold(configParams["parameter${i.toString().padLeft(3,"0")}"].name + " Level")), description: orangeRed(italic(configParams["parameter${i.toString().padLeft(3,"0")}"].description + "
Range=" + configParams["parameter${i.toString().padLeft(3,"0")}"].range + " Default=" + configParams["parameter${i.toString().padLeft(3,"0")}"].default)), //defaultValue: configParams["parameter${i.toString().padLeft(3,"0")}"].default, range: configParams["parameter${i.toString().padLeft(3,"0")}"].range } else { input "parameter${i}", "enum", title: "${i}. " + darkSlateBlue(bold(configParams["parameter${i.toString().padLeft(3,"0")}"].name + " Duration")), description: italic(configParams["parameter${i.toString().padLeft(3,"0")}"].description), //defaultValue: configParams["parameter${i.toString().padLeft(3,"0")}"].default, options: configParams["parameter${i.toString().padLeft(3,"0")}"].range } } if (i==95 || i==96) { if ((i==95 && parameter95custom==null)||(i==96 && parameter96custom==null)){ input "parameter${i}", "enum", title: "${i}. " + hue((settings?."parameter${i}"!=null?settings?."parameter${i}":configParams["parameter${i.toString().padLeft(3,"0")}"].default)?.toInteger(), bold(configParams["parameter${i.toString().padLeft(3,"0")}"].name)), description: italic(configParams["parameter${i.toString().padLeft(3,"0")}"].description), //defaultValue: configParams["parameter${i.toString().padLeft(3,"0")}"].default, options: configParams["parameter${i.toString().padLeft(3,"0")}"].range } else { input "parameter${i}", "enum", title: "${i}. " + hue((settings?."parameter${i}"!=null?settings?."parameter${i}":configParams["parameter${i.toString().padLeft(3,"0")}"].default)?.toInteger(), strike(configParams["parameter${i.toString().padLeft(3,"0")}"].name)) + hue((settings?."parameter${i}custom"!=null?(settings."parameter${i}custom"/360*255):configParams["parameter${i.toString().padLeft(3,"0")}"].default)?.toInteger(), italic(bold(" Overridden by Custom Hue Value"))), description: italic(configParams["parameter${i.toString().padLeft(3,"0")}"].description), //defaultValue: configParams["parameter${i.toString().padLeft(3,"0")}"].default, options: configParams["parameter${i.toString().padLeft(3,"0")}"].range } input "parameter${i}custom", "number", title: settings?."parameter${i}custom"!=null? (hue((settings."parameter${i}custom"/360*255)?.toInteger(), bold("Custom " + configParams["parameter${i.toString().padLeft(3,"0")}"].name))): ( bold("Custom " + configParams["parameter${i.toString().padLeft(3,"0")}"].name)), description: italic("Hue value to override " + configParams["parameter${i.toString().padLeft(3,"0")}"].name+".
Range: 0-360 chosen from a"+ underline(''' '''+ hue(0," h")+hue(20,"u")+hue(40,"e")+hue(60," c")+hue(80,"o")+hue(100,"l")+hue(120,"o")+hue(140,"r")+hue(160," w")+hue(180,"h")+hue(200,"e")+hue(220,"e")+hue(240,"l")+"")), required: false, range: "0..360" } } input name: "groupbinding1", type: "number", title: bold("Group Bind # 1"), description: italic("Enter the Zigbee Group ID or leave blank to UNBind"), defaultValue: null, range: "1..65527" input name: "groupbinding2", type: "number", title: bold("Group Bind # 2"), description: italic("Enter the Zigbee Group ID or leave blank to UNBind"), defaultValue: null, range: "1..65527" input name: "groupbinding3", type: "number", title: bold("Group Bind # 3"), description: italic("Enter the Zigbee Group ID or leave blank to UNBind"), defaultValue: null, range: "1..65527" input name: "infoEnable", type: "bool", title: bold("Enable Info Logging"), defaultValue: true input name: "traceEnable", type: "bool", title: bold("Enable Trace Logging"), defaultValue: false input name: "debugEnable", type: "bool", title: bold("Enable Debug Logging"), defaultValue: false input name: "disableInfoLogging", type: "number", title: bold("Disable Info Logging after this number of minutes"), description: italic("(0=Do not disable)"), defaultValue: 20 input name: "disableTraceLogging", type: "number", title: bold("Disable Trace Logging after this number of minutes"), description: italic("(0=Do not disable)"), defaultValue: 10 input name: "disableDebugLogging", type: "number", title: bold("Disable Debug Logging after this number of minutes"), description: italic("(0=Do not disable)"), defaultValue: 5 } } def getParameterNumbers() { //controls which options are available depending on whether the device is configured as a switch or a dimmer. if (parameter258 == "1") //on/off mode return [258,22,52,10,11,12,15,17,18,19,20,21,25,50,51,95,96,97,98,100,125,256,257,259,260,261,262] else //dimmer mode return [258,22,52,1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,17,18,19,20,21,25,50,51,53,54,55,56,95,96,97,98,100,125,256,257,260,262] } @Field static Integer defaultDelay = 333 //default delay to use for zigbee commands (in milliseconds) @Field static Integer longDelay = 3000 //long delay to use for changing modes (in milliseconds) @Field static Integer defaultQuickLevel=50 //default startup level for QuickStart emulation @Field static Map configParams = [ parameter001 : [ number: 1, name: "Dimming Speed - Up (Remote)", description: "This changes the speed that the light dims up when controlled from the hub. A setting of 'instant' turns the light immediately on. Increasing the value slows down the transition speed.
Default=2.5s", range: ["0":"instant","5":"500ms","6":"600ms","7":"700ms","8":"800ms","9":"900ms","10":"1.0s","11":"1.1s","12":"1.2s","13":"1.3s","14":"1.4s","15":"1.5s","16":"1.6s","17":"1.7s","18":"1.8s","19":"1.9s","20":"2.0s","21":"2.1s","22":"2.2s","23":"2.3s","24":"2.4s","25":"2.5s (default)","26":"2.6s","27":"2.7s","28":"2.8s","29":"2.9s","30":"3.0s","31":"3.1s","32":"3.2s","33":"3.3s","34":"3.4s","35":"3.5s","36":"3.6s","37":"3.7s","38":"3.8s","39":"3.9s","40":"4.0s","41":"4.1s","42":"4.2s","43":"4.3s","44":"4.4s","45":"4.5s","46":"4.6s","47":"4.7s","48":"4.8s","49":"4.9s","50":"5.0s","51":"5.1s","52":"5.2s","53":"5.3s","54":"5.4s","55":"5.5s","56":"5.6s","57":"5.7s","58":"5.8s","59":"5.9s","60":"6.0s","61":"6.1s","62":"6.2s","63":"6.3s","64":"6.4s","65":"6.5s","66":"6.6s","67":"6.7s","68":"6.8s","69":"6.9s","70":"7.0s","71":"7.1s","72":"7.2s","73":"7.3s","74":"7.4s","75":"7.5s","76":"7.6s","77":"7.7s","78":"7.8s","79":"7.9s","80":"8.0s","81":"8.1s","82":"8.2s","83":"8.3s","84":"8.4s","85":"8.5s","86":"8.6s","87":"8.7s","88":"8.8s","89":"8.9s","90":"9.0s","91":"9.1s","92":"9.2s","93":"9.3s","94":"9.4s","95":"9.5s","96":"9.6s","97":"9.7s","98":"9.8s","99":"9.9s","100":"10.0s","101":"10.1s","102":"10.2s","103":"10.3s","104":"10.4s","105":"10.5s","106":"10.6s","107":"10.7s","108":"10.8s","109":"10.9s","110":"11.0s","111":"11.1s","112":"11.2s","113":"11.3s","114":"11.4s","115":"11.5s","116":"11.6s","117":"11.7s","118":"11.8s","119":"11.9s","120":"12.0s","121":"12.1s","122":"12.2s","123":"12.3s","124":"12.4s","125":"12.5s","126":"12.6s"], default: 25, size: 8, type: "enum", value: null ], parameter002 : [ number: 2, name: "Dimming Speed - Up (Local)", description: "This changes the speed that the light dims up when controlled at the switch. A setting of 'instant' turns the light immediately on. Increasing the value slows down the transition speed.
Default=Sync with parameter1", range: ["0":"instant","5":"500ms","6":"600ms","7":"700ms","8":"800ms","9":"900ms","10":"1.0s","11":"1.1s","12":"1.2s","13":"1.3s","14":"1.4s","15":"1.5s","16":"1.6s","17":"1.7s","18":"1.8s","19":"1.9s","20":"2.0s","21":"2.1s","22":"2.2s","23":"2.3s","24":"2.4s","25":"2.5s","26":"2.6s","27":"2.7s","28":"2.8s","29":"2.9s","30":"3.0s","31":"3.1s","32":"3.2s","33":"3.3s","34":"3.4s","35":"3.5s","36":"3.6s","37":"3.7s","38":"3.8s","39":"3.9s","40":"4.0s","41":"4.1s","42":"4.2s","43":"4.3s","44":"4.4s","45":"4.5s","46":"4.6s","47":"4.7s","48":"4.8s","49":"4.9s","50":"5.0s","51":"5.1s","52":"5.2s","53":"5.3s","54":"5.4s","55":"5.5s","56":"5.6s","57":"5.7s","58":"5.8s","59":"5.9s","60":"6.0s","61":"6.1s","62":"6.2s","63":"6.3s","64":"6.4s","65":"6.5s","66":"6.6s","67":"6.7s","68":"6.8s","69":"6.9s","70":"7.0s","71":"7.1s","72":"7.2s","73":"7.3s","74":"7.4s","75":"7.5s","76":"7.6s","77":"7.7s","78":"7.8s","79":"7.9s","80":"8.0s","81":"8.1s","82":"8.2s","83":"8.3s","84":"8.4s","85":"8.5s","86":"8.6s","87":"8.7s","88":"8.8s","89":"8.9s","90":"9.0s","91":"9.1s","92":"9.2s","93":"9.3s","94":"9.4s","95":"9.5s","96":"9.6s","97":"9.7s","98":"9.8s","99":"9.9s","100":"10.0s","101":"10.1s","102":"10.2s","103":"10.3s","104":"10.4s","105":"10.5s","106":"10.6s","107":"10.7s","108":"10.8s","109":"10.9s","110":"11.0s","111":"11.1s","112":"11.2s","113":"11.3s","114":"11.4s","115":"11.5s","116":"11.6s","117":"11.7s","118":"11.8s","119":"11.9s","120":"12.0s","121":"12.1s","122":"12.2s","123":"12.3s","124":"12.4s","125":"12.5s","126":"12.6s","127":"Sync with parameter1"], default: 127, size: 8, type: "enum", value: null ], parameter003 : [ number: 3, name: "Ramp Rate - Off to On (Remote)", description: "This changes the speed that the light turns on when controlled from the hub. A setting of 'instant' turns the light immediately on. Increasing the value slows down the transition speed.
Default=Sync with parameter1", range: ["0":"instant","5":"500ms","6":"600ms","7":"700ms","8":"800ms","9":"900ms","10":"1.0s","11":"1.1s","12":"1.2s","13":"1.3s","14":"1.4s","15":"1.5s","16":"1.6s","17":"1.7s","18":"1.8s","19":"1.9s","20":"2.0s","21":"2.1s","22":"2.2s","23":"2.3s","24":"2.4s","25":"2.5s","26":"2.6s","27":"2.7s","28":"2.8s","29":"2.9s","30":"3.0s","31":"3.1s","32":"3.2s","33":"3.3s","34":"3.4s","35":"3.5s","36":"3.6s","37":"3.7s","38":"3.8s","39":"3.9s","40":"4.0s","41":"4.1s","42":"4.2s","43":"4.3s","44":"4.4s","45":"4.5s","46":"4.6s","47":"4.7s","48":"4.8s","49":"4.9s","50":"5.0s","51":"5.1s","52":"5.2s","53":"5.3s","54":"5.4s","55":"5.5s","56":"5.6s","57":"5.7s","58":"5.8s","59":"5.9s","60":"6.0s","61":"6.1s","62":"6.2s","63":"6.3s","64":"6.4s","65":"6.5s","66":"6.6s","67":"6.7s","68":"6.8s","69":"6.9s","70":"7.0s","71":"7.1s","72":"7.2s","73":"7.3s","74":"7.4s","75":"7.5s","76":"7.6s","77":"7.7s","78":"7.8s","79":"7.9s","80":"8.0s","81":"8.1s","82":"8.2s","83":"8.3s","84":"8.4s","85":"8.5s","86":"8.6s","87":"8.7s","88":"8.8s","89":"8.9s","90":"9.0s","91":"9.1s","92":"9.2s","93":"9.3s","94":"9.4s","95":"9.5s","96":"9.6s","97":"9.7s","98":"9.8s","99":"9.9s","100":"10.0s","101":"10.1s","102":"10.2s","103":"10.3s","104":"10.4s","105":"10.5s","106":"10.6s","107":"10.7s","108":"10.8s","109":"10.9s","110":"11.0s","111":"11.1s","112":"11.2s","113":"11.3s","114":"11.4s","115":"11.5s","116":"11.6s","117":"11.7s","118":"11.8s","119":"11.9s","120":"12.0s","121":"12.1s","122":"12.2s","123":"12.3s","124":"12.4s","125":"12.5s","126":"12.6s","127":"Sync with parameter1"], default: 127, size: 8, type: "enum", value: null ], parameter004 : [ number: 4, name: "Ramp Rate - Off to On (Local)", description: "This changes the speed that the light turns on when controlled at the switch. A setting of 'instant' turns the light immediately on. Increasing the value slows down the transition speed.
Default=Sync with parameter3", range: ["0":"instant","5":"500ms","6":"600ms","7":"700ms","8":"800ms","9":"900ms","10":"1.0s","11":"1.1s","12":"1.2s","13":"1.3s","14":"1.4s","15":"1.5s","16":"1.6s","17":"1.7s","18":"1.8s","19":"1.9s","20":"2.0s","21":"2.1s","22":"2.2s","23":"2.3s","24":"2.4s","25":"2.5s","26":"2.6s","27":"2.7s","28":"2.8s","29":"2.9s","30":"3.0s","31":"3.1s","32":"3.2s","33":"3.3s","34":"3.4s","35":"3.5s","36":"3.6s","37":"3.7s","38":"3.8s","39":"3.9s","40":"4.0s","41":"4.1s","42":"4.2s","43":"4.3s","44":"4.4s","45":"4.5s","46":"4.6s","47":"4.7s","48":"4.8s","49":"4.9s","50":"5.0s","51":"5.1s","52":"5.2s","53":"5.3s","54":"5.4s","55":"5.5s","56":"5.6s","57":"5.7s","58":"5.8s","59":"5.9s","60":"6.0s","61":"6.1s","62":"6.2s","63":"6.3s","64":"6.4s","65":"6.5s","66":"6.6s","67":"6.7s","68":"6.8s","69":"6.9s","70":"7.0s","71":"7.1s","72":"7.2s","73":"7.3s","74":"7.4s","75":"7.5s","76":"7.6s","77":"7.7s","78":"7.8s","79":"7.9s","80":"8.0s","81":"8.1s","82":"8.2s","83":"8.3s","84":"8.4s","85":"8.5s","86":"8.6s","87":"8.7s","88":"8.8s","89":"8.9s","90":"9.0s","91":"9.1s","92":"9.2s","93":"9.3s","94":"9.4s","95":"9.5s","96":"9.6s","97":"9.7s","98":"9.8s","99":"9.9s","100":"10.0s","101":"10.1s","102":"10.2s","103":"10.3s","104":"10.4s","105":"10.5s","106":"10.6s","107":"10.7s","108":"10.8s","109":"10.9s","110":"11.0s","111":"11.1s","112":"11.2s","113":"11.3s","114":"11.4s","115":"11.5s","116":"11.6s","117":"11.7s","118":"11.8s","119":"11.9s","120":"12.0s","121":"12.1s","122":"12.2s","123":"12.3s","124":"12.4s","125":"12.5s","126":"12.6s","127":"Sync with parameter3"], default: 127, size: 8, type: "enum", value: null ], parameter005 : [ number: 5, name: "Dimming Speed - Down (Remote)", description: "This changes the speed that the light dims down when controlled from the hub. A setting of 'instant' turns the light immediately off. Increasing the value slows down the transition speed.
Default=Sync with parameter1", range: ["0":"instant","5":"500ms","6":"600ms","7":"700ms","8":"800ms","9":"900ms","10":"1.0s","11":"1.1s","12":"1.2s","13":"1.3s","14":"1.4s","15":"1.5s","16":"1.6s","17":"1.7s","18":"1.8s","19":"1.9s","20":"2.0s","21":"2.1s","22":"2.2s","23":"2.3s","24":"2.4s","25":"2.5s","26":"2.6s","27":"2.7s","28":"2.8s","29":"2.9s","30":"3.0s","31":"3.1s","32":"3.2s","33":"3.3s","34":"3.4s","35":"3.5s","36":"3.6s","37":"3.7s","38":"3.8s","39":"3.9s","40":"4.0s","41":"4.1s","42":"4.2s","43":"4.3s","44":"4.4s","45":"4.5s","46":"4.6s","47":"4.7s","48":"4.8s","49":"4.9s","50":"5.0s","51":"5.1s","52":"5.2s","53":"5.3s","54":"5.4s","55":"5.5s","56":"5.6s","57":"5.7s","58":"5.8s","59":"5.9s","60":"6.0s","61":"6.1s","62":"6.2s","63":"6.3s","64":"6.4s","65":"6.5s","66":"6.6s","67":"6.7s","68":"6.8s","69":"6.9s","70":"7.0s","71":"7.1s","72":"7.2s","73":"7.3s","74":"7.4s","75":"7.5s","76":"7.6s","77":"7.7s","78":"7.8s","79":"7.9s","80":"8.0s","81":"8.1s","82":"8.2s","83":"8.3s","84":"8.4s","85":"8.5s","86":"8.6s","87":"8.7s","88":"8.8s","89":"8.9s","90":"9.0s","91":"9.1s","92":"9.2s","93":"9.3s","94":"9.4s","95":"9.5s","96":"9.6s","97":"9.7s","98":"9.8s","99":"9.9s","100":"10.0s","101":"10.1s","102":"10.2s","103":"10.3s","104":"10.4s","105":"10.5s","106":"10.6s","107":"10.7s","108":"10.8s","109":"10.9s","110":"11.0s","111":"11.1s","112":"11.2s","113":"11.3s","114":"11.4s","115":"11.5s","116":"11.6s","117":"11.7s","118":"11.8s","119":"11.9s","120":"12.0s","121":"12.1s","122":"12.2s","123":"12.3s","124":"12.4s","125":"12.5s","126":"12.6s","127":"Sync with parameter1"], default: 127, size: 8, type: "enum", value: null ], parameter006 : [ number: 6, name: "Dimming Speed - Down (Local)", description: "This changes the speed that the light dims down when controlled at the switch. A setting of 'instant' turns the light immediately off. Increasing the value slows down the transition speed.
Default=Sync with parameter2", range: ["0":"instant","5":"500ms","6":"600ms","7":"700ms","8":"800ms","9":"900ms","10":"1.0s","11":"1.1s","12":"1.2s","13":"1.3s","14":"1.4s","15":"1.5s","16":"1.6s","17":"1.7s","18":"1.8s","19":"1.9s","20":"2.0s","21":"2.1s","22":"2.2s","23":"2.3s","24":"2.4s","25":"2.5s","26":"2.6s","27":"2.7s","28":"2.8s","29":"2.9s","30":"3.0s","31":"3.1s","32":"3.2s","33":"3.3s","34":"3.4s","35":"3.5s","36":"3.6s","37":"3.7s","38":"3.8s","39":"3.9s","40":"4.0s","41":"4.1s","42":"4.2s","43":"4.3s","44":"4.4s","45":"4.5s","46":"4.6s","47":"4.7s","48":"4.8s","49":"4.9s","50":"5.0s","51":"5.1s","52":"5.2s","53":"5.3s","54":"5.4s","55":"5.5s","56":"5.6s","57":"5.7s","58":"5.8s","59":"5.9s","60":"6.0s","61":"6.1s","62":"6.2s","63":"6.3s","64":"6.4s","65":"6.5s","66":"6.6s","67":"6.7s","68":"6.8s","69":"6.9s","70":"7.0s","71":"7.1s","72":"7.2s","73":"7.3s","74":"7.4s","75":"7.5s","76":"7.6s","77":"7.7s","78":"7.8s","79":"7.9s","80":"8.0s","81":"8.1s","82":"8.2s","83":"8.3s","84":"8.4s","85":"8.5s","86":"8.6s","87":"8.7s","88":"8.8s","89":"8.9s","90":"9.0s","91":"9.1s","92":"9.2s","93":"9.3s","94":"9.4s","95":"9.5s","96":"9.6s","97":"9.7s","98":"9.8s","99":"9.9s","100":"10.0s","101":"10.1s","102":"10.2s","103":"10.3s","104":"10.4s","105":"10.5s","106":"10.6s","107":"10.7s","108":"10.8s","109":"10.9s","110":"11.0s","111":"11.1s","112":"11.2s","113":"11.3s","114":"11.4s","115":"11.5s","116":"11.6s","117":"11.7s","118":"11.8s","119":"11.9s","120":"12.0s","121":"12.1s","122":"12.2s","123":"12.3s","124":"12.4s","125":"12.5s","126":"12.6s","127":"Sync with parameter2"], default: 127, size: 8, type: "enum", value: null ], parameter007 : [ number: 7, name: "Ramp Rate - On to Off (Remote)", description: "This changes the speed that the light turns off when controlled from the hub. A setting of 'instant' turns the light immediately off. Increasing the value slows down the transition speed.
Default=Sync with parameter3", range: ["0":"instant","5":"500ms","6":"600ms","7":"700ms","8":"800ms","9":"900ms","10":"1.0s","11":"1.1s","12":"1.2s","13":"1.3s","14":"1.4s","15":"1.5s","16":"1.6s","17":"1.7s","18":"1.8s","19":"1.9s","20":"2.0s","21":"2.1s","22":"2.2s","23":"2.3s","24":"2.4s","25":"2.5s","26":"2.6s","27":"2.7s","28":"2.8s","29":"2.9s","30":"3.0s","31":"3.1s","32":"3.2s","33":"3.3s","34":"3.4s","35":"3.5s","36":"3.6s","37":"3.7s","38":"3.8s","39":"3.9s","40":"4.0s","41":"4.1s","42":"4.2s","43":"4.3s","44":"4.4s","45":"4.5s","46":"4.6s","47":"4.7s","48":"4.8s","49":"4.9s","50":"5.0s","51":"5.1s","52":"5.2s","53":"5.3s","54":"5.4s","55":"5.5s","56":"5.6s","57":"5.7s","58":"5.8s","59":"5.9s","60":"6.0s","61":"6.1s","62":"6.2s","63":"6.3s","64":"6.4s","65":"6.5s","66":"6.6s","67":"6.7s","68":"6.8s","69":"6.9s","70":"7.0s","71":"7.1s","72":"7.2s","73":"7.3s","74":"7.4s","75":"7.5s","76":"7.6s","77":"7.7s","78":"7.8s","79":"7.9s","80":"8.0s","81":"8.1s","82":"8.2s","83":"8.3s","84":"8.4s","85":"8.5s","86":"8.6s","87":"8.7s","88":"8.8s","89":"8.9s","90":"9.0s","91":"9.1s","92":"9.2s","93":"9.3s","94":"9.4s","95":"9.5s","96":"9.6s","97":"9.7s","98":"9.8s","99":"9.9s","100":"10.0s","101":"10.1s","102":"10.2s","103":"10.3s","104":"10.4s","105":"10.5s","106":"10.6s","107":"10.7s","108":"10.8s","109":"10.9s","110":"11.0s","111":"11.1s","112":"11.2s","113":"11.3s","114":"11.4s","115":"11.5s","116":"11.6s","117":"11.7s","118":"11.8s","119":"11.9s","120":"12.0s","121":"12.1s","122":"12.2s","123":"12.3s","124":"12.4s","125":"12.5s","126":"12.6s","127":"Sync with parameter3"], default: 127, size: 8, type: "enum", value: null ], parameter008 : [ number: 8, name: "Ramp Rate - On to Off (Local)", description: "This changes the speed that the light turns off when controlled at the switch. A setting of 'instant' turns the light immediately off. Increasing the value slows down the transition speed.
Default=Sync with parameter4", range: ["0":"instant","5":"500ms","6":"600ms","7":"700ms","8":"800ms","9":"900ms","10":"1.0s","11":"1.1s","12":"1.2s","13":"1.3s","14":"1.4s","15":"1.5s","16":"1.6s","17":"1.7s","18":"1.8s","19":"1.9s","20":"2.0s","21":"2.1s","22":"2.2s","23":"2.3s","24":"2.4s","25":"2.5s","26":"2.6s","27":"2.7s","28":"2.8s","29":"2.9s","30":"3.0s","31":"3.1s","32":"3.2s","33":"3.3s","34":"3.4s","35":"3.5s","36":"3.6s","37":"3.7s","38":"3.8s","39":"3.9s","40":"4.0s","41":"4.1s","42":"4.2s","43":"4.3s","44":"4.4s","45":"4.5s","46":"4.6s","47":"4.7s","48":"4.8s","49":"4.9s","50":"5.0s","51":"5.1s","52":"5.2s","53":"5.3s","54":"5.4s","55":"5.5s","56":"5.6s","57":"5.7s","58":"5.8s","59":"5.9s","60":"6.0s","61":"6.1s","62":"6.2s","63":"6.3s","64":"6.4s","65":"6.5s","66":"6.6s","67":"6.7s","68":"6.8s","69":"6.9s","70":"7.0s","71":"7.1s","72":"7.2s","73":"7.3s","74":"7.4s","75":"7.5s","76":"7.6s","77":"7.7s","78":"7.8s","79":"7.9s","80":"8.0s","81":"8.1s","82":"8.2s","83":"8.3s","84":"8.4s","85":"8.5s","86":"8.6s","87":"8.7s","88":"8.8s","89":"8.9s","90":"9.0s","91":"9.1s","92":"9.2s","93":"9.3s","94":"9.4s","95":"9.5s","96":"9.6s","97":"9.7s","98":"9.8s","99":"9.9s","100":"10.0s","101":"10.1s","102":"10.2s","103":"10.3s","104":"10.4s","105":"10.5s","106":"10.6s","107":"10.7s","108":"10.8s","109":"10.9s","110":"11.0s","111":"11.1s","112":"11.2s","113":"11.3s","114":"11.4s","115":"11.5s","116":"11.6s","117":"11.7s","118":"11.8s","119":"11.9s","120":"12.0s","121":"12.1s","122":"12.2s","123":"12.3s","124":"12.4s","125":"12.5s","126":"12.6s","127":"Sync with parameter4"], default: 127, size: 8, type: "enum", value: null ], parameter009 : [ number: 9, name: "Minimum Level", description: "The minimum percent level that the light can be dimmed. Useful when the user has a light that does not turn on or flickers at a lower level.", range: "1..99", default: 1, size: 8, type: "number", value: null ], parameter010 : [ number: 10, name: "Maximum Level", description: "The maximum percent level that the light can be dimmed. Useful when the user wants to limit the maximum brighness.", range: "2..100", default: 100, size: 8, type: "number", value: null ], parameter011 : [ number: 11, name: "Invert Switch", description: "Inverts the orientation of the switch. Useful when the switch is installed upside down. Essentially up becomes down and down becomes up.", range: ["0":"No (default)", "1":"Yes"], default: 0, size: 1, type: "enum", value: null ], parameter012 : [ number: 12, name: "Auto Off Timer", description: "Automatically turns the switch off after this many seconds. When the switch is turned on a timer is started. When the timer expires the switch turns off.
0=Auto Off Disabled.", range: "0..32767", default: 0, size: 16, type: "number", value: null ], parameter013 : [ number: 13, name: "Default Level (Local)", description: "Default level for the dimmer when turned on at the switch.
1-100=Set Level
101=Use previous level.", range: "1..101", default: 101, size: 8, type: "number", value: null ], parameter014 : [ number: 14, name: "Default Level (Remote)", description: "Default level for the dimmer when turned on from the hub.
1-100=Set Level
101=Use previous level.", range: "1..101", default: 101, size: 8, type: "number", value: null ], parameter015 : [ number: 15, name: "Level After Power Restored", description: "Level the switch will return to when power is restored after power failure (if Switch is in On/Off Mode any level 1-100 will convert to 100).
0=Off
1-100=Set Level
101=Use previous level.", range: "0..101", default: 101, size: 8, type: "number", value: null ], parameter017 : [ number: 17, name: "Load Level Indicator Timeout", description: "Shows the level that the load is at for x number of seconds after the load is adjusted and then returns to the Default LED state.", range: ["0":"Do not display Load Level","1":"1 Second","2":"2 Seconds","3":"3 Seconds","4":"4 Seconds","5":"5 Seconds","6":"6 Seconds","7":"7 Seconds","8":"8 Seconds","9":"9 Seconds","10":"10 Seconds","11":"Display Load Level with no timeout (default)"], default: 11, size: 8, type: "enum", value: null ], parameter018 : [ number: 18, name: "Active Power Reports", description: "Percent power level change that will result in a new power report being sent.
0 = Disabled", range: "0..100", default: 10, size: 8, type: "number", value: null ], parameter019 : [ number: 19, name: "Periodic Power & Energy Reports", description: "Time period between consecutive power & energy reports being sent (in seconds). The timer is reset after each report is sent.", range: "0..32767", default: 3600, size: 16, type: "number", value: null ], parameter020 : [ number: 20, name: "Active Energy Reports", description: "Energy level change that will result in a new energy report being sent.
0 = Disabled
1-32767 = 0.01kWh-327.67kWh.", range: "0..32767", default: 10, size: 16, type: "number", value: null ], parameter021 : [ number: 21, name: "Power Source (read only)", description: "Neutral or Non-Neutral wiring is automatically sensed.", range: [0:"Non Neutral", 1:"Neutral"], default: 1, size: 1, type: "enum", value: null ], parameter022 : [ number: 22, name: "Aux Switch Type", description: "Set the Aux switch type.", range: ["0":"No Aux (default)", "1":"Dumb 3-Way Switch", "2":"Smart Aux Switch", "3":"No Aux Full Wave (On/Off only)"], default: 0, size: 8, type: "enum", value: null ], parameter023 : [ //implemented in firmware for the fan, emulated in this driver for 2-in-1 Dimmer number: 23, name: "Quick Start", description: "EXPERIMENTAL (hub commands only): Startup Level from OFF to ON (for LEDs that need higher level to turn on but can be dimmed lower) 0=Disabled", range: "0..100", default: 0, size: 8, type: "number", value: null ], parameter025 : [ number: 25, name: "Higher Output in non-Neutral", description: "Ability to increase level in non-neutral mode but may cause problems with high level ficker or aux switch detection. Adjust max level (P10) if you have problems with this enabled.", range: ["0":"Disabled (default)","1":"Enabled"], default:0, size: 1, type: "enum", value: null ], parameter050 : [ number: 50, name: "Button Press Delay", description: "Adjust the button delay used in scene control. 0=no delay (disables multi-tap scenes), Default=500ms", range: ["0":"0ms","3":"300ms","4":"400ms","5":"500ms (default)","6":"600ms","7":"700ms","8":"800ms","9":"900ms"], default: 5, size: 8, type: "enum", value: null ], parameter051 : [ number: 51, name: "Device Bind Number (read only)", description: "Number of devices currently bound and counts one group as two devices.", range: "0..255", default: 0, size: 8, type: "number", value: null ], parameter052 : [ number: 52, name: "Smart Bulb Mode", description: "For use with Smart Bulbs that need constant power and are controlled via commands rather than power.", range: ["0":"Disabled (default)", "1":"Smart Bulb Mode Enabled"], default: 0, size: 1, type: "enum", value: null ], parameter053 : [ number: 53, name: "Double-Tap UP to parameter 55", description: "Enable or Disable setting brightness to parameter 55 on double-tap UP.", range: ["0":"Disabled (default)", "1":"Enabled"], default: 0, size: 1, type: "enum", value: null ], parameter054 : [ number: 54, name: "Double-Tap DOWN to parameter 56", description: "Enable or Disable setting brightness to parameter 56 on double-tap DOWN.", range: ["0":"Disabled (default)", "1":"Enabled"], default: 0, size: 1, type: "enum", value: null ], parameter055 : [ number: 55, name: "Brightness level for Double-Tap UP", description: "Set this level on double-tap UP (if enabled by P53)", range: "0..100", default: 100, size: 8, type: "number", value: null ], parameter056 : [ number: 56, name: "Brightness level for Double-Tap DOWN", description: "Set this level on double-tap DOWN (if enabled by P54)", range: "0..100", default: 1, size: 8, type: "number", value: null ], parameter058 : [ number: 58, name: "Exclusion Behavior", description: "How device behaves during Exclusion", range: ["0":"LED Bar does not pulse", "1":"LED Bar pulses blue (default)", "2":"Device does not enter exclusion mode (requires factory reset to leave network or change this parameter)"], default: 1, size: 1, type: "enum", value: null ], parameter059 : [ number: 59, name: "Association Behavior", description: "Choose when the switch sends commands to associated devices", range: ["0":"Never", "1":"Local (default)", "2":"Z-Wave", "3":"Both"], default: 1, size: 1, type: "enum", value: null ], parameter064 : [ number: 64, name: "LED1 Notification", description: "4-byte encoded LED Notification", range: "0..4294967295", default: 0, size: 32, type: "number", value: null ], parameter069 : [ number: 69, name: "LED2 Notification", description: "4-byte encoded LED Notification", range: "0..4294967295", default: 0, size: 32, type: "number", value: null ], parameter074 : [ number: 74, name: "LED3 Notification", description: "4-byte encoded LED Notification", range: "0..4294967295", default: 0, size: 32, type: "number", value: null ], parameter079 : [ number: 79, name: "LED4 Notification", description: "4-byte encoded LED Notification", range: "0..4294967295", default: 0, size: 32, type: "number", value: null ], parameter084 : [ number: 84, name: "LED5 Notification", description: "4-byte encoded LED Notification", range: "0..4294967295", default: 0, size: 32, type: "number", value: null ], parameter089 : [ number: 89, name: "LED6 Notification", description: "4-byte encoded LED Notification", range: "0..4294967295", default: 0, size: 32, type: "number", value: null ], parameter094 : [ number: 94, name: "LED7 Notification", description: "4-byte encoded LED Notification", range: "0..4294967295", default: 0, size: 32, type: "number", value: null ], parameter095 : [ number: 95, name: "LED Bar Color (when On)", description: "Set the color of the LED Bar when the load is on.", range: ["0":"Red","7":"Orange","28":"Lemon","57":"Lime","85":"Green","106":"Teal","127":"Cyan","149":"Aqua","170":"Blue (default)","191":"Violet","212":"Magenta","234":"Pink","255":"White"], default: 170, size: 8, type: "enum", value: null ], parameter096 : [ number: 96, name: "LED Bar Color (when Off)", description: "Set the color of the LED Bar when the load is off.", range: ["0":"Red","7":"Orange","28":"Lemon","57":"Lime","85":"Green","106":"Teal","127":"Cyan","149":"Aqua","170":"Blue (default)","191":"Violet","212":"Magenta","234":"Pink","255":"White"], default: 170, size: 8, type: "enum", value: null ], parameter097 : [ number: 97, name: "LED Bar Intensity (when On)", description: "Set the intensity of the LED Bar when the load is on.", range: "0..100", default: 33, size: 8, type: "number", value: null ], parameter098 : [ number: 98, name: "LED Bar Intensity (when Off)", description: "Set the intensity of the LED Bar when the load is off.", range: "0..100", default: 3, size: 8, type: "number", value: null ], parameter099 : [ number: 99, name: "All LED Notification", description: "4-byte encoded LED Notification", range: "0..4294967295", default: 0, size: 32, type: "number", value: null ], parameter100 : [ number: 100, name: "LED Bar Scaling", description: "Method used for scaling. This allows you to match the scaling when two different generations are in the same gang box", range: ["0":"Gen3 method (VZM-style)","1":"Gen2 method (LZW-style)"], default: 0, size: 1, type: "enum", value: null ], parameter123 : [ number: 123, name: "Aux Switch Unique Scenes", description: "Have unique scene numbers for scenes activated with the aux switch", range: ["0":"Disabled (default)","1":"Enabled"], default: 0, size: 1, type: "enum", value: null ], parameter125 : [ number: 125, name: "Binding Off-to-On Sync Level", description: "Send Move_To_Level using Default Level with Off/On to bound devices", range: ["0":"Disabled (default)","1":"Enabled"], default: 0, size: 1, type: "enum", value: null ], parameter256 : [ number: 256, name: "Local Protection", description: "Ability to control switch from the wall.", range: ["0":"Local control enabled (default)", "1":"Local control disabled"], default: 0, size: 1, type: "enum", value: null ] , parameter257 : [ number: 257, name: "Remote Protection (read only) use Remote Control command to change.", description: "Ability to control switch from the hub.", range: ["0":"Remote control enabled (default)", "1":"Remote control disabled"], default: 0, size: 1, type: "enum", value: null ], parameter258 : [ number: 258, name: "Switch Mode", description: "Dimmer or On/Off only", range: ["0":"Dimmer", "1":"On/Off (default)"], default: 1, size: 1, type: "enum", value: null ], parameter259 : [ number: 259, name: "LED Bar in On/Off Switch Mode", description: "When the device is in On/Off mode, use full LED bar or just one LED", range: ["0":"Full bar (default)", "1":"One LED"], default: 0, size: 1, type: "enum", value: null ], parameter260 : [ number: 260, name: "Firmware Update-In-Progess Bar", description: "Display firmware update progress on LED Bar", range: ["1":"Enabled (default)", "0":"Disabled"], default: 1, size: 1, type: "enum", value: null ], parameter261 : [ number: 261, name: "Relay Click", description: "Audible Click in On/Off mode", range: ["0":"Enabled (default)", "1":"Disabled"], default: 0, size: 1, type: "enum", value: null ], parameter262 : [ number: 262, name: "Double-Tap config to clear notification", description: "Double-Tap the Config button to clear notifications", range: ["0":"Enabled (default)", "1":"Disabled"], default: 0, size: 1, type: "enum", value: null ], parameter263 : [ number: 263, name: "LED bar display levels", description: "Levels of the LED bar in Smart Bulb Mode
0=full range", range: "0..9", default: 3, size: 8, type: "number", value: null ] ] def infoLogsOff() { log.warn "${device.displayName}: "+fireBrick("Disabling Info logging after timeout") device.updateSetting("infoEnable",[value:"false",type:"bool"]) //device.updateSetting("disableInfoLogging",[value:"",type:"number"]) } def traceLogsOff() { log.warn "${device.displayName}: "+fireBrick("Disabling Trace logging after timeout") device.updateSetting("traceEnable",[value:"false",type:"bool"]) //device.updateSetting("disableTraceLogging",[value:"",type:"number"]) } def debugLogsOff() { log.warn "${device.displayName}: "+fireBrick("Disabling Debug logging after timeout") device.updateSetting("debugEnable",[value:"false",type:"bool"]) //device.updateSetting("disableDebugLogging",[value:"",type:"number"]) } def bind(cmds=[]) { if (infoEnable||traceEnable) log.info "${device.displayName}: bind(${cmds})" state.lastCommandSent = "bind(${cmds})" state.lastCommandTime = nowFormatted() return cmds } def bindSlave() { if (infoEnable) log.info "${device.displayName}: bindSlave()" state.lastCommandSent = "bindSlave()" state.lastCommandTime = nowFormatted() def cmds = zigbee.command(0x0003, 0x00, [:], defaultDelay, "20 00") if (traceEnable) log.trace "${device.displayName}: bindSlave $cmds" return cmds } def bindSource() { if (infoEnable) log.info "${device.displayName}: bindSource()" state.lastCommandSent = "bindSource()" state.lastCommandTime = nowFormatted() def cmds = zigbee.command(0xfc31,0x04,["mfgCode":"0x122F"],defaultDelay,"0") if (traceEnable) log.trace "${device.displayName}: bindSource $cmds" return cmds } def calculateParameter(number) { def value = Math.round((settings."parameter${number}"!=null?settings."parameter${number}":configParams["parameter${number.toString().padLeft(3,'0')}"].default).toFloat()).toInteger() switch (number){ case 21: //Read-Only (Neutral/non-Neutral) case 51: //Read-Only (Bindings) case 257: //Read-Only (Remote Protecion) value = configParams["parameter${number.toString().padLeft(3,'0')}"].default //Read-Only parameters calculate to their default values break case 9: //Min Level case 10: //Max Level case 13: //Default Level (local) case 14: //Default Level (remote) case 15: //Level after power restored case 55: //Double-Tap UP Level case 56: //Double-Tap DOWN Level value = convertPercentToByte(value) //convert levels from percent to byte values before sending to the device break case 18: //Active Power Reports (percent change) value = Math.min(Math.max(value.toInteger(),0),100) break case 95: //custom hue for LED Bar (when On) case 96: //custom hue for LED Bar (when Off) //360-hue values need to be converted to byte values before sending to the device if (settings."parameter${number}custom" =~ /^([0-9]{1}|[0-9]{2}|[0-9]{3})$/) { value = Math.round((settings."parameter${number}custom").toInteger()/360*255) } else { //else custom hue is invalid format or not selected if(settings."parameter${number}custom"!=null) { device.clearSetting("parameter${number}custom") if (infoEnable) log.warn "${device.displayName}: "+fireBrick("Cleared invalid custom hue: ${settings."parameter${number}custom"}") } } break case 97: //LED Bar Intensity(when On) case 98: //LED Bar Intensity(when Off) value = Math.min(Math.max(value.toInteger(),0),100) break } return value } def calculateSize(size=8) { //if (debugEnable) log.debug "${device.displayName}: calculateSize(${size})" if (size.toInteger() == 1) return 0x10 //1-bit boolean else if (size.toInteger() == 8) return 0x20 //1-byte unsigned integer else if (size.toInteger() == 16) return 0x21 //2-byte unsigned integer else if (size.toInteger() == 24) return 0x22 //3-byte unsigned integer else if (size.toInteger() == 32) return 0x23 //4-byte unsigned integer else if (size.toInteger() == 40) return 0x24 //5-byte unsigned integer else if (size.toInteger() == 48) return 0x25 //6-byte unsigned integer else if (size.toInteger() == 56) return 0x26 //7-byte unsigned integer else if (size.toInteger() == 64) return 0x27 //8-byte unsigned integer else return 0x20 //default to 1-byte unsigned if no other matches } def clusterLookup(cluster) { return zigbee?.clusterLookup(cluster)?:cluster==0x8021?"Binding Cluster": cluster==0x8022?"UNBinding Cluster": cluster==0x8032?"Routing Table Cluster": cluster==0xFC31?"Private Cluster": "Cluster:0x${zigbee.convertToHexString(cluster,4)}" } def configure(option) { //THIS GETS CALLED AUTOMATICALLY WHEN NEW DEVICE IS DISCOVERED OR WHEN CONFIGURE BUTTON SELECTED ON DEVICE PAGE option = (option==null||option==" ")?"":option if (infoEnable) log.info "${device.displayName}: configure($option)" state.lastCommandSent = "configure($option)" state.lastCommandTime = nowFormatted() state.driverDate = getDriverDate() state.model = device.getDataValue('model') if (infoEnable||traceEnable||debugEnable) log.info "${device.displayName}: Driver Date $state.driverDate" sendEvent(name: "numberOfButtons", value: 14) def cmds = [] // cmds += ["zdo bind ${device.deviceNetworkId} 0x01 0x01 0x0000 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //Basic Cluster // cmds += ["zdo bind ${device.deviceNetworkId} 0x01 0x01 0x0003 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //Identify Cluster // cmds += ["zdo bind ${device.deviceNetworkId} 0x02 0x01 0x0003 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //Identify Cluster ep2 // cmds += ["zdo bind ${device.deviceNetworkId} 0x01 0x01 0x0004 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //Group Cluster // cmds += ["zdo bind ${device.deviceNetworkId} 0x01 0x01 0x0005 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //Scenes Cluster cmds += ["zdo bind ${device.deviceNetworkId} 0x01 0x01 0x0006 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //On/Off Cluster // cmds += ["zdo bind ${device.deviceNetworkId} 0x02 0x01 0x0006 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //On/Off Cluster ep2 cmds += ["zdo bind ${device.deviceNetworkId} 0x01 0x01 0x0008 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //Level Control Cluster // cmds += ["zdo bind ${device.deviceNetworkId} 0x02 0x01 0x0008 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //Level Control Cluster ep2 cmds += ["zdo bind ${device.deviceNetworkId} 0x01 0x01 0x0019 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //OTA Upgrade Cluster if (state.model?.substring(0,5)!="VZM35") { //Fan does not support power/energy reports cmds += ["zdo bind ${device.deviceNetworkId} 0x01 0x01 0x0702 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //Simple Metering - to get energy reports cmds += ["zdo bind ${device.deviceNetworkId} 0x01 0x01 0x0B04 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //Electrical Measurement - to get power reports } // cmds += ["zdo bind ${device.deviceNetworkId} 0x01 0x01 0x8021 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //Binding Cluster - to get binding reports // cmds += ["zdo bind ${device.deviceNetworkId} 0x01 0x01 0x8022 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //UnBinding Cluster - to get Unbinding reports cmds += ["zdo bind ${device.deviceNetworkId} 0x01 0x01 0xFC31 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //Private Cluster cmds += ["zdo bind ${device.deviceNetworkId} 0x02 0x01 0xFC31 {${device.zigbeeId}} {}", "delay ${defaultDelay}"] //Private Cluster ep2 //read back some key attributes // cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0000 0x0000 {}", "delay ${defaultDelay}"] //get ZCL Version // cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0000 0x0001 {}", "delay ${defaultDelay}"] //get Application Version // cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0000 0x0004 {}", "delay ${defaultDelay}"] //get manufacturer cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0000 0x0005 {}", "delay ${defaultDelay}"] //get model cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0000 0x0006 {}", "delay ${defaultDelay}"] //get firmware date cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0000 0x0007 {}", "delay ${defaultDelay}"] //get power source // cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0000 0x0008 {}", "delay ${defaultDelay}"] //get Generic device class // cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0000 0x0009 {}", "delay ${defaultDelay}"] //get Generic device type // cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0000 0x000A {}", "delay ${defaultDelay}"] //get product code // cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0000 0x000B {}", "delay ${defaultDelay}"] //get product url cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0000 0x4000 {}", "delay ${defaultDelay}"] //get sw build id cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0006 0x0000 {}", "delay ${defaultDelay}"] //get on/off state cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0006 0x4003 {}", "delay ${defaultDelay}"] //get Startup OnOff state cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0008 0x0000 {}", "delay ${defaultDelay}"] //get current level if (state.model?.substring(0,5)!="VZM35") //Fan does not support on_off transition time cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0008 0x0010 {}", "delay ${defaultDelay}"] //get OnOff Transition Time cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0008 0x0011 {}", "delay ${defaultDelay}"] //get Default Remote On Level cmds += ["he rattr ${device.deviceNetworkId} 0x01 0x0008 0x4000 {}", "delay ${defaultDelay}"] //get Startup Level if (option!="All" && option!="Default") { //if we didn't pick option "All" or "Default" (so we don't read them twice) then preload the dimming/ramp rates and key parameters so they are not null in calculations for(int i = 1;i<=8;i++) if (state."parameter${i}value"==null) cmds += getParameter(i) cmds += getParameter(258) //switch mode cmds += getParameter(22) //aux switch type cmds += getParameter(52) //smart bulb mode cmds += getParameter(21) //power source (read-only) cmds += getParameter(51) //number of bindings (read-only) cmds += getParameter(257) //remote protection (read-only) } if (option!="") cmds += updated(option) //if option was selected on Configure button, pass it on to update settings. if (traceEnable) log.trace "${device.displayName}: configure $cmds" return cmds } def convertByteToPercent(int value=0) { //convert a 0-254 range where 254=100%. 255 is reserved for special meaning. //if (debugEnable) log.debug "${device.displayName}: convertByteToPercent(${value})" value = value==null?0:value //default to 0 if null value = Math.min(Math.max(value.toInteger(),0),255) //make sure input byte value is in the 0-255 range value = value>=255?256:value //this ensures that byte values of 255 get rounded up to 101% value = Math.ceil(value/255*100) //convert to 0-100 where 254=100% and 255 becomes 101 for special meaning return value } def convertPercentToByte(int value=0) { //convert a 0-100 range where 100%=254. 255 is reserved for special meaning. //if (debugEnable) log.debug "${device.displayName}: convertPercentToByte(${value})" value = value==null?0:value //default to 0 if null value = Math.min(Math.max(value.toInteger(),0),101) //make sure input percent value is in the 0-101 range value = Math.floor(value/100*255) //convert to 0-255 where 100%=254 and 101 becomes 255 for special meaning value = value==255?254:value //this ensures that 100% rounds down to byte value 254 value = value>255?255:value //this ensures that 101% rounds down to byte value 255 return value } def cycleSpeed() { // FOR FAN ONLY def cmds =[] if (parameter258=="1") cmds += toggle() //if we are in on/off mode then do a toggle instead of cycle else { def currentLevel = device.currentValue("level")==null?0:device.currentValue("level").toInteger() if (device.currentValue("switch")=="off") currentLevel = 0 boolean smartMode = device.currentValue("smartFan")=="Enabled" def newLevel = 0 def newSpeed ="" if (currentLevel<=0 ) {newLevel=20; newSpeed="low" } else if (currentLevel<=20) {newLevel=smartMode?40:60; newSpeed=smartMode?"medium-low":"medium"} else if (currentLevel<=40) {newLevel=60; newSpeed="medium"} else if (currentLevel<=60) {newLevel=smartMode?80:100; newSpeed=smartMode?"medium-high":"high"} else if (currentLevel<=80) {newLevel=100; newSpeed="high"} else {newLevel=0; newSpeed="off"} if (infoEnable) log.info "${device.displayName}: cycleSpeed(${device.currentValue("speed")}->${newSpeed})" state.lastCommandSent = "cycleSpeed(${device.currentValue("speed")}->${newSpeed})" state.lastCommandTime = nowFormatted() cmds += zigbee.setLevel(newLevel) if (traceEnable) log.trace "${device.displayName}: cycleSpeed $cmds" } return cmds } def identify(seconds) { if (infoEnable) log.info "${device.displayName}: identify(${seconds==null?"":seconds})" state.lastCommandSent = "identify(${seconds==null?"":seconds})" state.lastCommandTime = nowFormatted() setZigbeeAttribute(3,0,seconds,16) } def initialize() { //CALLED DURING HUB BOOTUP IF "INITIALIZE" CAPABILITY IS DECLARED IN METADATA SECTION //Typically used for things that need refreshing or re-connecting at bootup (e.g. LAN integrations but not zigbee bindings) state.clear() if (infoEnable) log.info "${device.displayName}: initialize()" state.lastCommandSent = "initialize()" state.lastCommandTime = nowFormatted() state.driverDate = getDriverDate() state.model = device.getDataValue('model') device.clearSetting("parameter23level") device.clearSetting("parameter95custom") device.clearSetting("parameter96custom") ledEffectOne(1234567,255,0,0,0) //clear any outstanding oneLED Effects ledEffectAll(255,0,0,0) //clear any outstanding allLED Effects //if (infoEnable||traceEnable||debugEnable) log.info "${device.displayName}: Driver Date $state.driverDate" //this is also done in refresh() def cmds = [] cmds += refresh() if (traceEnable) log.trace "${device.displayName}: initialize $cmds" return cmds } def installed() { //THIS IS CALLED WHEN A DEVICE IS INSTALLED log.info "${device.displayName}: installed()" state.lastCommandSent = "installed()" state.lastCommandTime = nowFormatted() state.driverDate = getDriverDate() state.model = device.getDataValue('model') //configure() //I confirmed configure() gets called at Install time so this isn't needed here return } def intTo8bitUnsignedHex(value) { return zigbee.convertToHexString(value?.toInteger(),2) } def intTo16bitUnsignedHex(value) { def hexStr = zigbee.convertToHexString(value.toInteger(),4) return new String(hexStr.substring(2, 4) + hexStr.substring(0, 2)) } def intTo32bitUnsignedHex(value) { return hexStr = zigbee.convertToHexString(value.toInteger(),8) } def ledEffectAll(effect=255, color=0, level=100, duration=60) { effect = effect.toString().split(/=/)[0] def effectName = "unknown($effect)" switch (effect){ case "255": effectName = "Stop"; break case "1": effectName = "Solid"; break case "2": effectName = "Fast Blink"; break case "3": effectName = "Slow Blink"; break case "4": effectName = "Pulse"; break case "5": effectName = "Chase"; break case "6": effectName = "Open/Close"; break case "7": effectName = "Small-to-Big"; break case "8": effectName = "Aurora"; break case "9": effectName = "Slow Falling"; break case "10": effectName = "Medium Falling"; break case "11": effectName = "Fast Falling"; break case "12": effectName = "Slow Rising"; break case "13": effectName = "Medium Rising"; break case "14": effectName = "Fast Rising"; break case "15": effectName = "Medium Blink"; break case "16": effectName = "Slow Chase"; break case "17": effectName = "Fast Chase"; break case "18": effectName = "Fast Siren"; break case "19": effectName = "Slow Siren"; break case "0": effectName = "LEDs Off"; break } sendEvent(name:"ledEffect", value: "$effectName All") if (infoEnable) log.info "${device.displayName}: ledEffectAll(${effect},${color},${level},${duration})" state.lastCommandSent = "ledEffectAll(${effect},${color},${level},${duration})" state.lastCommandTime = nowFormatted() effect = Math.min(Math.max((effect!=null?effect:255).toInteger(),0),255) color = Math.min(Math.max((color!=null?color:0).toInteger(),0),255) level = Math.min(Math.max((level!=null?level:100).toInteger(),0),100) duration = Math.min(Math.max((duration!=null?duration:60).toInteger(),0),255) def cmds =[] Integer cmdEffect = effect.toInteger() Integer cmdColor = color.toInteger() Integer cmdLevel = level.toInteger() Integer cmdDuration = duration.toInteger() cmds += zigbee.command(0xfc31,0x01,["mfgCode":"0x122F"],defaultDelay,"${intTo8bitUnsignedHex(cmdEffect)} ${intTo8bitUnsignedHex(cmdColor)} ${intTo8bitUnsignedHex(cmdLevel)} ${intTo8bitUnsignedHex(cmdDuration)}") if (traceEnable) log.trace "${device.displayName}: ledEffectAll $cmds" return cmds } def ledEffectOne(lednum, effect=255, color=0, level=100, duration=60) { lednum = lednum.toString().split(/ /)[0].replace(",","") effect = effect.toString().split(/=/)[0] def effectName = "unknown($effect)" switch (effect){ case "255": effectName = "Stop"; break case "1": effectName = "Solid"; break case "2": effectName = "Fast Blink"; break case "3": effectName = "Slow Blink"; break case "4": effectName = "Pulse"; break case "5": effectName = "Chase"; break case "6": effectName = "Falling"; break case "7": effectName = "Rising"; break case "8": effectName = "Aurora"; break case "0": effectName = "LEDs Off"; break } sendEvent(name:"ledEffect", value: "$effectName LED${lednum.toString().split(/ /)[0]}") if (infoEnable) log.info "${device.displayName}: ledEffectOne(${lednum},${effect},${color},${level},${duration})" state.lastCommandSent = "ledEffectOne(${lednum},${effect},${color},${level},${duration})" state.lastCommandTime = nowFormatted() effect = Math.min(Math.max((effect!=null?effect:255).toInteger(),0),255) color = Math.min(Math.max((color!=null?color:0).toInteger(),0),255) level = Math.min(Math.max((level!=null?level:100).toInteger(),0),100) duration = Math.min(Math.max((duration!=null?duration:60).toInteger(),0),255) def cmds = [] lednum.toString().each { it= Math.min(Math.max((it!=null?it:1).toInteger(),1),7) Integer cmdLedNum = (it.toInteger()-1) //lednum is 0-based in firmware Integer cmdEffect = effect.toInteger() Integer cmdColor = color.toInteger() Integer cmdLevel = level.toInteger() Integer cmdDuration = duration.toInteger() cmds += zigbee.command(0xfc31,0x03,["mfgCode":"0x122F"],150,"${intTo8bitUnsignedHex(cmdLedNum)} ${intTo8bitUnsignedHex(cmdEffect)} ${intTo8bitUnsignedHex(cmdColor)} ${intTo8bitUnsignedHex(cmdLevel)} ${intTo8bitUnsignedHex(cmdDuration)}") } if (traceEnable) log.trace "${device.displayName}: ledEffectOne $cmds" return cmds } def nowFormatted() { if(location.timeZone) return new Date().format("yyyy-MMM-dd h:mm:ss a", location.timeZone) else return new Date().format("yyyy MMM dd EEE h:mm:ss a") } def off() { if (infoEnable) log.info "${device.displayName}: off()" state.lastCommandSent = "off()" state.lastCommandTime = nowFormatted() def cmds = [] cmds += zigbee.off(defaultDelay) if (traceEnable) log.trace "${device.displayName}: off $cmds" return cmds } def on() { if (infoEnable) log.info "${device.displayName}: on()" state.lastCommandSent = "on()" state.lastCommandTime = nowFormatted() def cmds = [] if (settings.parameter23?.toInteger()>0) cmds += quickStart() //do quickStart if enabled else cmds += zigbee.on(defaultDelay) //ELSE just turn On if (traceEnable) log.trace "${device.displayName}: on $cmds" return cmds } def parse(String description) { if (debugEnable) log.debug "${device.displayName}: parse($description)" Map descMap = zigbee.parseDescriptionAsMap(description) if (debugEnable) log.debug "${device.displayName}: $descMap" try { if (debugEnable && (zigbee.getEvent(description)!=[:])) log.debug "${device.displayName}: zigbee.getEvent ${zigbee.getEvent(description)}" } catch (e) { if (debugEnable) log.debug "${device.displayName}: "+magenta(bold("There was an error while calling zigbee.getEvent: $description")) } def attrHex = descMap.attrInt==null?null:"0x${zigbee.convertToHexString(descMap.attrInt,4)}" def attrInt = descMap.attrInt==null?null:descMap.attrInt.toInteger() def clusterHex = descMap.clusterId==null?"0x${descMap.cluster}":"0x${descMap.clusterId}" def clusterInt = descMap.clusterInt==null?null:descMap.clusterInt.toInteger() def clusterName= clusterLookup(clusterInt) def valueStr = descMap.value ?: "unknown" switch (clusterInt){ case 0x0000: //BASIC CLUSTER if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + //(zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" switch (attrInt) { case 0x0000: if (infoEnable) log.info "${device.displayName}: Report received ZCLVersion: $valueStr" state.zclVersion = valueStr break case 0x0001: if (infoEnable) log.info "${device.displayName}: Report received ApplicationVersion: $valueStr" state.applicationVersion = valueStr break case 0x0004: if (infoEnable) log.info "${device.displayName}: Report received Mfg: $valueStr" state.manufacturer = valueStr break case 0x0005: if (infoEnable) log.info "${device.displayName}: Report received Model: $valueStr" state.model = valueStr break case 0x0006: if (infoEnable) log.info "${device.displayName}: Report received FW Date: $valueStr" state.fwDate = valueStr break case 0x0007: def valueInt = Integer.parseInt(descMap['value'],16) valueStr = valueInt==0?"Non-Neutral":"Neutral" if (infoEnable) log.info "${device.displayName}: " + green("Report received Power Source: $valueInt ($valueStr)") state.powerSource = valueStr state.parameter21value = valueInt device.updateSetting("parameter21",[value:"${valueInt}",type:"enum"]) break case 0x4000: if (infoEnable) log.info "${device.displayName}: Report received FW Version: $valueStr" state.fwVersion = valueStr break default: if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN ATTRIBUTE:$attrInt DESCMAP:$descMap") break } break case 0x0003: //IDENTIFY CLUSTER if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" switch (attrInt) { case 0x0000: if (infoEnable) log.info "${device.displayName}: Report received IdentifyTime: ${zigbee.convertHexToInt(valueStr)}" break default: if ((infoEnable && attrInt!=null)||traceEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN ATTRIBUTE:$attrInt DESCMAP:$descMap") break } break case 0x0004: //GROUP CLUSTER if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" switch (attrInt) { case 0x0000: if (infoEnable) log.info "${device.displayName}: Report received Group Name Support: $valueStr" break default: if (attrInt==null && descMap.messageType=="00" && descMap.direction=="01") { def groupNum = zigbee.convertHexToInt(descMap.data[2]+descMap.data[1]) switch (descMap.command) { case "00": if (infoEnable) log.info "${device.displayName}: Add Group $groupNum" break case "01": if (infoEnable) log.info "${device.displayName}: View Group $groupNum" break case "02": if (infoEnable) log.info "${device.displayName}: Get Group $groupNum" break case "03": if (infoEnable) log.info "${device.displayName}: Remove Group $groupNum" break case "04": if (infoEnable) log.info "${device.displayName}: Remove all groups" break case "05": if (infoEnable) log.info "${device.displayName}: Add group if Identifying" break default: if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND:$descMap.command DESCMAP:$descMap") break } } else { if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN ATTRIBUTE:$attrInt DESCMAP:$descMap") } break } break case 0x0005: //SCENES CLUSTER if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" switch (attrInt) { case 0x0000: if (infoEnable) log.info "${device.displayName}: Report received Scene Count: $valueStr" break case 0x0001: if (infoEnable) log.info "${device.displayName}: Report received Current Scene: $valueStr" break case 0x0002: if (infoEnable) log.info "${device.displayName}: Report received Current Group: $valueStr" break case 0x0003: if (infoEnable) log.info "${device.displayName}: Report received Scene Valid: $valueStr" break case 0x0004: if (infoEnable) log.info "${device.displayName}: Report received Scene Name Support: $valueStr" break default: if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN ATTRIBUTE:$attrInt DESCMAP:$descMap") break } break case 0x0006: //ON_OFF CLUSTER if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" switch (attrInt) { case 0x0000: if (descMap.command == "01" || descMap.command == "0A" || descMap.command == "0B"){ def valueInt = Integer.parseInt(descMap['value'],16) valueStr = valueInt == 0? "off": "on" if (infoEnable) log.info "${device.displayName}: Report received Switch: $valueInt ($valueStr)" sendEvent(name:"switch", value: valueStr) def currentLevel = device.currentValue("level")==null?0:device.currentValue("level").toInteger() if (state.model?.substring(0,5)=="VZM35") { //FOR FAN ONLY if (device.currentValue("smartFan")=="Enabled") { if (currentLevel<=20) newSpeed="low" else if (currentLevel<=40) newSpeed="medium-low" else if (currentLevel<=60) newSpeed="medium" else if (currentLevel<=80) newSpeed="medium-high" else if (currentLevel<=100) newSpeed="high" } else { if (currentLevel<=33) newSpeed = "low" else if (currentLevel<=66) newSpeed = "medium" else if (currentLevel<=100) newSpeed = "high" } if (valueStr=="off") newSpeed = "off" else if (parameter258=="1") newSpeed = "high" if (infoEnable) log.info "${device.displayName}: Report received Speed: ${newSpeed}" sendEvent(name:"speed", value: "${newSpeed}") } } else if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND(${descMap.command})") break case 0x4003: if (descMap.command == "01" || descMap.command == "0A" || descMap.command == "0B"){ def valueInt = Integer.parseInt(descMap['value'],16) valueStr = (valueInt==0?"Off":(valueInt==255?"Previous":"On")) if (infoEnable) log.info "${device.displayName}: Report received Power-On State: $valueInt ($valueStr)" state.powerOnState = valueStr } else if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND(${descMap.command})") break default: if (descMap.profileId=="0000" && descMap.command=="00" && descMap.direction=="00") { if (traceEnable||debugEnable) log.info "${device.displayName}: " + fireBrick("MATCH DESCRIPTOR REQUEST Device:${descMap.data[2]}${descMap.data[1]} Profile:${descMap.data[4]}${descMap.data[3]} Cluster:${descMap.data[7]}${descMap.data[6]}") } else if (attrInt==null && descMap.command=="0B" && descMap.direction=="01") { if (parameter51.toInteger()>0) { //not sure why the firmware sends these when there are no bindings if (descMap.data[0]=="00" && infoEnable) log.info "${device.displayName}: Bind Command Sent: Switch OFF" if (descMap.data[0]=="01" && infoEnable) log.info "${device.displayName}: Bind Command Sent: Switch ON" if (descMap.data[0]=="02" && infoEnable) log.info "${device.displayName}: Bind Command Sent: Toggle" } } else if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN ATTRIBUTE:$attrInt DESCMAP:$descMap") break } break case 0x0008: //LEVEL CONTROL CLUSTER if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" switch (attrInt) { case 0x0000: if (descMap.command == "01" || descMap.command == "0A" || descMap.command == "0B"){ def valueInt = Integer.parseInt(descMap['value'],16) valueInt=Math.min(Math.max(valueInt.toInteger(),0),254) def percentValue = convertByteToPercent(valueInt) valueStr = percentValue.toString()+"%" if (infoEnable) log.info "${device.displayName}: Report received Level: $valueInt ($valueStr)" sendEvent(name:"level", value: percentValue, unit: "%") def newSpeed ="" if (state.model?.substring(0,5)=="VZM35") { //FOR FAN ONLY if (device.currentValue("smartFan")=="Enabled") { if (percentValue<=20) newSpeed="low" else if (percentValue<=40) newSpeed="medium-low" else if (percentValue<=60) newSpeed="medium" else if (percentValue<=80) newSpeed="medium-high" else if (percentValue<=100) newSpeed="high" } else { if (percentValue<=33) newSpeed = "low" else if (percentValue<=66) newSpeed = "medium" else if (percentValue<=100) newSpeed = "high" } if (device.currentValue("switch")=="off") newSpeed="off" else if (parameter258=="1") newSpeed="high" if (infoEnable) log.info "${device.displayName}: Report received Speed: ${newSpeed}" sendEvent(name:"speed", value: "${newSpeed}") } } else if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND(${descMap.command})") break case 0x0001: if(descMap.command == "01" || descMap.command == "0A" || descMap.command == "0B"){ def valueInt = Integer.parseInt(descMap['value'],16) if (infoEnable) log.info "${device.displayName}: Report received Remaining Time: ${valueInt/10}s" } else if (infoEnable || debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND(${descMap.command})") break case 0x000F: if(descMap.command == "01" || descMap.command == "0A" || descMap.command == "0B"){ def valueInt = Integer.parseInt(descMap['value'],8) if (infoEnable) log.info "${device.displayName}: Report received Level Control Options: 0x${zigbee.convertToHexString(valueInt,2)}" } else if (infoEnable || debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND(${descMap.command})") break case 0x0010: if(descMap.command == "01" || descMap.command == "0A" || descMap.command == "0B"){ def valueInt = Integer.parseInt(descMap['value'],16) if (infoEnable) log.info "${device.displayName}: Report received On/Off Transition: ${valueInt/10}s" state.parameter3value = valueInt device.updateSetting("parameter3",[value:"${valueInt}",type:configParams["parameter003"].type?.toString()]) } else if (infoEnable || debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND(${descMap.command})") break case 0x0011: if (descMap.command == "01" || descMap.command == "0A" || descMap.command == "0B"){ def valueInt = Integer.parseInt(descMap['value'],16) valueStr = (valueInt==255?"Previous":convertByteToPercent(valueInt).toString()+"%") if (infoEnable) log.info "${device.displayName}: Report received Remote-On Level: $valueInt ($valueStr)" } else if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND(${descMap.command})") break case 0x4000: if (descMap.command == "01" || descMap.command == "0A" || descMap.command == "0B"){ def valueInt = Integer.parseInt(descMap['value'],16) valueStr = (valueInt==255?"Previous":convertByteToPercent(valueInt).toString()+"%") if (infoEnable) log.info "${device.displayName}: Report received Power-On Level: $valueInt ($valueStr)" state.parameter15value = convertByteToPercent(valueInt) device.updateSetting("parameter15",[value:"${convertByteToPercent(valueInt)}",type:configParams["parameter015"].type?.toString()]) } else if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND(${descMap.command})") break default: if (attrInt==null && descMap.command=="0B" && descMap.direction=="01") { if (parameter51.toInteger()>0) { //not sure why the firmware sends these when there are no bindings if (descMap.data[0]=="00" && infoEnable) log.info "${device.displayName}: Bind Command Sent: Move To Level ${descMap.data}" if (descMap.data[0]=="01" && infoEnable) log.info "${device.displayName}: Bind Command Sent: Move Up/Down ${descMap.data}" if (descMap.data[0]=="02" && infoEnable) log.info "${device.displayName}: Bind Command Sent: Step ${descMap.data}" if (descMap.data[0]=="03" && infoEnable) log.info "${device.displayName}: Bind Command Sent: Stop Level Change ${descMap.data}" if (descMap.data[0]=="04" && infoEnable) log.info "${device.displayName}: Bind Command Sent: Move To Level (with On/Off) ${descMap.data}" if (descMap.data[0]=="05" && infoEnable) log.info "${device.displayName}: Bind Command Sent: Move Up/Down (with On/Off) ${descMap.data}" if (descMap.data[0]=="06" && infoEnable) log.info "${device.displayName}: Bind Command Sent: Step (with On/Off) ${descMap.data}" } } else if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN ATTRIBUTE:$attrInt DESCMAP:$descMap") break } break case 0x0013: //ALEXA CLUSTER if (infoEnable||debugEnable) log.info "${device.displayName}: "+darkOrange("Alexa Heartbeat") if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" break case 0x0019: //OTA CLUSTER if (infoEnable||debugEnable) log.info "${device.displayName}: "+darkOrange("OTA CLUSTER") if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" switch (attrInt) { case 0x0000: if (infoEnable) log.info "${device.displayName}: Report received Server ID: $valueStr" break case 0x0001: if (infoEnable) log.info "${device.displayName}: Report received File Offset: $valueStr" break case 0x0006: if (infoEnable) log.info "${device.displayName}: Report received Upgrade Status: $valueStr" break default: if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN ATTRIBUTE:$attrInt DESCMAP:$descMap") break } break case 0x0702: //SIMPLE METERING CLUSTER if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" switch (attrInt) { case 0x0000: if (descMap.command == "01" || descMap.command == "0A" || descMap.command == "0B"){ def valueInt = Integer.parseInt(descMap['value'],16) float energy energy = valueInt/100 if (infoEnable) log.info "${device.displayName}: Report received Energy: ${energy}kWh" sendEvent(name:"energy",value:energy ,unit: "kWh") } else if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND(${descMap.command})") break default: if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN ATTRIBUTE:$attrInt DESCMAP:$descMap") break } break case 0x0B04: //ELECTRICAL MEASUREMENT CLUSTER if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" switch (attrInt) { case 0x0501: if (descMap.command == "01" || descMap.command == "0A" || descMap.command == "0B"){ def valueInt = Integer.parseInt(descMap['value'],16) float amps amps = valueInt/100 if (infoEnable) log.info "${device.displayName}: Report received Amps: ${amps}A" sendEvent(name:"amps",value:amps ,unit: "A") } else if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND(${descMap.command})") break case 0x050b: if (descMap.command == "01" || descMap.command == "0A" || descMap.command == "0B"){ def valueInt = Integer.parseInt(descMap['value'],16) float power power = valueInt/10 if (infoEnable) log.info "${device.displayName}: Report received Power: ${power}W" sendEvent(name: "power", value: power, unit: "W") } else if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND(${descMap.command})") break default: if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN ATTRIBUTE:$attrInt DESCMAP:$descMap") break } break case 0x8021: //BINDING CLUSTER if (infoEnable||debugEnable) log.info "${device.displayName}: "+darkOrange("Binding Cluster") if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" break case 0x8022: //UNBINDING CLUSTER if (infoEnable||debugEnable) log.info "${device.displayName}: "+darkOrange("UNBinding Cluster") if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" break case 0x8032: //ROUTING TABLE CLUSTER if (infoEnable||debugEnable) log.info "${device.displayName}: "+darkOrange("Routing Table Cluster") if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + "clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" break case 0xfc31: //PRIVATE CLUSTER if (traceEnable) log.trace "${device.displayName}: ${clusterName} (" + //"clusterId:${descMap.cluster?:descMap.clusterId}" + (descMap.attrId==null?"":" attrId:${descMap.attrId}") + (descMap.value==null?"":" value:${descMap.value}") + (zigbee.getEvent(description)==[:]?(descMap.data==null?"":" data:${descMap.data}"):(" ${zigbee.getEvent(description)}")) + ")" if (attrInt == null) { if (descMap.isClusterSpecific) { if (descMap.command == "00") ZigbeePrivateCommandEvent(descMap.data) //Button Events if (descMap.command == "04") BindSource() //Start Binding if (descMap.command == "24") ZigbeePrivateLEDeffectStopEvent(descMap.data) //LED start/stop events } } else if (descMap.command == "01" || descMap.command == "0A" || descMap.command == "0B"){ def valueInt = Integer.parseInt(descMap['value'],16) def valueHex = intTo32bitUnsignedHex(valueInt) def infoDev = "${device.displayName}: " def infoTxt = "Receive parameter ${attrInt} value ${valueInt}" def infoMsg = infoDev + infoTxt switch (attrInt){ case 1: infoMsg += " (Remote Dim Rate Up: " + (valueInt<127?((valueInt/10).toString()+"s)"):"default)") break case 2: infoMsg += " (Local Dim Rate Up: " + (valueInt<127?((valueInt/10).toString()+"s)"):"sync with 1)") break case 3: infoMsg += " (Remote Ramp Rate On: " + (valueInt<127?((valueInt/10).toString()+"s)"):"sync with 1)") break case 4: infoMsg += " (Local Ramp Rate On: " + (valueInt<127?((valueInt/10).toString()+"s)"):"sync with 3)") break case 5: infoMsg += " (Remote Dim Rate Down: " + (valueInt<127?((valueInt/10).toString()+"s)"):"sync with 1)") break case 6: infoMsg += " (Local Dim Rate Down: " + (valueInt<127?((valueInt/10).toString()+"s)"):"sync with 2)") break case 7: infoMsg += " (Remote Ramp Rate Off: " + (valueInt<127?((valueInt/10).toString()+"s)"):"sync with 3)") break case 8: infoMsg += " (Local Ramp Rate Off: " + (valueInt<127?((valueInt/10).toString()+"s)"):"sync with 4)") break case 9: //Min Level infoMsg += " (min level ${convertByteToPercent(valueInt)}%)" break case 10: //Max Level infoMsg += " (max level ${convertByteToPercent(valueInt)}%)" break case 11: //Invert Switch infoMsg += valueInt==0?" (not Inverted)":" (Inverted)" break case 12: //Auto Off Timer infoMsg += " (Auto Off Timer " + (valueInt==0?red("disabled"):"${valueInt}s") + ")" break case 13: //Default Level (local) infoMsg += " (default local level " + (valueInt==255?" = previous)":" ${convertByteToPercent(valueInt)}%)") sendEvent(name:"levelPreset", value:convertByteToPercent(valueInt)) break case 14: //Default Level (remote) infoMsg += " (default remote level " + (valueInt==255?" = previous)":"${convertByteToPercent(valueInt)}%)") break case 15: //Level After Power Restored infoMsg += " (power-on level " + (valueInt==255?" = previous)":"${convertByteToPercent(valueInt)}%)") break case 17: //Load Level Timeout infoMsg += (valueInt==0?" (do not display load level)":(valueInt==11?" (always display load level)":"s load level timeout")) break case 18: infoMsg += " (Active Power Report" + (valueInt==0?red(" disabled"):" ${valueInt}% change") + ")" break case 19: infoMsg += "s (Periodic Power/Energy " + (valueInt==0?red(" disabled"):"") + ")" break case 20: infoMsg += " (Active Energy Report " + (valueInt==0?red(" disabled"):" ${valueInt/100}kWh change") + ")" break case 21: //Power Source infoMsg = infoDev + green(infoTxt + (valueInt==0?" (Non-Neutral)":" (Neutral)")) state.powerSource = valueInt==0?"Non-Neutral":"Neutral" break case 22: //Aux Type switch (state.model?.substring(0,5)){ case "VZM31": //Blue 2-in-1 Dimmer case "VZW31": //Red 2-in-1 Dimmer infoMsg = infoDev + crimson(infoTxt + " " + (valueInt==0?"(No Aux)":(valueInt==1?"(Dumb 3-way)":(valueInt==2?"(Smart Aux)":(valueInt==3?"(No Aux Full Wave)":"(unknown type)"))))) state.auxType = (valueInt==0? "No Aux": (valueInt==1? "Dumb 3-way": (valueInt==2? "Smart Aux": (valueInt==3? "No Aux Full Wave": "unknown type $valueInt")))) break case "VZM35": //Fan Switch infoMsg = infoDev + crimson(infoTxt + " " + (valueInt==0?"(No Aux)":"(Smart Aux)")) state.auxType = valueInt==0? "No Aux": "Smart Aux" break default: infoMsg = infoDev + crimson(infoTxt + " unknown model $state.model") state.auxType = "unknown model ${state.model}" break } break case 23: //Quick Start (in firmware on Fan, emulated in this driver for dimmer) if (state.model?.substring(0,5)!="VZM35") infoMsg += " (Quick Start " + (valueInt==0?red("disabled"):"${valueInt}%") + ")" else infoMsg += " (Quick Start " + (valueInt==0?red("disabled"):"${valueInt} seconds") + ")" break case 25: //Higher Output in non-Neutral infoMsg += " (non-Neutral High Output " + (valueInt==0?red("disabled"):green("enabled")) + ")" break case 50: //Button Press Delay infoMsg += " (${valueInt*100}ms Button Delay)" break case 51: //Device Bind Number infoMsg = infoDev + green(infoTxt + " (Bindings)") sendEvent(name:"numberOfBindings", value:valueInt) break case 52: //Smart Bulb/Fan Mode if (state.model?.substring(0,5)=="VZM35") { //FOR FAN ONLY infoMsg = infoDev + crimson(infoTxt) + (valueInt==0?red(" (SFM disabled)"):green(" (SFM enabled)")) sendEvent(name:"smartFan", value:valueInt==0?"Disabled":"Enabled") } else { infoMsg = infoDev + crimson(infoTxt) + (valueInt==0?red(" (SBM disabled)"):green(" (SBM enabled)")) sendEvent(name:"smartBulb", value:valueInt==0?"Disabled":"Enabled") } break case 53: //Double-Tap UP infoMsg += " (Double-Tap Up " + (valueInt==0?red("disabled"):green("enabled")) + ")" break case 54: //Double-Tap DOWN infoMsg += " (Double-Tap Down " + (valueInt==0?red("disabled"):green("enabled")) + ")" break case 55: //Double-Tap UP level infoMsg += " (Double-Tap Up level ${convertByteToPercent(valueInt)}%)" break case 56: //Double-Tap DOWN level infoMsg += " (Double-Tap Down level ${convertByteToPercent(valueInt)}%)" break case 58: //Exclusion Behavior infoMsg += " (Exclusion: " + (valueInt==0?"LED Bar does not pulse":valueInt==1?"LED Bar pulses blue":valueInt==2?"do not Exclude":"unknown") + ")" break case 59: //Association Behavior infoMsg += " (Association: " + (valueInt==0?"None":valueInt==1?"Local":valueInt==2?"Hub":"Local+Hub") + ")" break case 64: //LED1 Notification case 69: //LED2 Notification case 74: //LED3 Notification case 79: //LED4 Notification case 84: //LED5 Notification case 89: //LED6 Notification case 94: //LED7 Notification case 99: //All LED Notification def effectHex = valueHex.substring(0,2) int effectInt = Integer.parseInt(effectHex,16) infoMsg += " [0x${valueHex}] " + (effectInt==255?"(Stop Effect)":"(Start Effect ${effectInt})") break case 95: //LED bar color when on case 96: //LED bar color when off infoMsg += hue(valueInt," (LED bar color when " + (attrInt==95?"On:":"Off:") + " ${Math.round(valueInt/255*360)}°)") break case 97: //LED bar intensity when on case 98: //LED bar intensity when off infoMsg += "% (LED bar intensity when " + (attrInt==97?"On)":"Off)") break case 100: //LED Bar Scaling infoMsg += " (LED Scaling " + (valueInt==0?blue("VZM-style"):red("LZW-style")) + ")" break case 123: //Aux Switch Scenes infoMsg += " (Aux Scenes " + (valueInt==0?red("disabled"):green("enabled")) + ")" break case 125: //Binding Off-to-On Sync Level infoMsg += " (Send Level with Binding Off/On " + (valueInt==0?red("disabled"):green("enabled")) + ")" break case 156: //Local Protection case 256: infoMsg += " (Local Control " + (valueInt==0?green("enabled"):red("disabled")) + ")" break case 157: //Remote Protection case 257: infoMsg = infoDev + green(infoTxt) + green(" (Remote Control ") + (valueInt==0?green("enabled"):red("disabled")) + ")" break case 158: //Switch Mode case 258: switch (state.model?.substring(0,5)){ case "VZM31": //Blue 2-in-1 Dimmer case "VZW31": //Red 2-in-1 Dimmer infoMsg = infoDev + crimson(infoTxt + " " + (valueInt==0?"(Dimmer mode)":"(On/Off mode)")) sendEvent(name:"switchMode", value:valueInt==0?"Dimmer":"On/Off") break case "VZM35": //Fan Switch infoMsg = infoDev + crimson(infoTxt + " " + (valueInt==0?"(Multi-Speed mode)":"(On/Off mode)")) sendEvent(name:"switchMode", value:valueInt==0?"Multi-Speed":"On/Off") break default: infoMsg = infoDev + crimson(infoTxt + " unknown model $state.model") sendEvent(name:"switchMode", value:"unknown model") break } break case 159: //On-Off LED case 259: infoMsg += " (On-Off LED mode: " + (valueInt==0?"All)":"One)") break case 160: //Firmware Update Indicator case 260: infoMsg += " (Firmware Update Indicator " + (valueInt==0?red("disabled"):green("enabled")) + ")" break case 161: //Relay Click case 261: infoMsg += " (Relay Click " + (valueInt==0?green("enabled"):red("disabled")) + ")" break case 162: //Double-Tap config button to clear notification case 262: infoMsg += " (Double-Tap config button " + (valueInt==0?green("enabled"):red("disabled")) + ")" break case 163: //LED bar display levels case 263: infoMsg += " (LED bar display levels: ${valueInt?:'full range'})" break default: infoMsg += " [0x${intTo32bitUnsignedHex(valueInt)}] " + orangeRed("Undefined Parameter $attrInt") break } if (infoEnable) log.info infoMsg state."parameter${attrInt}value" = valueInt //update state variable with value received from device if ((attrInt==9) || (attrInt==10) || (attrInt==13) || (attrInt==14) || (attrInt==15) || (attrInt==55) || (attrInt==56)) valueInt = convertByteToPercent(valueInt) //these attributes are stored as bytes but presented as percentages if (attrInt>0) device.updateSetting("parameter${attrInt}",[value:"${valueInt}",type:configParams["parameter${attrInt.toString().padLeft(3,"0")}"].type?.toString()]) //update local setting with value received from device if ((attrInt==95 && parameter95custom!=null)||(attrInt==96 && parameter96custom!=null)) { //if custom hue was set, update the custom state variable also device.updateSetting("parameter${attrInt}custom",[value:"${Math.round(valueInt/255*360)}",type:configParams["parameter${attrInt.toString().padLeft(3,"0")}"].type?.toString()]) state."parameter${attrInt}custom" = Math.round(valueInt/255*360) } if (state.model?.substring(0,5)!="VZM35" && (attrInt==21 || attrInt==22 || attrInt==158 || attrInt==258)) { //fan does not support leading/trailing edge dimming state.dimmingMethod = "Leading Edge" //default to Leading Edge if (parameter21=="1") { //if neutral wiring then select based on remote switch type if (parameter22=="0") state.dimmingMethod = "Trailing Edge" //no aux if (parameter22=="1") state.dimmingMethod = "Leading Edge" //dumb 3-way if (parameter22=="2") state.dimmingMethod = "Trailing Edge" //smart aux if (parameter22=="3") { if (parameter158=="1" || parameter258=="1") { //Switch Mode is On-Off state.dimmingMethod = "Full Wave" } else { //Switch Mode is Dimmer state.dimmingMethod = "Trailing Edge" device.updateSetting("parameter22",[value:"0",type:"enum"]) state.parameter22value=0 } } } if (traceEnable||debugEnable) log.trace "${device.displayName}: Dimming Method = ${state.dimmingMethod}" } if ((valueInt==configParams["parameter${attrInt.toString()?.padLeft(3,"0")}"]?.default?.toInteger()) //IF setting is the default && (attrInt!=21)&&(attrInt!=22)&&(attrInt!=51)&&(attrInt!=52)&&(attrInt!=257)&&(attrInt!=158)&&(attrInt!=258)) { //AND not read-only or primary config params device.clearSetting("parameter${attrInt}") //THEN clear the setting (so only changed settings are displayed) if (traceEnable||debugEnable) log.trace "${device.displayName}: parse() cleared parameter${attrInt} since it is the default" } } else if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("${clusterName} UNKNOWN COMMAND(${descMap.command})" + (debugEnable?" ${zigbee.getEvent(description)}":"")) break default: if (infoEnable||debugEnable) log.warn "${device.displayName}: "+fireBrick("UNKNOWN CLUSTER($clusterHex) $descMap ${zigbee.getEvent(description)}") break } state.lastEventReceived = clusterName state.lastEventTime = nowFormatted() state.lastEventAttribute = attrInt state.lastEventValue = descMap.value } def ping() { if (infoEnable) log.info "${device.displayName}: ping()" refresh() } def poll() { if (infoEnable) log.info "${device.displayName}: poll()" refresh() } def presetLevel(value) { if (infoEnable) log.info "${device.displayName}: presetLevel(${value})" state.lastCommandSent = "presetLevel(${value})" state.lastCommandTime = nowFormatted() def cmds = [] Integer scaledValue = value==null?null:Math.min(Math.max(convertPercentToByte(value.toInteger()),1),255) //ZigBee levels range from 0x01-0xfe with 00 and ff = 'use previous' cmds += setParameter(13, scaledValue) if (traceEnable) log.trace "${device.displayName}: preset $cmds" return cmds } def quickStart() { quickStartVariables() def startLevel = device.currentValue("level").toInteger() def cmds= [] if (settings.parameter23?.toInteger()>0 ) { //only do quickStart if enabled if (infoEnable) log.info "${device.displayName}: quickStart(" + (state.model?.substring(0,5)!="VZM35"?"${settings.parameter23}%)":"${settings.parameter23}s)") if (state.model?.substring(0,5)!="VZM35") { //IF not the Fan switch THEN emulate quickStart if (startLevel if (i==23 && (state.model?.substring(0,5)!="VZM35")) { //quickStart is implemented in firmware for the fan, emulated in this driver for 2-in-1 Dimmer quickStartVariables() } switch (option) { case "": case " ": case null: if (((i>=1)&&(i<=8))&&(state?."parameter${i}value"==null)||(i==21)||(i==22)||(i==51)||(i==52)||(i==157)||(i==158)||(i==257)||(i==258)) cmds += getParameter(i) //if option is blank or null then refresh primary and read-only settings break case "User": if (settings."parameter${i}"!=null) cmds += getParameter(i) //if option is User then refresh settings that are non-blank break case "All": cmds += getParameter(i) //if option is All then refresh all settings break default: if (infoEnable||traceEnable||debugEnable) log.warn "${device.displayName}: Unknonwn refresh option '${option}'" break } } return cmds } def remoteControl(option) { if (infoEnable) log.info "${device.displayName}: remoteControl($option)" state.lastCommandSent = "remoteControl($option)" state.lastCommandTime = nowFormatted() def cmds = [] cmds += zigbee.command(0xfc31,0x10,["mfgCode":"0x122F"],defaultDelay,"${option=="Disabled"?"01":"00"}") //if (state.model?.substring(0,5)!="VZM35") cmds += zigbee.readAttribute(CLUSTER_PRIVATE, 0x0101,["mfgCode":"0x122F"],defaultDelay ) //Fan sends this automatically, but Dimmer does not if (traceEnable) log.trace "${device.displayName}: remoteControl $cmds" return cmds } def resetEnergyMeter() { if (infoEnable) log.info "${device.displayName}: resetEnergyMeter(" + device.currentValue("energy") + "kWh)" state.lastCommandSent = "resetEnergyMeter(" + device.currentValue("energy") + "kWh)" state.lastCommandTime = nowFormatted() def cmds = [] cmds += zigbee.command(0xfc31,0x02,["mfgCode":"0x122F"],defaultDelay,"0") cmds += zigbee.readAttribute(CLUSTER_SIMPLE_METERING, 0x0000) if (traceEnable) log.trace "${device.displayName}: resetEnergyMeter $cmds" return cmds } def setAttribute(Integer cluster, Integer attrInt, Integer dataType, Integer value, Map additionalParams = [:], Integer delay=defaultDelay) { if (cluster==0xfc31) additionalParams = ["mfgCode":"0x122F"] if ((delay==null)||(delay==0)) delay = defaultDelay if (debugEnable) log.debug "${device.displayName} setAttribute(" + "0x${zigbee.convertToHexString(cluster,4)}, " + "0x${zigbee.convertToHexString(attrInt,4)}, " + "0x${zigbee.convertToHexString(dataType,2)}, " + "${value}, ${additionalParams}, ${delay})" def infoMsg = "${device.displayName}: " if (cluster==0xfc31) { infoMsg += " Parameter ${attrInt} value " switch (attrInt) { case 9: //min level case 10: //max level case 13: //default local level case 14: //default remote level case 15: //level after power restored infoMsg += "${value} = ${convertByteToPercent(value)}% on 255 scale" break case 23: quickStartVariables() infoMsg = "" break case 95: case 96: infoMsg += "${value} = ${Math.round(value/255*360)}° on 360° scale" break default: infoMsg = "" break } } else { infoMsg += "" + (cluster==0xfc31?"":clusterLookup(cluster)) + " parameter 0x${zigbee.convertToHexString(attrInt,4)} value ${value}" } if (infoEnable && infoMsg) log.info infoMsg + (delay==defaultDelay?"":" [delay ${delay}]") def cmds = zigbee.writeAttribute(cluster, attrInt, dataType, value, additionalParams, delay) if (traceEnable) log.trace "${device.displayName}: setAttribute $cmds" return cmds } def getAttribute(Integer cluster, Integer attrInt=0, Map additionalParams = [:], Integer delay=defaultDelay) { if (cluster==0xfc31) additionalParams = ["mfgCode":"0x122F"] if (delay==null||delay==0) delay = defaultDelay if (traceEnable) log.trace "${device.displayName}: Get "+(cluster==0xfc31?"":clusterLookup(cluster))+" parameter ${attrInt}"+(delay==defaultDelay?"":" [delay ${delay}]") if (debugEnable) log.debug "${device.displayName} getAttribute(0x${zigbee.convertToHexString(cluster,4)}, " + (attrInt==null?null:"0x${zigbee.convertToHexString(attrInt,4)}") + ", ${additionalParams}, ${delay})" if (cluster==0xfc31 && attrInt==23 && state.model?.substring(0,5)!="VZM35") { //if not Fan, get the quickStart values from state variables since dimmer does not store these if (infoEnable) log.info "${device.displayName}: Receive parameter ${attrInt} value ${state?.parameter23value} (QuickStart " + (state.parameter23value?.toInteger()==0?red("disabled"):"${state.parameter23value?.toInteger()}" + state.model?.substring(0,5)!="VZM35"?"${settings.parameter23}%":"${settings.parameter23}s") + ")" if (infoEnable) log.info "${device.displayName}: Receive parameter ${attrInt} level ${state?.parameter23level} (QuickStart startup level)" if (settings.parameter23level?.toInteger()==defaultQuickLevel) device.clearSetting("parameter23level") } def cmds = [] //String mfgCode = "{}" //if(additionalParams.containsKey("mfgCode")) mfgCode = "{${additionalParams.get("mfgCode")}}" //String rattrArgs = "0x${device.deviceNetworkId} 0x01 0x${zigbee.convertToHexString(cluster,4)} " + // "0x${zigbee.convertToHexString(attrInt,4)} " + // "$mfgCode" //cmds += ["he rattr $rattrArgs", "delay $delay"] cmds += zigbee.readAttribute(cluster, attrInt, additionalParams, delay) if (traceEnable) log.trace "${device.displayName}: getAttribute $cmds" return cmds } def setLevel(value, duration=0xFFFF) { if (infoEnable) log.info "${device.displayName}: setLevel($value" + (duration==0xFFFF?")":", ${duration}s)") state.lastCommandSent = "setLevel($value" + (duration==0xFFFF?")":", ${duration}s)") state.lastCommandTime = nowFormatted() if (duration!=0xFFFF) duration = duration.toInteger()*10 //firmware duration in 10ths def cmds = [] if (device.currentValue("switch")=="off" && value!=0) cmds += zigbee.setLevel(1,0,0) //if device is off then turn on at 1 //hack for firmware bug. should be fixed in v2.11 cmds += zigbee.setLevel(value.toInteger(),duration,defaultDelay) if (traceEnable) log.trace "${device.displayName}: setLevel $cmds" return cmds } def setParameter(number, value=null, size=null) { if (size==null || size==" ") size = configParams["parameter${number.toString().padLeft(3,'0')}"]?.size?:8 if (infoEnable) log.info value!=null?"${device.displayName}: setParameter($number, $value, $size)":"${device.displayName}: getParameter($number)" state.lastCommandSent = value!=null? "setParameter($number, $value, $size)": "getParameter($number)" state.lastCommandTime = nowFormatted() def cmds = [] Integer paramId = number.toInteger() Integer paramValue = (value?:0).toInteger() Integer paramSize = calculateSize(size).toInteger() if (value!=null) cmds += setAttribute(0xfc31,paramId,paramSize,paramValue) if (number==52 || number==158 || number==258) cmds += "delay $longDelay" //allow extra time when changing modes cmds += getParameter(paramId) if (traceEnable) log.trace value!=null?"${device.displayName}: setParameter $cmds":"${device.displayName}: getParameter $cmds" return cmds } def getParameter(number=null) { //if (infoEnable) log.info "${device.displayName}: getParameter($number)" //state.lastCommandSent = "getParameter($number)" //state.lastCommandTime = nowFormatted() //this is not a custom command. Only use state variable for commands on the device details page def cmds = [] cmds += getAttribute(0xfc31, number) if (traceEnable) log.trace "${device.displayName}: getParameter $cmds" return cmds } def setPrivateCluster(attributeId, value=null, size=8) { //for backward compatibility log.warn "${device.displayName}: setPrivateCluster(${red(italic(bold('command is depreciated. Use setParameter instead')))})" return setParameter(attributeId, value, size.toInteger()) } def setSpeed(value) { // FOR FAN ONLY if (infoEnable) log.info "${device.displayName}: setSpeed(${value})" state.lastCommandSent = "setSpeed(${value})" state.lastCommandTime = nowFormatted() def currentLevel = device.currentValue("level")==null?0:device.currentValue("level").toInteger() if (device.currentValue("switch")=="off") currentLevel = 0 boolean smartMode = device.currentValue("smartFan")=="Enabled" def newLevel = 0 def cmds = [] switch (value) { case "off": cmds += zigbee.off(defaultDelay) break case "low": cmds += zigbee.setLevel(smartMode?20:33) break case "medium-low": //placeholder since Hubitat natively supports 5-speed fans cmds += zigbee.setLevel(40) break case "medium": cmds += zigbee.setLevel(smartMode?60:66) break case "medium-high": //placeholder since Hubitat natively supports 5-speed fans cmds += zigbee.setLevel(80) break case "high": cmds += zigbee.setLevel(100) break case "on": cmds += zigbee.on(defaultDelay) break case "up": if (currentLevel<=0 ) {newLevel=20} else if (currentLevel<=20) {newLevel=smartMode?40:60} else if (currentLevel<=40) {newLevel=60} else if (currentLevel<=60) {newLevel=smartMode?80:100} else if (currentLevel<=100) {newLevel=100} cmds += zigbee.setLevel(newLevel) break case "down": if (currentLevel>80) {newLevel=smartMode?80:60} else if (currentLevel>60) {newLevel=60} else if (currentLevel>40) {newLevel=smartMode?40:20} else if (currentLevel>20) {newLevel=20} else if (currentLevel>0) {newLevel=currentLevel} cmds += zigbee.setLevel(newLevel) break } if (traceEnable) log.trace "${device.displayName}: setSpeed $cmds" return cmds } def setZigbeeAttribute(cluster, attributeId, value=null, size=8) { if (infoEnable) log.info value!=null?"${device.displayName}: setZigbeeAttribute(${cluster}, ${attributeId}, ${value}, ${size})":"${device.displayName}: getZigbeeAttribute(${cluster}, ${attributeId})" state.lastCommandSent = value!=null? "setZigbeeAttribute(${cluster}, ${attributeId}, ${value}, ${size})": "getZigbeeAttribute(${cluster}, ${attributeId})" state.lastCommandTime = nowFormatted() def cmds = [] Integer setCluster = cluster.toInteger() Integer attId = attributeId.toInteger() Integer attValue = (value?:0).toInteger() Integer attSize = calculateSize(size).toInteger() if (value!=null) cmds += setAttribute(setCluster,attId,attSize,attValue,[:],attId==258?longDelay:defaultDelay) cmds += getAttribute(setCluster, attId) //if (traceEnable) log.trace "${device.displayName}: setZigbeeAttribute $cmds" return cmds } def startLevelChange(direction, duration=null) { def newLevel = direction=="up"?100:device.currentValue("switch")=="off"?0:1 if (infoEnable) log.info "${device.displayName}: startLevelChange(${direction}" + (duration==null?")":", ${duration}s)") state.lastCommandSent = "startLevelChange(${direction}" + (duration==null?")":", ${duration}s)") state.lastCommandTime = nowFormatted() def cmds = [] cmds += duration==null?zigbee.setLevel(newLevel):zigbee.setLevel(newLevel, duration) if (traceEnable) log.trace "${device.displayName}: startLevelChange $cmds" return cmds } def stopLevelChange() { if (infoEnable) log.info "${device.displayName}: stopLevelChange()" // at level " + device.currentValue("level") state.lastCommandSent = "stopLevelChange()" state.lastCommandTime = nowFormatted() def cmds = [] cmds += ["he cmd 0x${device.deviceNetworkId} 0x${device.endpointId} ${CLUSTER_LEVEL_CONTROL} ${COMMAND_STOP} {}","delay $defaultDelay"] if (traceEnable) log.trace "${device.displayName}: stopLevelChange $cmds" return cmds } def toggle() { def toggleDirection = device.currentValue("switch")=="off"?"off->on":"on->off" if (infoEnable) log.info "${device.displayName}: toggle(${toggleDirection})" state.lastCommandSent = "toggle(${toggleDirection})" state.lastCommandTime = nowFormatted() def cmds = [] //cmds += zigbee.command(CLUSTER_ON_OFF, COMMAND_TOGGLE) //toggle is inconsistent with quickStart, so we emulate toggle with on/off instead //if having trouble keeping multiple bulbs in sync, use the below code to emulate toggle if (device.currentValue("switch")=="off") { if (settings.parameter23?.toInteger()>0) cmds += quickStart() //IF quickStart is enabled THEN quickStart else { cmds += zigbee.on(defaultDelay) } } else { cmds += zigbee.off(defaultDelay) } if (traceEnable) log.trace "${device.displayName}: toggle $cmds" return cmds } def updated(option) { // called when "Save Preferences" is requested option = (option==null||option==" ")?"":option if (infoEnable) log.info "${device.displayName}: updated(${option})" state.lastCommandSent = "updated(${option})" state.lastCommandTime = nowFormatted() state.driverDate = getDriverDate() state.model = device.getDataValue('model') def changedParams = [] def cmds = [] def nothingChanged = true int defaultValue int newValue int oldValue getParameterNumbers().each{ i -> defaultValue=configParams["parameter${i.toString().padLeft(3,'0')}"].default.toInteger() oldValue=state."parameter${i}value"!=null?state."parameter${i}value".toInteger():defaultValue if ((i==9)||(i==10)||(i==13)||(i==14)||(i==15)||(i==55)||(i==56)) { //convert the percent preferences back to byte values before testing for changes defaultValue=convertPercentToByte(defaultValue) oldValue=convertPercentToByte(oldValue) } if (i==23 && parameter23level!=null) { if (parameter23level.toInteger()==defaultQuickLevel.toInteger()) device.clearSetting("parameter23level") } if ((i==95 && parameter95custom!=null)||(i==96 && parameter96custom!=null)) { //IF a custom hue value is set if ((Math.round(settings?."parameter${i}custom"?.toInteger()/360*255)==settings?."parameter${i}"?.toInteger())) { //AND custom setting is same as normal setting device.clearSetting("parameter${i}custom") //THEN clear custom hue and use normal color if (infoEnable) log.info "${device.displayName}: Cleared Custom Hue setting since it equals standard color setting" } oldvalue=state."parameter${i}custom"!=null?state."parameter${i}custom".toInteger():oldValue } newValue = calculateParameter(i) if ((option == "Default")&&(i!=21)&&(i!=22)&&(i!=51)&&(i!=52)&&(i!=157)&&(i!=158)&&(i!=257)&&(i!=258)){ //if DEFAULT option was selected then use the default value (but don't change switch modes) newValue = defaultValue if (traceEnable||debugEnable) log.trace "${device.displayName}: updated() has cleared parameter${attrInt}" device.clearSetting("parameter${i}") //and clear the local settings so they go back to default values if (i==23) device.clearSetting("parameter${i}level") //clear the custom quickstart level if (i==95 || i==96) device.clearSetting("parameter${i}custom") //clear the custom custom hue values } //If a setting changed OR we selected ALL then update parameters in the switch (but don't change switch modes when ALL is selected) //log.debug "Param:$i default:$defaultValue oldValue:$oldValue newValue:$newValue setting:${settings."parameter$i"} `$option`" if ((newValue!=oldValue) || ((option=="User")&&(settings."parameter${i}"!=null)) || ((option=="Default"||option=="All")&&(i!=158)&&(i!=258))) { if (i!=21 && i!=51 && i!=257) { //IF this is not a read-only parameter cmds += setParameter(i, newValue.toInteger()) //THEN set the new value changedParams += i nothingChanged = false } } if ((i==23)&&(state.model?.substring(0,5)!="VZM35")) { //IF not Fan switch THEN manually update the quickStart state variables since Dimmer does not store these quickStartVariables() } } if (groupbinding1 && !state?.groupbinding1) { cmds += bind(["zdo bind 0x${device.deviceNetworkId} 0x02 0x01 0x0006 {${device.zigbeeId}} {${zigbee.convertToHexString(groupbinding1?.toInteger(), 4)}}"]) cmds += bind(["zdo bind 0x${device.deviceNetworkId} 0x02 0x01 0x0008 {${device.zigbeeId}} {${zigbee.convertToHexString(groupbinding1?.toInteger(), 4)}}"]) state.groupbinding1 = groupbinding1 cmds += "delay 60000" //binding seems to take up to 60 seconds cmds += getParameter(51) //update number of bindings counter nothingChanged = false } else { if (!groupbinding1 && state?.groupbinding1) { cmds += bind(["zdo unbind 0x${device.deviceNetworkId} 0x02 0x01 0x0006 {${device.zigbeeId}} {${zigbee.convertToHexString(state.groupbinding1?.toInteger(), 4)}}"]) cmds += bind(["zdo unbind 0x${device.deviceNetworkId} 0x02 0x01 0x0008 {${device.zigbeeId}} {${zigbee.convertToHexString(state.groupbinding1?.toInteger(), 4)}}"]) state.groupbinding1 = null cmds += "delay 60000" //binding seems to take up to 60 seconds cmds += getParameter(51) //update number of bindings counter nothingChanged = false } } if (groupbinding2 && !state?.groupbinding2) { cmds += bind(["zdo bind 0x${device.deviceNetworkId} 0x02 0x01 0x0006 {${device.zigbeeId}} {${zigbee.convertToHexString(groupbinding2?.toInteger(), 4)}}"]) cmds += bind(["zdo bind 0x${device.deviceNetworkId} 0x02 0x01 0x0008 {${device.zigbeeId}} {${zigbee.convertToHexString(groupbinding2?.toInteger(), 4)}}"]) state.groupbinding2 = groupbinding2 cmds += "delay 60000" //binding seems to take up to 60 seconds cmds += getParameter(51) //update number of bindings counter nothingChanged = false } else { if (!groupbinding2 && state?.groupbinding2) { cmds += bind(["zdo unbind 0x${device.deviceNetworkId} 0x02 0x01 0x0006 {${device.zigbeeId}} {${zigbee.convertToHexString(state.groupbinding2?.toInteger(), 4)}}"]) cmds += bind(["zdo unbind 0x${device.deviceNetworkId} 0x02 0x01 0x0008 {${device.zigbeeId}} {${zigbee.convertToHexString(state.groupbinding2?.toInteger(), 4)}}"]) state.groupbinding2 = null cmds += "delay 60000" //binding seems to take up to 60 seconds cmds += getParameter(51) //update number of bindings counter nothingChanged = false } } if (groupbinding3 && !state?.groupbinding3) { cmds += bind(["zdo bind 0x${device.deviceNetworkId} 0x02 0x01 0x0006 {${device.zigbeeId}} {${zigbee.convertToHexString(groupbinding3?.toInteger(), 4)}}"]) cmds += bind(["zdo bind 0x${device.deviceNetworkId} 0x02 0x01 0x0008 {${device.zigbeeId}} {${zigbee.convertToHexString(groupbinding3?.toInteger(), 4)}}"]) state.groupbinding3 = groupbinding3 cmds += "delay 60000" //binding seems to take up to 60 seconds cmds += getParameter(51) //update number of bindings counter nothingChanged = false } else { if (!groupbinding3 && state?.groupbinding3) { cmds += bind(["zdo unbind 0x${device.deviceNetworkId} 0x02 0x01 0x0006 {${device.zigbeeId}} {${zigbee.convertToHexString(state.groupbinding3?.toInteger(), 4)}}"]) cmds += bind(["zdo unbind 0x${device.deviceNetworkId} 0x02 0x01 0x0008 {${device.zigbeeId}} {${zigbee.convertToHexString(state.groupbinding3?.toInteger(), 4)}}"]) state.groupbinding3 = null cmds += "delay 60000" //binding seems to take up to 60 seconds cmds += getParameter(51) //update number of bindings counter nothingChanged = false } } if ((infoEnable||traceEnable||debugEnable)) { if (nothingChanged) { log.info "${device.displayName}: No DEVICE settings were changed" log.info "${device.displayName}: Info logging " + (infoEnable?green("Enabled"):red("Disabled")) log.trace "${device.displayName}: Trace logging " + (traceEnable?green("Enabled"):red("Disabled")) log.debug "${device.displayName}: Debug logging " + (debugEnable?green("Enabled"):red("Disabled")) } } if (infoEnable && disableInfoLogging) { log.info "${device.displayName}: Info Logging will be disabled in $disableInfoLogging minutes" runIn(disableInfoLogging*60,infoLogsOff) } if (traceEnable && disableTraceLogging) { log.trace "${device.displayName}: Trace Logging will be disabled in $disableTraceLogging minutes" runIn(disableTraceLogging*60,traceLogsOff) } if (debugEnable && disableDebugLogging) { log.debug "${device.displayName}: Debug Logging will be disabled in $disableDebugLogging minutes" runIn(disableDebugLogging*60,debugLogsOff) } return cmds } List updateFirmware() { if (infoEnable) log.info "${device.displayName}: updateFirmware(switch's fwDate: ${state.fwDate}, switch's fwVersion: ${state.fwVersion})" state.lastCommandSent = "updateFirmware()" state.lastCommandTime = nowFormatted() def cmds = [] if (state?.lastUpdateFw && now() - state.lastUpdateFw < 2000) { state.remove("lastUpdateFw") cmds += zigbee.updateFirmware() if (traceEnable) log.trace "${device.displayName}: updateFirmware $cmds" } else { log.warn "Firmware in this channel may be \"beta\" quality. Please check https://community.inovelli.com/t/blue-series-2-1-firmware-changelog-vzm31-sn/12326 before proceeding. Double-click \"Update Firmware\" to proceed" state.lastUpdateFw = now() runIn(2,lastUpdateFwRemove) } return cmds } def lastUpdateFwRemove() {if (state?.lastUpdateFw) state.remove("lastUpdateFw")} void ZigbeePrivateCommandEvent(data) { if (infoEnable) log.info "${device.displayName}: SceneButton: ${data[0]} ButtonAttributes: ${data[1]}" Integer ButtonNumber = Integer.parseInt(data[0],16) Integer ButtonAttributes = Integer.parseInt(data[1],16) switch(zigbee.convertToHexString(ButtonNumber,2) + zigbee.convertToHexString(ButtonAttributes,2)) { case "0200": //Tap Up 1x //if (state.model?.substring(0,5)!="VZM35") quickStart() //If not Fan then emulate quickStart for local button push (this doesn't appear to work - not sure why) buttonEvent(1, "pushed", "physical") break case "0203": //Tap Up 2x buttonEvent(2, "pushed", "physical") break case "0204": //Tap Up 3x buttonEvent(3, "pushed", "physical") break case "0205": //Tap Up 4x buttonEvent(4, "pushed", "physical") break case "0206": //Tap Up 5x buttonEvent(5, "pushed", "physical") break case "0202": //Hold Up buttonEvent(6, "pushed", "physical") break case "0201": //Release Up buttonEvent(7, "pushed", "physical") break case "0100": //Tap Down 1x buttonEvent(1, "held", "physical") break case "0103": //Tap Down 2x buttonEvent(2, "held", "physical") break case "0104": //Tap Down 3x buttonEvent(3, "held", "physical") break case "0105": //Tap Down 4x buttonEvent(4, "held", "physical") break case "0106": //Tap Down 5x buttonEvent(5, "held", "physical") break case "0102": //Hold Down buttonEvent(6, "held", "physical") break case "0101": //Release Down buttonEvent(7, "held", "physical") break case "0300": //Tap Config 1x buttonEvent(8, "pushed", "physical") break case "0303": //Tap Config 2x buttonEvent(9, "pushed", "physical") break case "0304": //Tap Config 3x buttonEvent(10, "pushed", "physical") break case "0305": //Tap Config 4x buttonEvent(11, "pushed", "physical") break case "0306": //Tap Config 5x buttonEvent(12, "pushed", "physical") break case "0302": //Hold Config buttonEvent(13, "pushed", "physical") break case "0301": //Release Config buttonEvent(14, "pushed", "physical") break default: //undefined button function log.warn "${device.displayName}: "+fireBrick("Undefined button function Scene: ${data[0]} Attributes: ${data[1]}") break } } void ZigbeePrivateLEDeffectStopEvent(data) { Integer ledNumber = Integer.parseInt(data[0],16)+1 //internal LED number is 0-based String ledStatus = ledNumber==17?"Stop All":ledNumber==256?"User Cleared":"Stop LED${ledNumber}" if (infoEnable) log.info "${device.displayName}: ledEffectStopEvent: ${ledStatus}" switch(ledNumber){ case 1: case 2: case 3: case 4: case 5: case 6: case 7: case 17: //Full LED bar effects case 256: //user double-pressed the config button to clear the notification sendEvent(name:"ledEffect", value: "${ledStatus}") break default: log.warn "${device.displayName}: "+fireBrick("Undefined LEDeffectStopEvent: ${data[0]}") break } } void buttonEvent(button, action, type = "digital") { if (infoEnable) log.info "${device.displayName}: ${type} Button ${button} was ${action}" sendEvent(name: action, value: button, isStateChange: true, type: type) switch (button) { case 1: case 2: case 3: case 4: case 5: sendEvent(name:"lastButton", value: "${action=='pushed'?'Tap '.padRight(button+4, '▲'):'Tap '.padRight(button+4, '▼')}") break case 6: sendEvent(name:"lastButton", value: "${action=='pushed'?'Hold ▲':'Hold ▼'}") break case 7: sendEvent(name:"lastButton", value: "${action=='pushed'?'Release ▲':'Release ▼'}") break case 8: case 9: case 10: case 11: case 12: sendEvent(name:"lastButton", value: "Tap ".padRight(button-3, "►")) break case 13: sendEvent(name:"lastButton", value: "Hold ►") break case 14: sendEvent(name:"lastButton", value: "Release ►") break } } void hold(button) { buttonEvent(button, "held", "digital") } void push(button) { buttonEvent(button, "pushed", "digital") } void release(button) { buttonEvent(button, "released", "digital") } def pressUpX1() { buttonEvent(1, "pushed", "digital") } def pressDownX1() { buttonEvent(1, "held", "digital") } def pressUpX2() { buttonEvent(2, "pushed", "digital") } def pressDownX2() { buttonEvent(2, "held", "digital") } def pressUpX3() { buttonEvent(3, "pushed", "digital") } def pressDownX3() { buttonEvent(3, "held", "digital") } def pressUpX4() { buttonEvent(4, "pushed", "digital") } def pressDownX4() { buttonEvent(4, "held", "digital") } def pressUpX5() { buttonEvent(5, "pushed", "digital") } def pressDownX5() { buttonEvent(5, "held", "digital") } def holdUp() { buttonEvent(6, "pushed", "digital") } def holdDown() { buttonEvent(6, "held", "digital") } def releaseUp() { buttonEvent(7, "pushed", "digital") } def releaseDown() { buttonEvent(7, "held", "digital") } def pressConfigX1() { buttonEvent(8, "pushed", "digital") } def pressConfigX2() { buttonEvent(9, "pushed", "digital") } def pressConfigX3() { buttonEvent(10, "pushed", "digital") } def pressConfigX4() { buttonEvent(11, "pushed", "digital") } def pressConfigX5() { buttonEvent(12, "pushed", "digital") } def holdConfig() { buttonEvent(13, "held", "digital") } def releaseConfig() { buttonEvent(14, "released", "digital") } /** * ----------------------------------------------------------------------------- * Everything below here are LIBRARY includes and should NOT be edited manually! * ----------------------------------------------------------------------------- * --- Nothings to edit here, move along! -------------------------------------- * ----------------------------------------------------------------------------- * --- user by xiaomi ---------------------------------------------------------- */ private getCLUSTER_BASIC() { 0x0000 } private getCLUSTER_POWER() { 0x0001 } private getCLUSTER_IDENTIFY() { 0x0003 } private getCLUSTER_GROUP() { 0x0004 } private getCLUSTER_SCENES() { 0x0005 } private getCLUSTER_ON_OFF() { 0x0006 } private getCLUSTER_LEVEL_CONTROL() { 0x0008 } private getCLUSTER_WINDOW_POSITION() { 0x000d } private getCLUSTER_WINDOW_COVERING() { 0x0102 } private getCLUSTER_SIMPLE_METERING() { 0x0702 } private getCLUSTER_ELECTRICAL_MEASUREMENT() { 0x0b04 } private getCLUSTER_PRIVATE() { 0xFC31 } private getCOMMAND_MOVE_LEVEL() { 0x00 } private getCOMMAND_MOVE() { 0x01 } private getCOMMAND_STEP() { 0x02 } private getCOMMAND_STOP() { 0x03 } private getCOMMAND_MOVE_LEVEL_ONOFF() { 0x04 } private getCOMMAND_MOVE_ONOFF() { 0x05 } private getCOMMAND_STEP_ONOFF() { 0x06 } private getBASIC_ATTR_POWER_SOURCE() { 0x0007 } private getPOWER_ATTR_BATTERY_PERCENTAGE_REMAINING() { 0x0021 } private getPOSITION_ATTR_VALUE() { 0x0055 } private getCOMMAND_OPEN() { 0x00 } private getCOMMAND_CLOSE() { 0x01 } private getCOMMAND_OFF() { 0x00 } private getCOMMAND_ON() { 0x01 } private getCOMMAND_TOGGLE() { 0x02 } private getCOMMAND_PAUSE() { 0x02 } private getENCODING_SIZE() { 0x39 } //Functions to enhance text appearance String bold(s) { return "$s" } String italic(s) { return "$s" } String mark(s) { return "$s" } //yellow background String strike(s) { return "$s" } String underline(s) { return "$s" } String hue(h,s) { h = Math.min(Math.max((h!=null?h:170),1),255) //170 is Inovelli factory default blue if (h==255) s = ' ' + s + ' ' else s = '' + s + '' return s } //Reds String indianRed(s) { return '' + s + ''} String lightCoral(s) { return '' + s + ''} String crimson(s) { return '' + s + ''} String red(s) { return '' + s + ''} String fireBrick(s) { return '' + s + ''} String coral(s) { return '' + s + ''} //Oranges String orangeRed(s) { return '' + s + ''} String darkOrange(s) { return '' + s + ''} String orange(s) { return '' + s + ''} //Yellows String gold(s) { return '' + s + ''} String yellow(s) { return '' + s + ''} String paleGoldenRod(s) { return '' + s + ''} String peachPuff(s) { return '' + s + ''} String darkKhaki(s) { return '' + s + ''} //Greens String limeGreen(s) { return '' + s + ''} String green(s) { return '' + s + ''} String darkGreen(s) { return '' + s + ''} String olive(s) { return '' + s + ''} String darkOliveGreen(s) { return '' + s + ''} String lightSeaGreen(s) { return '' + s + ''} String darkCyan(s) { return '' + s + ''} String teal(s) { return '' + s + ''} //Blues String cyan(s) { return '' + s + ''} String lightSteelBlue(s) { return '' + s + ''} String steelBlue(s) { return '' + s + ''} String lightSkyBlue(s) { return '' + s + ''} String deepSkyBlue(s) { return '' + s + ''} String dodgerBlue(s) { return '' + s + ''} String blue(s) { return '' + s + ''} String midnightBlue(s) { return '' + s + ''} //Purples String magenta(s) { return '' + s + ''} String rebeccaPurple(s) { return '' + s + ''} String blueViolet(s) { return '' + s + ''} String slateBlue(s) { return '' + s + ''} String darkSlateBlue(s) { return '' + s + ''} //Browns String burlywood(s) { return '' + s + ''} String goldenrod(s) { return '' + s + ''} String darkGoldenrod(s) { return '' + s + ''} String sienna(s) { return '' + s + ''} //Grays String lightGray(s) { return '' + s + ''} String gray(s) { return '' + s + ''} String dimGray(s) { return '' + s + ''} String slateGray(s) { return '' + s + ''} String black(s) { return '' + s + ''} //********************************************************************************** //****** End of HTML enhancement functions. //**********************************************************************************