├── .github └── ISSUE_TEMPLATE │ ├── bug_report.md │ └── feature_request.md ├── README.md ├── README.md.old ├── alarm.yaml ├── automation.yaml ├── automation └── panic_mode.yaml ├── custom_components └── alarm_control_panel │ └── bwalarm.py ├── example automations └── panic_mode.yaml ├── guidance └── configuration.md ├── historic_changelog.md ├── panel_custom.yaml ├── panels └── alarm.html └── www ├── alarm ├── alarm.css ├── custom-element.html ├── donate │ ├── bch.png │ ├── btc.png │ ├── eth.png │ ├── ltc.png │ ├── paypal.png │ └── xrp.png └── lobster.woff2 ├── images ├── camera-garage.jpg ├── camera-outdoor.jpg ├── camera-pool.jpg ├── camera-tv-room.jpg ├── gazos.jpg └── ha.png └── lib ├── countdown360.js ├── jquery-3.2.1.min.js ├── jscolor.js └── sha256.js /.github/ISSUE_TEMPLATE/bug_report.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Bug report 3 | about: Create a report to help us improve 4 | 5 | --- 6 | 7 | **Describe the bug** 8 | A clear and concise description of what the bug is. 9 | 10 | **To Reproduce** 11 | Steps to reproduce the behavior: 12 | 1. Go to '...' 13 | 2. Click on '....' 14 | 3. Scroll down to '....' 15 | 4. See error 16 | 17 | **Expected behavior** 18 | A clear and concise description of what you expected to happen. 19 | 20 | **Screenshots** 21 | If applicable, add screenshots to help explain your problem. 22 | 23 | **Browser (please complete the following information):** 24 | - Browser [e.g. chrome, safari] 25 | - Version [e.g. 22] 26 | 27 | **Additional context** 28 | Add any other context about the problem here. 29 | -------------------------------------------------------------------------------- /.github/ISSUE_TEMPLATE/feature_request.md: -------------------------------------------------------------------------------- 1 | --- 2 | name: Feature request 3 | about: Suggest an idea for this project 4 | 5 | --- 6 | 7 | **Is your feature request related to a problem? Please describe.** 8 | A clear and concise description of what the problem is. Ex. I'm always frustrated when [...] 9 | 10 | **Describe the solution you'd like** 11 | A clear and concise description of what you want to happen. 12 | 13 | **Describe alternatives you've considered** 14 | A clear and concise description of any alternative solutions or features you've considered. 15 | 16 | **Additional context** 17 | Add any other context or screenshots about the feature request here. 18 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Home Assistant - Custom Alarm Interface! 2 | ## Intro :-) 3 | 4 | Welcome my fellow modders, tinkerers, home assistant wizards!! 5 | 6 | Follow the thread [here](https://community.home-assistant.io/t/yet-another-take-on-an-alarm-system/32386) 7 | 8 | Consider donating to this project to keep it going as anything contributed will be placed back in to enable more hardware integration, new features and bug squashing. 9 | 10 | This is very much a community project so if you wish to chip in then please do!! I could really use a CSS, animation, design guru to make this look amazing. Also please feel free to leave comments, suggestions, enhancements and fixes!! 11 | 12 | **NOTE!!! MAJOR CHANGE** It's time to publish the New UI and settings into the master release. 13 | 14 | ## Installation 15 | 16 | You will need to copy the following files into your home assistant configuration directory 17 | 18 | alarm.yaml *This files stores your alarm configuration. An options page will be created for this file* 19 | custom_components/alarm_control_panel/bwalarm.py *The brains of the operation. This is the logic of the custom alarm system* 20 | panels/alarm.html *This is the interface for the custom alarm component. It's actually optional as the alarm will function without it but recommended for ease of setup* 21 | www/alarm/[ALL FILES] *These files control how the interface looks and feels* 22 | www/lib/[ALL FILES] *These files add additional functionality to the interface in order to work* 23 | www/images/ha.png *An image file used for the interface log* 24 | 25 | To get things working with Home Assistant (HA) you will need to adjust your configuration.yaml to instruct HA to use your new custom alarm component, add the following to this file: 26 | ``` 27 | alarm_control_panel: !include alarm.yaml 28 | ``` 29 | You will also need to tell HA where your new panel interface file is. Also add the following to your configuration.yaml: 30 | ``` 31 | panel_custom: !include panel_custom.yaml 32 | ``` 33 | You may need to restart HA if the component doesn't load first time as HA will need to install a dependency (ruamel.yaml). 34 | 35 | It's advisable to start with a new alarm.yaml file with the minimum configuration set: 36 | ``` 37 | platform: bwalarm 38 | name: House 39 | ``` 40 | Your new interface can be used to modify your alarm.yaml directly. 41 | 42 | The default password to access the settings page is: **HG28!!&dn** 43 | 44 | Please test and provide feedback/suggestions. 45 | 46 | ### Features: 47 | - State specific groups and times (NEW) 48 | - User specific codes 49 | - Panic Mode 50 | - MQTT Integration 51 | - Floorplan Integration 52 | - Alarm State Persistence on reboots/power restore 53 | - Lockout of HA sidebar when armed 54 | - Custom Panel allowing your own html to display whatever you choose (Cameras, Sliding Images etc) 55 | - Passcode Attemps/Lockout 56 | - Support for custom device states 57 | - Code panel 0-9 on disarm only 58 | - Weather Status (Optional) - **NOTE:** Weather sensor nows supports generic sensors (sensor.weather_summary & sensor.weather_temperature) if these are not found then it will default to the dark sky sensors (sensor.dark_sky_summary & sensor.dark_sky_temperature) 59 | - Perimeter Mode (Optional) - I use this to only arm a particular set of sensors (doors) whilst I'm using all floors. 60 | - Masks passcode on entry 61 | - clock display (Optional) 62 | - Digit code entry on disarm 63 | - Themed colours depending on alarm state 64 | - Countdown timer on 'Pending' state 65 | - Notification of Open Sensors with the option to override 66 | - Information/Debug panel 67 | 68 | ### Testing 69 | - Tested on HA v0.87 and below. 70 | 71 | ### Change Log: 72 | - 10/02/19: 73 | - [REQUEST] Added option to hide sensors when alarm armed 74 | - [REQUEST] Added option to display in fahrenheit 75 | - [FIX] Panel lockout display 76 | - [FIX] HA 0.87 compatibility 77 | - [REQUEST] Included switches into the sensor lists as requested 78 | - [ENHANCEMENT] Modified the layouts due to polymer changes 79 | 80 | - 27/11/18: 81 | - [FEATURE] Adding some basic error handling which will be enhanced at a later date 82 | - [FIX BUG] Fixed margin issue in firefox (settings) 83 | - [REQUEST] Sorted sensors alphabetically 84 | - [FIX BUG] Fixed clock, serif, weather, passcode display issues 85 | 86 | - 22/11/18: 87 | - Quite a few bugs and issues have been resolved on this release. There has also been a number of changes to the config file layout so you are likely required to start from scratch as the users, themes and panel settings have changed. 88 | 89 | - Updated alarm.html to 1.3.3 90 | - Updated bwalarm.py to 1.1.3 91 | 92 | - fixed duplicate sensors in settings panel 93 | - fixed passcode attempts setting 94 | - fixed code to arm display issues 95 | - fixed persistant mode 96 | - fixed sesnor groups 97 | - fixed code to arm panel display and alignment 98 | - reformated logs 99 | - fixed log (displaying name and image) 100 | - removed windows line feed 101 | - integrated HASS users into alarm automatically however these initially are disabled 102 | - fixed switch breaks on service call 103 | - fixed themes 104 | 105 | ## Note! 106 | Beware, here be dragons! There may be bugs, issues whilst I get this off the ground and there will definately be design problems when used with different size browsers etc. Hopefully we can conquer these in due course!.. 107 | 108 | ## Thanks! 109 | Thanks to the community for all the input into this. 110 | 111 | Consider supporting this project and donate! All funds will go towards bringing new features, hardware support and bug squashing!! 112 | 113 | - BTC Address: 1NFeyzpKKiKbBYSmCLQZQLxBqJbhSbqmwd 114 | - LTC Address: LTUViN3QUESkQk3mG2hvTzhLRQPVAd269f 115 | - XRP Address: rwuMp76ht6dmGvipxwKr5ZE6VpF7ZKC7qs 116 | - ETH Address: 0xCbeD2D2cf0434370c1ca126707009b876b736609 117 | - Paypal: ha.custom.alarm@gmail.com 118 | 119 | ## Credits 120 | [A great countdown JS that I have slightly modded](https://github.com/johnschult/jquery.countdown360) 121 | -------------------------------------------------------------------------------- /README.md.old: -------------------------------------------------------------------------------- 1 | # Home Assistant - Custom Alarm Interface! 2 | ## Intro :-) 3 | 4 | 5 | Welcome my fellow modders, tinkerers, home assistant wizards!! 6 | 7 | Follow the thread [here](https://community.home-assistant.io/t/yet-another-take-on-an-alarm-system/32386) 8 | 9 | Consider donating to this project to keep it going as anything contributed will be placed back in to enable more hardware integration, new features and bug squashing. 10 | 11 | This is very much a community project so if you wish to chip in then please do!! I could really use a CSS, animation, design guru to make this look amazing. Also please feel free to leave comments, suggestions, enhancements and fixes!! 12 | 13 | ### Features: 14 | - Multi Language Support (NEW) 15 | - State specific groups and times (NEW) 16 | - Panic Mode 17 | - MQTT Integration 18 | - Alarm State Persistence on reboots/power restore 19 | - Lockout of HA sidebar when armed 20 | - Custom Panel allowing your own html to display whatever you choose (Cameras, Sliding Images etc) 21 | - Passcode Attemps/Lockout 22 | - Support for custom device states 23 | - Code panel 0-9 on disarm only 24 | - Weather Status (Optional) - **NOTE:** Weather sensor nows supports generic sensors (sensor.weather_summary & sensor.weather_temperature) if these are not found then it will deault to the dark sky sensors (sensor.dark_sky_summary & sensor.dark_sky_temperature) 25 | - Perimeter Mode (Optional) - I use this to only arm a particular set of sensors (doors) whilst im using all floors. 26 | - Masks passcode on entry 27 | - clock display (Optional) 28 | - Digit code entry on disarm 29 | - Themed colours depending on alarm state 30 | - Countdown timer on 'Pending' state 31 | - Notification of Open Sensors with the option to override 32 | - Information/Debug panel 33 | 34 | ### To be implemented: 35 | - Settings page to adjust non-critical features (colours/information) 36 | - Information/Debug Mode to be enhanced 37 | - Screensaver 38 | - Customisable Themes 39 | - Time Based themes (Dark at Night - Light during day) 40 | - Possibly a full black one with a Cylon style bar when activated? 41 | - Please submit some ideas here 42 | - Guest mode / reduced feature set 43 | - Clean up of code (html/css/python) 44 | - Anything anyone else can think of? 45 | 46 | [Installation/Configuration Instructions](guidance/configuration.md) 47 | 48 | ### Testing 49 | - Tested on HA v0.65.5 and below. 50 | 51 | ### Recent Changelog 52 | - (26/03/18) FEATURE - Multi Language Support!!! English and Portuguese options are available out of the box however you can add your own translations by adding in your own translation file. SO technically you could translated the English GUI to say Esperanto! English is the default. This options is set in your alarm.yaml (language: 'portuguese'). Setting this option prompts the front end panel to look for a json styled translation file with the name of your chosen language saved in www/alarm/language/[your language].json The file must exist and use the json 'style' formatted as seen in the example portuguese file. Each english word in the panel will be translated into a matched translated word. See the portuguese json file as an example. 53 | 54 | - (25/03/18) BUG FIX - Moved comments line above the actual config to resolve the hassio issues 55 | - (25/03/18) BUG FIX - Fix to resolve slidebar constantly opening when using mobile devices (Panel 1.0.1 / Bwalarm 1.0.1) 56 | 57 | - (24/03/18) A Massive Thanks to those that have donated!!! IT is very much appreciated and helps to keep this project alive. Also keep the suggestions flowing and lets make this the best alarm system ever!!!!!!!!!!!!!!! 58 | - (24/03/18) MAJOR UPDATE! - State specific groups/times. Each state must! configure it's own groups. Home and Away are mandatory with Perimeter mode optional. The top level groups have been dropped so you will need to remove these from your alarm.yaml. You will need to update your alarm.yaml!. The ignore/notathome groups have been dropped from the setup. Please see the default alarm.yaml to inform your own setup. An example of the configuration below (if you get stuck then post an issue or ask in the forum): 59 | ``` 60 | armed_home: #Either home/away with perimeter as optional 61 | pending_time: 10 #[OPTIONAL] State specific overrides default time 62 | trigger_time: 600 #[OPTIONAL] State specific overrides default time 63 | immediate: #[OPTIONAL however either an immediate or delayed group must exist] 64 | - binary_sensor.whatever 65 | delayed: #[OPTIONAL] 66 | - binary_sensor.whatever 67 | override: #[OPTIONAL] 68 | - binary_sensor.whatever 69 | ``` 70 | - (24/03/18) FEATURE - Added an information button in the bottom right of the panel which shows any detected errors and version information for debugging, needs a little finesse 71 | - (24/03/18) UPDATE - Weather sensor nows supports generic sensors (sensor.weather_summary & sensor.weather_temperature) if these are not found then it will deault to the dark sky sensors (sensor.dark_sky_summary & sensor.dark_sky_temperature) 72 | - (24/03/18) UPDATE - Code cleanup in alarm.html 73 | - (24/03/18) BUG FIX - Removed the need for alarm_script.js (this may re-appear in a later release if we need extra js code) as the hide sidebar feature now natively supports HA close/open sidebar rather than a javascript hack. 74 | 75 | - (14/03/18) BUG FIX - UI fix on the sensor groups moving all active sensors into the immediate group when no pending time is set for that particular state. 76 | - (14/03/18) BUG FIX - Custom pending times now accurate set the countdown clock in the panel UI 77 | - (14/03/18) UPDATE - Perimeter Colours added to customisation 78 | 79 | [Historic Changelog](historic_changelog.md) 80 | 81 | ## Note! 82 | Beware, here be dragons! There may be bugs, issues whilst I get this off the ground and there will definately be design problems when used with different size browsers etc. Hopefully we can conquer these in due course!.. 83 | 84 | ## Thanks! 85 | Thanks to the community for all the input into this. 86 | 87 | Consider supporting this project and donate! All funds will go towards bringing new features, hardware support and bug squashing!! 88 | 89 | - BTC Address: 1NFeyzpKKiKbBYSmCLQZQLxBqJbhSbqmwd 90 | - LTC Address: LTUViN3QUESkQk3mG2hvTzhLRQPVAd269f 91 | - XRP Address: rwuMp76ht6dmGvipxwKr5ZE6VpF7ZKC7qs 92 | - ETH Address: 0xCbeD2D2cf0434370c1ca126707009b876b736609 93 | - Paypal: ha.custom.alarm@gmail.com 94 | 95 | ## Credits 96 | [A great countdown JS that I have slightly modded](https://github.com/johnschult/jquery.countdown360) 97 | -------------------------------------------------------------------------------- /alarm.yaml: -------------------------------------------------------------------------------- 1 | ########################################################## 2 | ## CUSTOM ALARM COMPONENT ALARM.YAML 3 | ## https://github.com/gazoscalvertos/Hass-Custom-Alarm 4 | ## VERSION: 1.0.2 5 | ## MODIFIED: 18/04/18 6 | ## CHANGE LOG: 7 | ## Add Multi Codes, names, pics 8 | ## optional code to arm alarm 9 | ## Default Interface password: HG28!!&dn 10 | ########################################################## 11 | 12 | platform: bwalarm 13 | name: House 14 | panel: 15 | camera_update_interval: '' 16 | cameras: [] 17 | enable_camera_panel: 'False' 18 | enable_clock: 'True' 19 | enable_clock_12hr: 'True' 20 | enable_custom_panel: 'False' 21 | enable_floorplan_panel: 'False' 22 | enable_sensors_panel: 'True' 23 | enable_serif_font: 'True' 24 | enable_weather: 'True' 25 | hide_passcode: 'True' 26 | panel_title: Surname Residence 27 | shadow_effect: 'True' 28 | enable_fahrenheit: false 29 | states: 30 | armed_away: 31 | immediate: 32 | - switch.skylight 33 | - binary_sensor.away_sensor 34 | delayed: 35 | - binary_sensor.away_delayed_sensor 36 | override: [] 37 | pending_time: 2 38 | warning_time: 2 39 | trigger_time: 5 40 | armed_home: 41 | immediate: 42 | - binary_sensor.home_sensor 43 | delayed: 44 | - binary_sensor.home_delayed_sensor 45 | override: [] 46 | pending_time: 2 47 | warning_time: 2 48 | trigger_time: 5 49 | armed_perimeter: 50 | immediate: 51 | - binary_sensor.perimeter_sensor 52 | delayed: 53 | - binary_sensor.perimeter_delayed_sensor 54 | override: [] 55 | pending_time: 0 56 | warning_time: '2' 57 | trigger_time: 600 58 | users: 59 | - id: 85a1f1f7b2f247dfafe718d0cfe5026d 60 | name: test 61 | enabled: false 62 | code: 85a1f1f7b2f247dfafe718d0cfe5026d 63 | picture: /local/images/ha.png 64 | - id: 36ad4844cafe4f3491e8e4e2ac38e0ff 65 | name: bart 66 | enabled: true 67 | code: '2345' 68 | picture: /local/images/hal.png 69 | - id: 2820bcde9f974ef18da3b44b1e494cf3 70 | name: Legacy API password user 71 | enabled: false 72 | code: 2820bcde9f974ef18da3b44b1e494cf3 73 | picture: /local/images/ha.png 74 | - id: 9b229b1e281643409c56bcfbb7e0088a 75 | name: Legacy API password user 76 | enabled: false 77 | code: 9b229b1e281643409c56bcfbb7e0088a 78 | picture: /local/images/ha.png 79 | admin_password: a 80 | code: '1234' 81 | enable_perimeter_mode: true 82 | code_to_arm: false 83 | themes: 84 | - name: aaa 85 | warning_color: '#995BFF' 86 | pending_color: '#FF2943' 87 | disarmed_color: '#FF22E6' 88 | triggered_color: '#FF0000' 89 | armed_home_color: '#C1B1FF' 90 | armed_away_color: '#FF8686' 91 | armed_perimeter_color: '#DAFF9E' 92 | active: false 93 | action_button_border_color: '#3ED5FF' 94 | panic_code: '9876' 95 | enable_log: true 96 | passcode_attempts: '2' 97 | passcode_attempts_timeout: '10' 98 | -------------------------------------------------------------------------------- /automation.yaml: -------------------------------------------------------------------------------- 1 | - id: alarm_armed_away 2 | alias: '[Alarm] Away Mode Armed' 3 | trigger: 4 | - platform: state 5 | entity_id: alarm_control_panel.house 6 | to: 'armed_away' 7 | action: 8 | - data: 9 | message: 'Alarm Away Mode Armed' 10 | target: email/example@gmail.com 11 | service: notify.pushbullet 12 | - data: 13 | message: 'The house alarm has been switched on in away mode. Goodbye' 14 | service: notify.example_phone_tts 15 | 16 | - id: alarm_armed_home 17 | alias: '[Alarm] Home Mode Armed' 18 | trigger: 19 | - platform: state 20 | entity_id: alarm_control_panel.house 21 | to: 'armed_home' 22 | action: 23 | - data: 24 | message: 'Alarm Home Mode Armed' 25 | target: email/example@gmail.com 26 | service: notify.pushbullet 27 | - data: 28 | message: 'The house alarm has been switched on in home mode. Goodnight' 29 | service: notify.example_phone_tts 30 | 31 | - id: alarm_arming_away 32 | alias: '[Alarm] Away Mode Arming' 33 | trigger: 34 | - platform: state 35 | entity_id: alarm_control_panel.house 36 | to: 'pending' 37 | action: 38 | - data: 39 | message: 'House alarm activating, ensure all doors and windows are closed' 40 | service: notify.example_phone_tts 41 | 42 | - id: alarm_disarmed 43 | alias: '[Alarm] Disarmed' 44 | trigger: 45 | - platform: state 46 | entity_id: alarm_control_panel.house 47 | to: 'disarmed' 48 | action: 49 | - service: notify.pushbullet 50 | data: 51 | message: 'Alarm Disabled' 52 | target: email/example@gmail.com 53 | - service: switch.turn_off 54 | entity_id: switch.siren_switch 55 | - data: 56 | message: 'The house alarm has been Deactivated' 57 | service: notify.example_phone_tts 58 | 59 | - id: alarm_triggered 60 | alias: '[Alarm] Triggered' 61 | trigger: 62 | - platform: state 63 | entity_id: alarm_control_panel.house 64 | to: 'triggered' 65 | action: 66 | - service: switch.turn_on 67 | entity_id: switch.siren_switch 68 | - service: notify.pushbullet 69 | data: 70 | message: 'ALARM TRIGGERED!!! {{ states[states.alarm_control_panel.house.attributes.changed_by.split(".")[0]][ states.alarm_control_panel.house.attributes.changed_by.split(".")[1]].name }}' 71 | target: email/example@gmail.com 72 | 73 | - id: alarm_warning 74 | alias: '[Alarm] Warning' 75 | trigger: 76 | - platform: state 77 | entity_id: alarm_control_panel.house 78 | to: 'warning' 79 | action: 80 | - service: notify.pushbullet 81 | data: 82 | message: 'ALARM Warning {{ states[states.alarm_control_panel.house.attributes.changed_by.split(".")[0]][ states.alarm_control_panel.house.attributes.changed_by.split(".")[1]].name }}' 83 | target: email/example@gmail.com 84 | - data: 85 | message: 'Hello, the house alarm has been tripped. Please deactivate' 86 | service: notify.example_phone_tts 87 | 88 | -------------------------------------------------------------------------------- /automation/panic_mode.yaml: -------------------------------------------------------------------------------- 1 | - alias: '[Alarm] Panic Mode' 2 | trigger: 3 | platform: state 4 | entity_id: alarm_control_panel.house 5 | value_template: '{{ state.attributes.panic_mode }}' 6 | to: 'ACTIVE' 7 | action: 8 | service: activate_self_defence_robot -------------------------------------------------------------------------------- /custom_components/alarm_control_panel/bwalarm.py: -------------------------------------------------------------------------------- 1 | """ 2 | CUSTOM ALARM COMPONENT BWALARM 3 | https://github.com/gazoscalvertos/Hass-Custom-Alarm 4 | 5 | VERSION: 1.1.4 6 | MODIFIED: 10/02/19 7 | GazosCalvertos: Yet another take on a custom alarm for Home Assistant 8 | 9 | CHANGE LOG: 10 | -Fixed username issue in log 11 | 12 | """ 13 | 14 | REQUIREMENTS = ['ruamel.yaml==0.15.42'] 15 | 16 | import asyncio 17 | import sys 18 | import copy 19 | import datetime 20 | import logging 21 | import enum 22 | import os 23 | import re 24 | import json 25 | import pytz 26 | import copy 27 | import hashlib 28 | import time 29 | import uuid 30 | 31 | from homeassistant.const import ( 32 | STATE_ALARM_ARMED_AWAY, STATE_ALARM_ARMED_HOME, STATE_ALARM_DISARMED, 33 | STATE_ALARM_PENDING, STATE_ALARM_TRIGGERED, CONF_PLATFORM, CONF_NAME, 34 | CONF_CODE, CONF_PENDING_TIME, CONF_TRIGGER_TIME, CONF_DISARM_AFTER_TRIGGER, 35 | CONF_DELAY_TIME, EVENT_STATE_CHANGED, EVENT_TIME_CHANGED, 36 | STATE_ON, STATE_OFF) 37 | 38 | from operator import attrgetter 39 | from homeassistant.core import callback 40 | from homeassistant.util.dt import utcnow as now 41 | from homeassistant.loader import bind_hass 42 | from homeassistant.helpers.event import async_track_point_in_time 43 | from homeassistant.helpers.event import async_track_state_change 44 | from homeassistant.util import sanitize_filename 45 | 46 | import voluptuous as vol 47 | import homeassistant.components.alarm_control_panel as alarm 48 | import homeassistant.components.switch as switch 49 | import homeassistant.helpers.config_validation as cv 50 | 51 | _LOGGER = logging.getLogger(__name__) 52 | 53 | VERSION = '1.1.3' 54 | 55 | DOMAIN = 'alarm_control_panel' 56 | #//--------------------SUPPORTED STATES---------------------------- 57 | STATE_ALARM_WARNING = 'warning' 58 | STATE_ALARM_ARMED_PERIMETER = 'armed_perimeter' 59 | SUPPORTED_STATES = [STATE_ALARM_ARMED_AWAY, STATE_ALARM_ARMED_HOME, STATE_ALARM_DISARMED, STATE_ALARM_PENDING, 60 | STATE_ALARM_TRIGGERED, STATE_ALARM_WARNING, STATE_ALARM_ARMED_PERIMETER] 61 | 62 | SUPPORTED_PENDING_STATES = [STATE_ALARM_ARMED_AWAY, STATE_ALARM_ARMED_HOME, STATE_ALARM_ARMED_PERIMETER] 63 | 64 | #//-------------------STATES TO CHECK------------------------------ 65 | STATE_TRUE = 'true' 66 | STATE_UNLOCKED = 'unlocked' 67 | STATE_OPEN = 'open' 68 | STATE_DETECTED = 'detected' 69 | STATE_MOTION = 'motion' 70 | STATE_MOTION_DETECTED = 'motion_detected' 71 | STATE_MOTION_DETECTED2 = 'motion detected' 72 | 73 | STATE_FALSE = 'false' 74 | STATE_LOCKED = 'locked' 75 | STATE_CLOSED = 'closed' 76 | STATE_UNDETECTED = 'undetected' 77 | STATE_NO_MOTION = 'no_motion' 78 | STATE_STANDBY = 'standby' 79 | 80 | CONF_CUSTOM_SUPPORTED_STATUSES_ON = 'custom_supported_statuses_on' 81 | CONF_CUSTOM_SUPPORTED_STATUSES_OFF = 'custom_supported_statuses_off' 82 | 83 | SUPPORTED_STATUSES_ON = [STATE_ON, STATE_TRUE, STATE_UNLOCKED, STATE_OPEN, STATE_DETECTED, STATE_MOTION, STATE_MOTION_DETECTED, STATE_MOTION_DETECTED2] 84 | SUPPORTED_STATUSES_OFF = [STATE_OFF, STATE_FALSE, STATE_LOCKED, STATE_CLOSED, STATE_UNDETECTED, STATE_NO_MOTION, STATE_STANDBY] 85 | 86 | #//-----------------YAML CONFIG OPTIONS---------------------------- 87 | CONF_STATES = 'states' 88 | CONF_USERS = 'users' 89 | CONF_NAME = 'name' 90 | CONF_ID = 'id' 91 | CONF_PICTURE = 'picture' 92 | CONF_HOME_PERM = 'home_permision' 93 | CONF_AWAY_PERM = 'away_permission' 94 | CONF_PERI_PERM = 'perimiter_permission' 95 | CONF_ENABLED = 'enabled' 96 | CONF_CODE_TO_ARM = 'code_to_arm' 97 | CONF_PANIC_CODE = 'panic_code' 98 | CONF_PASSCODE_ATTEMPTS = 'passcode_attempts' 99 | CONF_PASSCODE_ATTEMPTS_TIMEOUT = 'passcode_attempts_timeout' 100 | CONF_WARNING_TIME = 'warning_time' 101 | 102 | #//-------------------SENSOR GROUPS-------------------------------- 103 | CONF_IMMEDIATE = 'immediate' 104 | CONF_DELAYED = 'delayed' 105 | CONF_IGNORE = 'homemodeignore' 106 | CONF_NOTATHOME = 'notathome' 107 | CONF_OVERRIDE = 'override' 108 | CONF_PERIMETER = 'perimeter' 109 | 110 | #//-----------------DEVICES TO ENABLE/DISBALE----------------------- 111 | CONF_ALARM = 'alarm' 112 | CONF_WARNING = 'warning' 113 | 114 | #//----------------------OPTIONAL MODES------------------------------ 115 | CONF_ENABLE_PERIMETER_MODE = 'enable_perimeter_mode' 116 | CONF_ENABLE_PERSISTENCE = 'enable_persistence' 117 | 118 | #//----------------------PANEL RELATED------------------------------ 119 | CONF_GUI = 'gui' 120 | CONF_PANEL = 'panel' 121 | CONF_ColorS = 'colors' 122 | CONF_THEMES = 'themes' 123 | CONF_ADMIN_PASSWORD = 'admin_password' 124 | CONF_DISABLE_ANIMATIONS = 'disable_animations' 125 | 126 | #//-----------------------ColorS------------------------------------ 127 | CONF_WARNING_Color = 'warning_color' 128 | CONF_PENDING_Color = 'pending_color' 129 | CONF_DISARMED_Color = 'disarmed_color' 130 | CONF_TRIGGERED_Color = 'triggered_color' 131 | CONF_ARMED_AWAY_Color = 'armed_away_color' 132 | CONF_ARMED_HOME_Color = 'armed_home_color' 133 | CONF_PERIMETER_Color = 'perimeter_color' 134 | 135 | #//-----------------------MQTT RELATED------------------------------- 136 | CONF_MQTT = 'mqtt' 137 | CONF_ENABLE_MQTT = 'enable_mqtt' 138 | CONF_OVERRIDE_CODE = 'override_code' 139 | CONF_PAYLOAD_DISARM = 'payload_disarm' 140 | CONF_PAYLOAD_ARM_HOME = 'payload_arm_home' 141 | CONF_PAYLOAD_ARM_AWAY = 'payload_arm_away' 142 | CONF_PAYLOAD_ARM_NIGHT = 'payload_arm_night' 143 | CONF_QOS = 'qos' 144 | CONF_STATE_TOPIC = 'state_topic' 145 | CONF_COMMAND_TOPIC = 'command_topic' 146 | CONF_PENDING_ON_WARNING = 'pending_on_warning' 147 | 148 | #//-----------------------LOG RELATED-------------------------------- 149 | CONF_ENABLE_LOG = 'enable_log' 150 | CONF_LOG_SIZE = 'log_size' 151 | CONF_LOGS = 'logs' 152 | 153 | #//-----------------------CAMERA RELATED-------------------------------- 154 | CONF_CAMERAS = 'cameras' 155 | 156 | #//-----------------------YAML RELATED-------------------------------- 157 | # 158 | CONF_YAML_ALLOW_EDIT = 'yaml_allow_edit' 159 | 160 | class Events(enum.Enum): 161 | ImmediateTrip = 1 162 | DelayedTrip = 2 163 | ArmHome = 3 164 | ArmAway = 4 165 | Timeout = 5 166 | Disarm = 6 167 | Trigger = 7 168 | ArmPerimeter = 8 169 | 170 | class LOG(enum.Enum): 171 | DISARMED = 0 #'disarmed the alarm' 172 | DISARM_FAIL = 1 #'Failed to disarm alarm' 173 | TRIGGERED = 2 #'alarm has been triggered!' 174 | HOME = 3 #'set the alarm in Home mode' 175 | AWAY = 4 #'set the alarm in Away mode' 176 | TRIPPED = 5 #'Alarm has been tripped by: ' 177 | LOCKED = 6 #'Panel Locked 178 | PERIMETER = 8 #'set the alarm in Perimeter mode' 179 | 180 | DEFAULT_PENDING_TIME = 0 #0 Seconds 181 | DEFAULT_WARNING_TIME = 0 #0 Seconds 182 | DEFAULT_TRIGGER_TIME = 600 #Ten Minutes 183 | 184 | def _state_validator(config): #Place a default value in that timers if there isnt specific ones set 185 | """Validate the state.""" 186 | config = copy.deepcopy(config) 187 | for state in SUPPORTED_PENDING_STATES: 188 | if CONF_TRIGGER_TIME not in config[state]: 189 | config[state][CONF_TRIGGER_TIME] = config[CONF_TRIGGER_TIME] 190 | if CONF_PENDING_TIME not in config[state]: 191 | config[state][CONF_PENDING_TIME] = DEFAULT_STATE_PENDING_TIME if state != STATE_ALARM_ARMED_AWAY else config[CONF_PENDING_TIME] 192 | if CONF_WARNING_TIME not in config[state]: 193 | config[state][CONF_WARNING_TIME] = DEFAULT_STATE_WARNING_TIME if state != STATE_ALARM_ARMED_AWAY else config[CONF_WARNING_TIME] 194 | return config 195 | 196 | def _state_schema(): 197 | """Validate the state.""" 198 | schema = {} 199 | # if state in SUPPORTED_PENDING_STATES: 200 | schema[vol.Optional(CONF_TRIGGER_TIME, default=DEFAULT_TRIGGER_TIME)] = vol.All(vol.Coerce(int), vol.Range(min=-1)) 201 | schema[vol.Optional(CONF_PENDING_TIME, default=DEFAULT_PENDING_TIME)] = vol.All(vol.Coerce(int), vol.Range(min=0)) 202 | schema[vol.Optional(CONF_WARNING_TIME, default=DEFAULT_WARNING_TIME)] = vol.All(vol.Coerce(int), vol.Range(min=0)) 203 | schema[vol.Optional(CONF_IMMEDIATE, default=[])] = cv.entity_ids # things that cause an immediate alarm 204 | schema[vol.Optional(CONF_DELAYED, default=[])] = cv.entity_ids # things that allow a delay before alarm 205 | schema[vol.Optional(CONF_OVERRIDE, default=[])] = cv.entity_ids # sensors that can be ignored if open when trying to set alarm 206 | return vol.Schema(schema) 207 | 208 | PANEL_SCHEMA = vol.Schema({ 209 | vol.Optional(CONF_CAMERAS): cv.entity_ids, 210 | vol.Optional(cv.slug): cv.string, 211 | }) 212 | 213 | USER_SCHEMA = vol.Schema([{ 214 | vol.Required(CONF_ID, default=uuid.uuid4().hex): cv.string, 215 | vol.Required(CONF_NAME): cv.string, 216 | vol.Optional(CONF_PICTURE, default='/local/images/ha.png'): cv.string, 217 | vol.Required(CONF_CODE): cv.string, 218 | vol.Optional(CONF_ENABLED, default=True): cv.boolean, 219 | vol.Optional(CONF_DISABLE_ANIMATIONS, default=False): cv.boolean 220 | # vol.Optional(CONF_HOME_PERM, default=True): cv.boolean, 221 | # vol.Optional(CONF_AWAY_PERM, default=True): cv.boolean, 222 | # vol.Optional(CONF_PERI_PERM, default=True): cv.boolean 223 | ##ADD TIME BASED SETTINGS 224 | }]) 225 | 226 | THEMES_SCHEMA = vol.Schema([{ 227 | vol.Optional(cv.slug): cv.string, 228 | }]) 229 | 230 | MQTT_SCHEMA = vol.Schema({ 231 | vol.Required(CONF_ENABLE_MQTT, default=False): cv.boolean, 232 | vol.Optional(CONF_QOS, default=0): vol.All(vol.Coerce(int), vol.Range(min=0)), 233 | vol.Optional(CONF_STATE_TOPIC, default='home/alarm'): cv.string, 234 | vol.Optional(CONF_COMMAND_TOPIC, default='home/alarm/set'): cv.string, 235 | vol.Optional(CONF_PAYLOAD_ARM_AWAY, default='ARM_AWAY'): cv.string, 236 | vol.Optional(CONF_PAYLOAD_ARM_HOME, default='ARM_HOME'): cv.string, 237 | vol.Optional(CONF_PAYLOAD_ARM_NIGHT, default='ARM_NIGHT'): cv.string, 238 | vol.Optional(CONF_PAYLOAD_DISARM, default='DISARM'): cv.string, 239 | vol.Optional(CONF_OVERRIDE_CODE, default=False): cv.boolean, 240 | vol.Optional(CONF_PENDING_ON_WARNING, default=False): cv.boolean, 241 | }) 242 | 243 | PLATFORM_SCHEMA = vol.Schema(vol.All({ 244 | vol.Required(CONF_PLATFORM): 'bwalarm', 245 | vol.Optional(CONF_NAME, default='House'): cv.string, 246 | vol.Optional(CONF_PENDING_TIME, default=25): vol.All(vol.Coerce(int), vol.Range(min=0)), 247 | vol.Optional(CONF_WARNING_TIME, default=25): vol.All(vol.Coerce(int), vol.Range(min=0)), 248 | vol.Optional(CONF_TRIGGER_TIME, default=DEFAULT_TRIGGER_TIME): vol.All(vol.Coerce(int), vol.Range(min=1)), 249 | vol.Optional(CONF_ALARM): cv.entity_id, # switch/group to turn on when alarming [TODO] 250 | vol.Optional(CONF_WARNING): cv.entity_id, # switch/group to turn on when warning [TODO] 251 | vol.Optional(CONF_CUSTOM_SUPPORTED_STATUSES_ON): vol.Schema([cv.string]), 252 | vol.Optional(CONF_CUSTOM_SUPPORTED_STATUSES_OFF): vol.Schema([cv.string]), 253 | vol.Optional(CONF_CODE): cv.string, 254 | vol.Optional(CONF_USERS): USER_SCHEMA, # Schema to hold the list of names with codes allowed to disarm the alarm 255 | vol.Optional(CONF_PANIC_CODE): cv.string, 256 | 257 | #------------------------------STATE RELATED------------------------- 258 | vol.Optional(CONF_STATES): vol.Schema({cv.slug: _state_schema()}), 259 | vol.Optional(STATE_ALARM_ARMED_AWAY, default={}): _state_schema(), #state specific times ###REMOVE### 260 | vol.Optional(STATE_ALARM_ARMED_HOME, default={}): _state_schema(), #state specific times ###REMOVE### 261 | vol.Optional(STATE_ALARM_ARMED_PERIMETER, default={}): _state_schema(), #state specific times ###REMOVE### 262 | # vol.Optional(STATE_ALARM_DISARMED, default={}): _state_schema(STATE_ALARM_DISARMED), #state specific times ###REMOVE### 263 | # vol.Optional(STATE_ALARM_TRIGGERED, default={}): _state_schema(STATE_ALARM_TRIGGERED), #state specific times ###REMOVE### 264 | 265 | #------------------------------GUI----------------------------------- 266 | vol.Optional(CONF_PANEL): PANEL_SCHEMA, 267 | vol.Optional(CONF_THEMES): THEMES_SCHEMA, 268 | 269 | #---------------------------OPTIONAL MODES--------------------------- 270 | vol.Optional(CONF_ENABLE_LOG, default=True): cv.boolean, 271 | vol.Optional(CONF_LOG_SIZE, default=10): vol.All(vol.Coerce(int), vol.Range(min=-1)), 272 | vol.Optional(CONF_LOGS): vol.Schema([cv.string]), 273 | #---------------------------LOG RELATED------------------------------ 274 | 275 | vol.Optional(CONF_ENABLE_PERIMETER_MODE, default=False): cv.boolean, # Enable perimeter mode? 276 | vol.Optional(CONF_ENABLE_PERSISTENCE, default=False): cv.boolean, # Enables persistence for alarm state 277 | vol.Optional(CONF_CODE_TO_ARM, default=False): cv.boolean, # Require code to arm alarm? 278 | 279 | #---------------------------PANEL RELATED--------------------------- 280 | vol.Optional(CONF_ADMIN_PASSWORD, default='HG28!!&dn'): cv.string, # Admin panel password 281 | 282 | #--------------------------PASSWORD ATTEMPTS-------------------------- 283 | vol.Optional(CONF_PASSCODE_ATTEMPTS, default=-1): vol.All(vol.Coerce(int), vol.Range(min=-1)), 284 | vol.Optional(CONF_PASSCODE_ATTEMPTS_TIMEOUT, default=900): vol.All(vol.Coerce(int), vol.Range(min=1)), 285 | 286 | #---------------------------MQTT RELATED------------------------------ 287 | vol.Required(CONF_MQTT, default={CONF_ENABLE_MQTT: False}): MQTT_SCHEMA, #vol.Any(MQTT_SCHEMA, None), #cv.boolean, # Allows MQTT functionality 288 | 289 | #---------------------------YAML RELATED---------------------------- 290 | vol.Optional(CONF_YAML_ALLOW_EDIT, default=True): cv.boolean, #Allow alarm.yaml to be edited 291 | #-----------------------------END------------------------------------ 292 | }, _state_validator)) 293 | 294 | SERVICE_YAML_SAVE = 'ALARM_YAML_SAVE' 295 | SERVICE_YAML_USER = 'ALARM_YAML_USER' 296 | 297 | CONF_CONFIGURATION = 'configuration' 298 | CONF_VALUE = 'value' 299 | 300 | CONF_USER = 'user' 301 | CONF_COMMAND = 'command' 302 | 303 | try: 304 | from ruamel.yaml import YAML 305 | except Exception as e: 306 | _LOGGER.warning('Import Error: %s. Attempting to download and import', e) 307 | 308 | @asyncio.coroutine 309 | def async_setup_platform(hass, config, async_add_devices, discovery_info=None): 310 | 311 | #Setup MQTT if enabled 312 | mqtt = None 313 | if (config[CONF_MQTT][CONF_ENABLE_MQTT]): 314 | import homeassistant.components.mqtt as mqtt 315 | 316 | alarm = BWAlarm(hass, config, mqtt) 317 | hass.bus.async_listen(EVENT_STATE_CHANGED, alarm.state_change_listener) 318 | hass.bus.async_listen(EVENT_TIME_CHANGED, alarm.time_change_listener) 319 | hass.bus.async_listen(EVENT_TIME_CHANGED, alarm.passcode_timeout_listener) 320 | async_add_devices([alarm]) 321 | 322 | @callback 323 | def alarm_yaml_save(service): 324 | alarm.settings_save(service.data.get(CONF_CONFIGURATION), service.data.get(CONF_VALUE)) 325 | 326 | @callback 327 | def alarm_yaml_user(service): 328 | alarm.settings_user(service.data.get(CONF_USER), service.data.get(CONF_COMMAND)) 329 | 330 | hass.services.async_register(DOMAIN, SERVICE_YAML_SAVE, alarm_yaml_save) 331 | hass.services.async_register(DOMAIN, SERVICE_YAML_USER, alarm_yaml_user) 332 | 333 | class BWAlarm(alarm.AlarmControlPanel): 334 | 335 | def __init__(self, hass, config, mqtt): 336 | #------------------------------Initalize the alarm system---------------------------------- 337 | self._config = config 338 | self._mqtt = mqtt 339 | self._hass = hass 340 | 341 | self.init_variables() 342 | 343 | self._updateUI = False 344 | 345 | def init_variables(self): 346 | #-------------------------------------STATE SPECIFIC-------------------------------------------------- 347 | self._supported_statuses_on = self._config.get(CONF_CUSTOM_SUPPORTED_STATUSES_ON, []) + SUPPORTED_STATUSES_ON 348 | self._supported_statuses_off = self._config.get(CONF_CUSTOM_SUPPORTED_STATUSES_OFF, []) + SUPPORTED_STATUSES_OFF 349 | 350 | self._state = STATE_ALARM_DISARMED 351 | self._returnto = STATE_ALARM_DISARMED 352 | self._armstate = STATE_ALARM_DISARMED 353 | 354 | self._allsensors = [] 355 | self._states = {} 356 | for state in self._config.get(CONF_STATES, {}): 357 | self._states[state] = self._config[CONF_STATES][state] 358 | self._allsensors = set(self._allsensors) | set(self._states[state]['immediate']) | set(self._states[state]['delayed']) | set(self._states[state]['override']) 359 | 360 | #-------------------------------------SENSORS-------------------------------------------------- 361 | self.immediate = None 362 | self.delayed = None 363 | self.override = None 364 | self._opensensors = None 365 | 366 | #------------------------------------CORE ALARM RELATED------------------------------------- 367 | self._enable_perimeter_mode = self._config[CONF_ENABLE_PERIMETER_MODE] 368 | self._panic_mode = 'deactivated' 369 | self._lasttrigger = "" 370 | self._timeoutat = None 371 | self._passcode_timeoutat = None 372 | 373 | #------------------------------------PASSCODE RELATED------------------------------------- 374 | self._code = self._config.get(CONF_CODE, None) 375 | self._users = self._config.get(CONF_USERS, []) 376 | self._panic_code = self._config.get(CONF_PANIC_CODE, None) 377 | self._panel_locked = False 378 | self._passcodeAttemptNo = 0 379 | self._passcode_attempt_allowed = self._config[CONF_PASSCODE_ATTEMPTS] 380 | self._passcode_attempt_timeout = self._config[CONF_PASSCODE_ATTEMPTS_TIMEOUT] 381 | 382 | #------------------------------------PANEL RELATED------------------------------------- 383 | self.changedbyuser = None 384 | 385 | #-------------------------------------MQTT-------------------------------------------------- 386 | # IF MQTT Enabled define its configuration 387 | if (self._config[CONF_MQTT][CONF_ENABLE_MQTT]): 388 | # # If MQTT enabled but is empty then set default values 389 | # if (self._config[CONF_MQTT] == None): self._config[CONF_MQTT] = {} 390 | 391 | self._qos = self._config[CONF_MQTT].get(CONF_QOS) 392 | self._state_topic = self._config[CONF_MQTT].get(CONF_STATE_TOPIC) 393 | self._command_topic = self._config[CONF_MQTT].get(CONF_COMMAND_TOPIC) 394 | self._payload_disarm = self._config[CONF_MQTT].get(CONF_PAYLOAD_DISARM) 395 | self._payload_arm_home = self._config[CONF_MQTT].get(CONF_PAYLOAD_ARM_HOME) 396 | self._payload_arm_away = self._config[CONF_MQTT].get(CONF_PAYLOAD_ARM_AWAY) 397 | self._payload_arm_night = self._config[CONF_MQTT].get(CONF_PAYLOAD_ARM_NIGHT) 398 | self._override_code = self._config[CONF_MQTT].get(CONF_OVERRIDE_CODE) 399 | self._pending_on_warning = self._config[CONF_MQTT].get(CONF_PENDING_ON_WARNING) 400 | 401 | #------------------------------------LOGGING-------------------------------------------------------- 402 | # IF logging Enabled define its configuration 403 | if (CONF_ENABLE_LOG in self._config): 404 | self._config[CONF_LOGS] = [] 405 | self._log_size = self._config.get(CONF_LOG_SIZE, 10) 406 | 407 | # Get the log file or create one if it doesnt exist 408 | log_path = self._hass.config.path() 409 | if not os.path.isdir(log_path): 410 | _LOGGER.error("[ALARM] Activity Log path %s does not exist.", log_path) 411 | else: 412 | self._log_final_path = os.path.join(log_path, "alarm_log.json") 413 | self.log_load() 414 | 415 | 416 | #------------------------------------YAML-------------------------------------------------------- 417 | # self._yaml_allow_edit = self._config[CONF_YAML_ALLOW_EDIT] 418 | # if (self._yaml_allow_edit): 419 | self._yaml_content = self.yaml_load() 420 | 421 | # Reset Alarm 422 | self.clearsignals() 423 | 424 | #------------------------------------PERSISTENCE---------------------------------------------------- 425 | self._persistence_list = json.loads('{}') 426 | if (self._config[CONF_ENABLE_PERSISTENCE]): 427 | persistence_path = self._hass.config.path() 428 | 429 | if not os.path.isdir(persistence_path): 430 | _LOGGER.error("[ALARM] Persistence path %s does not exist.", persistence_path) 431 | else: 432 | self._persistence_final_path = os.path.join(persistence_path, "alarm.json") 433 | if (self.persistence_load()): 434 | self._state = self._persistence_list["state"] 435 | self._timeoutat = pytz.UTC.localize(datetime.datetime.strptime(self._persistence_list["timeoutat"].split(".")[0].replace("T"," "), '%Y-%m-%d %H:%M:%S')) if self._persistence_list["timeoutat"] != None else None 436 | self._returnto = self._persistence_list["returnto"] 437 | self._armstate = self._persistence_list["armstate"] 438 | 439 | for self._armstate in SUPPORTED_PENDING_STATES: 440 | self._states = self._persistence_list["states"] 441 | self.immediate = self._states[self._state]["immediate"] 442 | self.delayed = self._states[self._state]["delayed"] 443 | self.override = self._states[self._state]["override"] 444 | 445 | if (self._armstate == STATE_ALARM_WARNING or self._armstate == STATE_ALARM_TRIGGERED or self._armstate == STATE_ALARM_PENDING): 446 | self._states = self._persistence_list["states"] 447 | self.immediate = self._states[self._returnto]["immediate"] 448 | self.delayed = self._states[self._returnto]["delayed"] 449 | self.override = self._states[self._returnto]["override"] 450 | 451 | # Alarm properties 452 | @property 453 | def should_poll(self) -> bool: return False 454 | @property 455 | def name(self) -> str: return self._config[CONF_NAME] 456 | @property 457 | def changed_by(self) -> str: return self._lasttrigger 458 | @property 459 | def state(self) -> str: return self._state 460 | @property 461 | def device_state_attributes(self): 462 | 463 | results = { 464 | 465 | 'immediate': self.immediate, 466 | 'delayed': self.delayed, 467 | 'ignored': self.ignored, 468 | 'allsensors': self._allsensors, 469 | 470 | 'code_to_arm': self._config[CONF_CODE_TO_ARM], 471 | 472 | 'panel_locked': self._panel_locked, 473 | 'passcode_attempts': self._passcode_attempt_allowed, 474 | 'passcode_attempts_timeout': self._passcode_attempt_timeout, 475 | 476 | 'changedbyuser': self.changedbyuser, 477 | 'panic_mode': self._panic_mode, 478 | 479 | 'arm_state': self._armstate, 480 | 481 | 'enable_perimeter_mode': self._config[CONF_ENABLE_PERIMETER_MODE], 482 | 'enable_persistence': self._config[CONF_ENABLE_PERSISTENCE], 483 | 484 | 'enable_log': self._config[CONF_ENABLE_LOG], 485 | 'log_size': self._config[CONF_LOG_SIZE], 486 | 487 | 'supported_statuses_on': self._supported_statuses_on, 488 | 'supported_statuses_off': self._supported_statuses_off, 489 | 490 | 'updateUI': self._updateUI, 491 | 492 | 'admin_password': hashlib.sha256(str.encode(self._config[CONF_ADMIN_PASSWORD])).hexdigest(), 493 | 494 | 'bwalarm_version': VERSION, 495 | 'py_version': sys.version_info, 496 | } 497 | 498 | if (CONF_USERS in self._config): 499 | users = copy.deepcopy(self._config[CONF_USERS]) 500 | for user in users: 501 | user['code'] = '****' 502 | results[CONF_USERS] = users 503 | 504 | if (CONF_PANEL in self._config): 505 | results[CONF_PANEL] = self._config[CONF_PANEL] 506 | 507 | if (CONF_THEMES in self._config): 508 | results[CONF_THEMES] = self._config[CONF_THEMES] 509 | 510 | if (CONF_LOGS in self._config): 511 | results[CONF_LOGS] = self._config[CONF_LOGS][-10:] 512 | 513 | if (CONF_MQTT in self._config): 514 | results[CONF_MQTT] = self._config[CONF_MQTT] 515 | 516 | if ('states' in self._config): 517 | results['states'] = self._config['states'] 518 | 519 | return results; 520 | 521 | def yaml_load(self): 522 | try: 523 | self.yaml = YAML() 524 | with open(self._hass.config.path() + "/alarm.yaml") as stream: 525 | try: 526 | return self.yaml.load(stream) 527 | except self.yaml.YAMLError as exc: 528 | print(exc) 529 | return None 530 | except Exception as e: 531 | _LOGGER.warning(e); 532 | 533 | 534 | def settings_save(self, configuration=None, value=None): 535 | """Push the alarm state to the given value.""" 536 | self._yaml_content = self.yaml_load() 537 | 538 | configuration = configuration.lower() 539 | 540 | self._config[configuration] = value 541 | self._yaml_content[configuration] = value 542 | 543 | _LOGGER.debug("Set the yaml entry %s to %s", configuration, value) 544 | 545 | self.settings_yaml_save() 546 | 547 | def settings_user(self, user=None, command=None): 548 | """Push the alarm state to the given value.""" 549 | self._yaml_content = self.yaml_load() 550 | 551 | x = 0 552 | 553 | if (command == 'add'): 554 | if user['id'] == None: 555 | user['id'] = uuid.uuid4().hex 556 | if ('users' not in self._config): 557 | self._config['users'] = [user] 558 | self._yaml_content['users'] = [user] 559 | else: 560 | _LOGGER.warning(user) 561 | self._config['users'].append(user) 562 | self._yaml_content['users'].append(user) 563 | elif (command == 'update'): 564 | for _user in self._config['users']: 565 | if _user['id'] == user['id']: 566 | self._config['users'][x] = user 567 | self._yaml_content['users'][x] = user 568 | x = x + 1 569 | elif (command == 'delete'): 570 | for _user in self._config['users']: 571 | if _user['id'] == user: 572 | self._config['users'].pop(x) 573 | self._yaml_content['users'].pop(x) 574 | x = x + 1 575 | elif (command == True or command == False): 576 | for _user in self._config['users']: 577 | if _user['id'] == user: 578 | self._config['users'][x]['enabled'] = command 579 | self._yaml_content['users'][x]['enabled'] = command 580 | x = x + 1 581 | 582 | self.settings_yaml_save() 583 | 584 | def settings_yaml_save(self): 585 | #Trigger a GUI update 586 | self._updateUI = not self._updateUI 587 | 588 | with open(self._hass.config.path() + "/alarm.yaml", 'w') as fil: 589 | self.yaml.dump(self._yaml_content, fil) 590 | 591 | self.init_variables() 592 | self.schedule_update_ha_state() 593 | 594 | 595 | ### LOAD persistence previously saved 596 | def persistence_load(self): 597 | try: 598 | if os.path.isfile(self._persistence_final_path): #Find the persistence JSON file and load. Once found update the alarm_control_panel object 599 | self._persistence_list = json.load(open(self._persistence_final_path, 'r')) 600 | return True 601 | else: #No persistence file found 602 | _LOGGER.warning("[ALARM] Persistence file doesnt exist") 603 | return False 604 | #self._persistence_list = json.loads('{"state":"disarmed", "timeoutat":null, "returnto":null, "immediate":[], "delayed":[], "override":[], "states":{}, "armstate":"disarmed"}') 605 | 606 | except Exception as e: 607 | _LOGGER.error("[ALARM] Persistence error occured loading: %s", str(e)) 608 | 609 | ### UPDATE persistence 610 | def persistence_save(self, persistence): 611 | if persistence is not None: #Check we have something to save [TODO] validate this is a persistence object 612 | self._persistence_list = persistence 613 | try: 614 | if self._persistence_list is not None: #Check we have genuine persistence to save if so dump to file 615 | with open(self._persistence_final_path, 'w') as fil: 616 | fil.write(json.dumps(self._persistence_list, ensure_ascii=False)) 617 | else: 618 | _LOGGER.error("[ALARM] No persistence to save!") 619 | except Exception as e: 620 | _LOGGER.error("[ALARM] Persistence Error occured saving: %s", str(e)) 621 | 622 | ### LOAD activity log previously saved 623 | def log_load(self): 624 | try: 625 | if os.path.isfile(self._log_final_path): #Find the log file and load. 626 | self._config[CONF_LOGS] = json.load(open(self._log_final_path, 'r')) 627 | else: #No log file found 628 | _LOGGER.warning("[ALARM] Activity log file doesnt exist") 629 | self._config[CONF_LOGS] = [] 630 | self.log_save() 631 | except Exception as e: 632 | _LOGGER.error("[ALARM] Error occured loading: %s", str(e)) 633 | 634 | ### UPDATE activity log 635 | def log_save(self): 636 | try: 637 | if self._config[CONF_LOGS] is not []: #Check we have genuine log to save if so dump to file 638 | with open(self._log_final_path, 'w') as fil: 639 | fil.write(json.dumps(self._config[CONF_LOGS], ensure_ascii=False)) 640 | else: 641 | _LOGGER.error("[ALARM] No log to save!") 642 | except Exception as e: 643 | _LOGGER.error("[ALARM] Log Error occured saving: %s", str(e)) 644 | 645 | 646 | ### Save alarm state 647 | def save_alarm_state(self): 648 | self._persistence_list["state"] = self._state 649 | self._persistence_list["timeoutat"] = self._timeoutat.isoformat() if self._timeoutat != None else None 650 | self._persistence_list["returnto"] = self._returnto 651 | self._persistence_list["states"] = self._states 652 | self._persistence_list["armstate"] = self._armstate 653 | self.persistence_save(self._persistence_list) 654 | 655 | ### Actions from the outside world that affect us, turn into enum events for internal processing 656 | def time_change_listener(self, eventignored): 657 | """ I just treat the time events as a periodic check, its simpler then (re-/un-)registration """ 658 | if self._timeoutat is not None: 659 | if now() > self._timeoutat: 660 | self._timeoutat = None 661 | self.process_event(Events.Timeout) 662 | 663 | def state_change_listener(self, event): 664 | """ Something changed, we only care about things turning on at this point """ 665 | if self._state != STATE_ALARM_DISARMED: 666 | new = event.data.get('new_state', None) 667 | if new is None: 668 | return 669 | 670 | if new.state != None: 671 | if new.state.lower() in self._supported_statuses_on: 672 | eid = event.data['entity_id'] 673 | if eid in self.immediate: 674 | self._lasttrigger = eid 675 | self.process_event(Events.ImmediateTrip) 676 | elif eid in self.delayed: 677 | self._lasttrigger = eid 678 | self.process_event(Events.DelayedTrip) 679 | 680 | # def check_open_sensors(self): 681 | # for sensor in self._allsensors: 682 | # if self._hass.states.get(sensor).state != None: 683 | # if self._hass.states.get(sensor).state in self._supported_statuses_on: 684 | # _LOGGER.error(self._hass.states.get(sensor)) # do summit 685 | 686 | @property 687 | def code_format(self): 688 | """One or more characters.""" 689 | return None if self._code is None else '.+' 690 | 691 | def alarm_disarm(self, code=None): 692 | #If the provided code matches the panic alarm then deactivate the alarm but set the state of the panic mode to active. 693 | if self._validate_panic_code(code): 694 | self.process_event(Events.Disarm) 695 | self._panic_mode = "ACTIVE" 696 | self._update_log(None, LOG.DISARMED, None) #Show a default disarm message incase this is displayed on the interface 697 | # Let HA know that something changed 698 | self.schedule_update_ha_state() 699 | return 700 | 701 | if not self._validate_code(code): 702 | self._update_log(None, LOG.DISARM_FAIL, None) 703 | return 704 | self.process_event(Events.Disarm) 705 | 706 | def alarm_arm(self, code, mode): 707 | user = 'HA' 708 | if code == "override": #ARM THE ALARM IMMEDIATELY 709 | self.process_event(mode, True) 710 | else: 711 | for entity in self._users: 712 | if entity['enabled'] == True: 713 | if entity['code'] == code: 714 | user = entity['id'] 715 | self.process_event(mode) 716 | 717 | if self._config[CONF_CODE_TO_ARM]: 718 | if code == self._code: 719 | self.process_event(mode) 720 | else: 721 | self.process_event(mode) 722 | 723 | self._update_log(user, mode, None) 724 | 725 | def alarm_arm_home(self, code=None): 726 | self.alarm_arm(code, Events.ArmHome) 727 | 728 | def alarm_arm_away(self, code=None): 729 | self.alarm_arm(code, Events.ArmAway) 730 | 731 | def alarm_arm_night(self, code=None): 732 | self.alarm_arm(code, Events.ArmPerimeter) 733 | 734 | def alarm_trigger(self, code=None): 735 | self.process_event(Events.Trigger) 736 | self._update_log(None, LOG.TRIGGERED, None) 737 | 738 | ### Internal processing 739 | def setsignals(self, alarmMode): 740 | """ Figure out what to sense and how """ 741 | self.immediate = self._states[alarmMode]['immediate'].copy() 742 | self.delayed = self._states[alarmMode]['delayed'].copy() 743 | self.override = self._states[alarmMode]['override'].copy() 744 | self.ignored = set(self._allsensors) - (set(self.immediate) | set(self.delayed)) 745 | 746 | def clearsignals(self): 747 | """ Clear all our signals, we aren't listening anymore """ 748 | self._panic_mode = "deactivated" 749 | self._armstate = STATE_ALARM_DISARMED 750 | self.immediate = set() 751 | self.delayed = set() 752 | self.ignored = self._allsensors.copy() 753 | self._timeoutat = None 754 | 755 | def process_event(self, event, override_pending_time=False): 756 | old_state = self._state 757 | 758 | #Update the state of the alarm panel 759 | if event == Events.Disarm: 760 | self._state = STATE_ALARM_DISARMED 761 | 762 | elif event == Events.Trigger: 763 | self._state = STATE_ALARM_TRIGGERED 764 | 765 | #If there is a pending time set in either of the state configs then drop into pending mode else simply arm the alarm 766 | elif old_state == STATE_ALARM_DISARMED: 767 | if event == Events.ArmHome: 768 | if (datetime.timedelta(seconds=int(self._states[STATE_ALARM_ARMED_HOME][CONF_PENDING_TIME])) and override_pending_time == False): 769 | self._state = STATE_ALARM_PENDING 770 | else: 771 | self._state = STATE_ALARM_ARMED_HOME 772 | self._armstate = STATE_ALARM_ARMED_HOME 773 | 774 | elif event == Events.ArmAway: 775 | if (datetime.timedelta(seconds=int(self._states[STATE_ALARM_ARMED_AWAY][CONF_PENDING_TIME])) and override_pending_time == False): 776 | self._armstate = STATE_ALARM_ARMED_AWAY 777 | self._state = STATE_ALARM_PENDING 778 | else: 779 | self._state = STATE_ALARM_ARMED_AWAY 780 | self._armstate = STATE_ALARM_ARMED_AWAY 781 | 782 | elif event == Events.ArmPerimeter: 783 | if (datetime.timedelta(seconds=int(self._states[STATE_ALARM_ARMED_PERIMETER][CONF_PENDING_TIME])) and override_pending_time == False): 784 | self._armstate = STATE_ALARM_ARMED_PERIMETER 785 | self._state = STATE_ALARM_PENDING 786 | else: 787 | self._state = STATE_ALARM_ARMED_PERIMETER 788 | self._armstate = STATE_ALARM_ARMED_PERIMETER 789 | 790 | elif old_state == STATE_ALARM_PENDING: 791 | if event == Events.Timeout: self._state = self._armstate 792 | 793 | elif old_state == STATE_ALARM_ARMED_HOME or \ 794 | old_state == STATE_ALARM_ARMED_AWAY or \ 795 | old_state == STATE_ALARM_ARMED_PERIMETER: 796 | if event == Events.ImmediateTrip: self._state = STATE_ALARM_TRIGGERED 797 | elif event == Events.DelayedTrip: self._state = STATE_ALARM_WARNING 798 | 799 | elif old_state == STATE_ALARM_WARNING: 800 | if event == Events.Timeout: self._state = STATE_ALARM_TRIGGERED 801 | 802 | elif old_state == STATE_ALARM_TRIGGERED: 803 | if event == Events.Timeout: self._state = self._returnto 804 | 805 | new_state = self._state 806 | if old_state != new_state: 807 | _LOGGER.debug("[ALARM] Alarm changing from {} to {}".format(old_state, new_state)) 808 | # Things to do on entering state 809 | if new_state == STATE_ALARM_WARNING: 810 | _LOGGER.debug("[ALARM] Turning on warning") 811 | if self._config.get(CONF_WARNING): 812 | self._hass.services.call(self._config.get(CONF_WARNING).split('.')[0], 'turn_on', {'entity_id':self._config.get(CONF_WARNING)}) 813 | self._timeoutat = now() + datetime.timedelta(seconds=int(self._states[self._armstate][CONF_WARNING_TIME])) 814 | self._update_log(None, LOG.TRIPPED, self._lasttrigger) 815 | elif new_state == STATE_ALARM_TRIGGERED: 816 | _LOGGER.debug("[ALARM] Turning on alarm") 817 | if self._config.get(CONF_ALARM): 818 | self._hass.services.call(self._config.get(CONF_ALARM).split('.')[0], 'turn_on', {'entity_id':self._config.get(CONF_ALARM)}) 819 | if (self._states[self._armstate][CONF_TRIGGER_TIME] == -1): 820 | self._timeoutat = now() + datetime.timedelta(hours=int(24)) 821 | else: 822 | self._timeoutat = now() + datetime.timedelta(seconds=int(self._states[self._armstate][CONF_TRIGGER_TIME])) 823 | self._update_log(None, LOG.TRIPPED, self._lasttrigger) 824 | elif new_state == STATE_ALARM_PENDING: 825 | _LOGGER.debug("[ALARM] Pending user leaving house") 826 | if self._config.get(CONF_WARNING): 827 | self._hass.services.call(self._config.get(CONF_WARNING).split('.')[0], 'turn_on', {'entity_id':self._config.get(CONF_WARNING)}) 828 | self._timeoutat = now() + datetime.timedelta(seconds=int(self._states[self._armstate][CONF_PENDING_TIME])) 829 | #self._returnto = STATE_ALARM_ARMED_AWAY 830 | self.setsignals(self._armstate) 831 | elif new_state == STATE_ALARM_ARMED_HOME: 832 | self._returnto = new_state 833 | self.setsignals(STATE_ALARM_ARMED_HOME) 834 | elif new_state == STATE_ALARM_ARMED_AWAY: 835 | self._returnto = new_state 836 | self.setsignals(STATE_ALARM_ARMED_AWAY) 837 | elif new_state == STATE_ALARM_ARMED_PERIMETER: 838 | self._returnto = new_state 839 | self.setsignals(STATE_ALARM_ARMED_PERIMETER) 840 | elif new_state == STATE_ALARM_DISARMED: 841 | self._returnto = new_state 842 | self.clearsignals() 843 | 844 | # Things to do on leaving state 845 | if old_state == STATE_ALARM_WARNING or old_state == STATE_ALARM_PENDING: 846 | _LOGGER.debug("[ALARM] Turning off warning") 847 | if self._config.get(CONF_WARNING): 848 | self._hass.services.call(self._config.get(CONF_WARNING).split('.')[0], 'turn_off', {'entity_id':self._config.get(CONF_WARNING)}) 849 | 850 | elif old_state == STATE_ALARM_TRIGGERED: 851 | _LOGGER.debug("[ALARM] Turning off alarm") 852 | if self._config.get(CONF_ALARM): 853 | self._hass.services.call(self._config.get(CONF_ALARM).split('.')[0], 'turn_off', {'entity_id':self._config.get(CONF_ALARM)}) 854 | 855 | # Let HA know that something changed 856 | if self._config[CONF_ENABLE_PERSISTENCE]: 857 | self.save_alarm_state() 858 | self.schedule_update_ha_state() 859 | 860 | def _validate_code(self, code): 861 | """Validate given code.""" 862 | if ((int(self._passcode_attempt_allowed) == -1) or (self._passcodeAttemptNo <= int(self._passcode_attempt_allowed))): 863 | check = self._code is None or code == self._code or self._validate_user_codes(code) 864 | if code == self._code: 865 | self._update_log(None, LOG.DISARMED, None) 866 | return self._validate_code_attempts(check) 867 | else: 868 | _LOGGER.warning("[ALARM] Too many passcode attempts, try again later") 869 | return False 870 | 871 | def _validate_user_codes(self, code): 872 | for entity in self._users: 873 | if entity['enabled'] == True: 874 | if entity['code'] == code: 875 | self._update_log(entity['id'], LOG.DISARMED, None) 876 | return True 877 | return False 878 | 879 | def _validate_code_attempts(self, check): 880 | if check: 881 | self._passcodeAttemptNo = 0 882 | else: 883 | _LOGGER.debug("[ALARM] Invalid code given") 884 | self._passcodeAttemptNo += 1 885 | if (int(self._passcode_attempt_allowed) != -1 and self._passcodeAttemptNo > int(self._passcode_attempt_allowed)): 886 | self._panel_locked = True 887 | self._passcode_timeoutat = now() + datetime.timedelta(seconds=int(self._passcode_attempt_timeout)) 888 | _LOGGER.warning("[ALARM] Panel locked, too many passcode attempts!") 889 | self._update_log(None, LOG.LOCKED, None) 890 | self.schedule_update_ha_state() 891 | return check 892 | 893 | def _validate_panic_code(self, code): 894 | """Validate given code.""" 895 | check = code == self._panic_code 896 | if check: 897 | _LOGGER.warning("[ALARM] PANIC MODE ACTIVATED!!!") 898 | self._passcodeAttemptNo = 0 899 | return check 900 | 901 | def _update_log(self, id, message, entity_id): 902 | if (id == None or id == ''): 903 | id = 'HA' 904 | self.changedbyuser = id 905 | if (CONF_ENABLE_LOG in self._config): 906 | self._log_size = int(self._config[CONF_LOG_SIZE]) if CONF_LOG_SIZE in self._config else 10 907 | if self._log_size != -1 and len(self._config[CONF_LOGS]) >= self._log_size: 908 | self._config[CONF_LOGS].remove(self._config[CONF_LOGS][0]) 909 | self._config[CONF_LOGS].append([time.time(), id, message.value, entity_id]) 910 | self.log_save() 911 | 912 | ### Actions from the outside world that affect us, turn into enum events for internal processing 913 | def passcode_timeout_listener(self, eventignored): 914 | if self._passcode_timeoutat is not None: 915 | if now() > self._passcode_timeoutat: 916 | self._panel_locked = False 917 | self._passcode_timeoutat = None 918 | self._passcodeAttemptNo = 0 919 | self.schedule_update_ha_state() 920 | 921 | @asyncio.coroutine 922 | def async_added_to_hass(self): 923 | """Subscribe mqtt events. 924 | This method must be run in the event loop and returns a coroutine. 925 | """ 926 | if (self._config[CONF_MQTT][CONF_ENABLE_MQTT]): 927 | async_track_state_change( 928 | self._hass, self.entity_id, self._async_state_changed_listener 929 | ) 930 | 931 | @callback 932 | def message_received(topic, payload, qos): 933 | """Run when new MQTT message has been received.""" 934 | #_LOGGER.warning("[ALARM] MQTT Topic: %s Payload: %s", topic, payload) 935 | if payload.split(" ")[0] == self._payload_disarm: 936 | #_LOGGER.warning("Disarming %s", payload) 937 | #TODO self._hass.states.get('binary_sensor.siren_sensor') #Use this method to relay open states 938 | if (self._override_code): 939 | self.alarm_disarm(self._code) 940 | else: 941 | self.alarm_disarm(payload.split(" ")[1]) 942 | elif payload == self._payload_arm_home: 943 | self.alarm_arm_home('') 944 | elif payload == self._payload_arm_away: 945 | self.alarm_arm_away('') 946 | elif payload == self._payload_arm_night: 947 | self.alarm_arm_night('') 948 | else: 949 | _LOGGER.warning("[ALARM/MQTT] Received unexpected payload: %s", payload) 950 | return 951 | if (self._config[CONF_MQTT][CONF_ENABLE_MQTT]): 952 | return self._mqtt.async_subscribe( 953 | self._hass, self._command_topic, message_received, self._qos) 954 | 955 | @asyncio.coroutine 956 | def _async_state_changed_listener(self, entity_id, old_state, new_state): 957 | """Publish state change to MQTT.""" 958 | if (self._config[CONF_MQTT][CONF_ENABLE_MQTT]): 959 | state = new_state.state 960 | if (self._pending_on_warning == True and state == STATE_ALARM_WARNING): 961 | state = STATE_ALARM_PENDING 962 | 963 | self._mqtt.async_publish(self._hass, self._state_topic, state, self._qos, True) 964 | 965 | _LOGGER.debug("[ALARM/MQTT] State changed") 966 | -------------------------------------------------------------------------------- /example automations/panic_mode.yaml: -------------------------------------------------------------------------------- 1 | - alias: '[Alarm] Panic Mode' 2 | trigger: 3 | platform: state 4 | entity_id: alarm_control_panel.house 5 | value_template: '{{ state.attributes.panic_mode }}' 6 | to: 'ACTIVE' 7 | action: 8 | service: activate_self_defence_robot -------------------------------------------------------------------------------- /guidance/configuration.md: -------------------------------------------------------------------------------- 1 | ## Installation Guide 2 | To get this running add the files (alarm.yaml, panels/alarm.html, custom_components/alarm_control_panel/bwalarm.py, www/lib/countdown360.js, www/lib/jquery-3.2.1.min.js, www/alarm/alarm.css) from this repo into your home assistant configuration directory, then add the following to your configuration.yaml file: 3 | 4 | **NOTE:** If you already have a panel_custom.yaml for say floorplan then just copy and paste the code from this repo file into your own panel_custom.yaml to prevent floorplan from being overritten. 5 | **NOTE:** Same goes for Automations.yaml. Append the samples inside of this file into your own automations.yaml 6 | 7 | ``` 8 | alarm_control_panel: !include alarm.yaml 9 | panel_custom: !include panel_custom.yaml 10 | ``` 11 | 12 | **NOTE:** If you experience issues with the page not displaying then add the following: 13 | ``` 14 | #CONFIGURATION.YAML 15 | frontend: 16 | javascript_version: latest 17 | ``` 18 | ## Configuration variables: 19 | 20 | **Alarm.yaml configuration settings:** 21 | 22 | - platform: bwalarm **#[REQUIRED, String] Name of the custom alarm component. Do not change** 23 | - name: House **#[REQUIRED, String]This can be changed to whatever suits your need, ensure this attribute matches the one in your panel_custom.yaml 'alarmid: alarm_control_panel.house'** 24 | 25 | - code: '9876' **#[REQUIRED, digits] should consist of one or more digits ie '6482' ensure your passcode is encapsulated by quotes** 26 | - panic_code: '1234' **#[OPTIONAL, digits] Panic Code should consist of one or more digits ie '1234' ensure your passcode is encapsulated by quotes, it needs to be different to your standard alarm code. This enables a special panic mode. This can be used under duress to deactivate the alarm which would appear to the unseeing eye as deactivated however a special attribute [panic_mode] listed under the alarm_control_panel.[identifier] will change to ACTIVE. This status could be used in your automations to send a notification to someone else police/spouse/sibling/neighbour that you are under duress. To deactive this mode arm then disarm your alarm in the usual manner.** 27 | 28 | - alarm: automation.alarm_triggered **#[REQUIRED, String] The automation to fire when the alarm has been triggered** 29 | - warning: automation.alarm_warning **#[OPTIONAL, String] The automation to fire when the alarm has been tripped** 30 | #### [OPTIONAL SETTINGS] 31 | - clock: True **#[OPTIONAL, Boolean] False by default. True enables a clock in the center of the status bar** 32 | - perimeter_mode: True **#[OPTIONAL, Boolean] False by default. True enables perimeter mode, this could be known as 'Day Mode' i.e. only arm the doors whilst there is someone using all floors** 33 | - weather: True **#[OPTIONAL, Boolean] False by Default. Allows a weather summary to be displayed on the status bar. Dark Sky weather component must be enabled with the name sensor.dark_sky_summary** 34 | - persistence: False **#[OPTIONAL, Boolean] False by Default. Allows this custom component to save the state of the alarm to file then reinstate it in the event of power loss.** 35 | - hide_passcode: True **#[OPTIONAL, Boolean] True by default. This is a security feature when enabled hides the passcode while entering disarm code.** 36 | - hide_sidebar: True **#[OPTIONAL, Boolean] False by default. This is a security feature when enabled hides the HA sidebar when the alarm is armed. The sidebar re-appears when the alarm is disarmed.** 37 | - hide_sensor_groups: True **#[OPTIONAL, Boolean] - False by default. Setting this to True hides sensor groups (all sensors, immediate sensors, delayed sensors, inactive sensors) from the display. Open sensors will still appear** 38 | -hide_custom_panel: True **#[OPTIONAL, Boolean] - True by default. Setting this to False enables a custom panel below the sensors groups which allows you to add your own html code. Use this to bring any other features you would like to see for example displaying live camera feeds, a rotating image gallery, custom HA buttons and sensors. To use this enable the custom panel in alarm.yaml (custom_panel: True) then ensure you take a copy of custom-element.html and add it to you www/alarm/ folder. Edit the html code between the template tags. I'm have added a custom sample folder where I will upload examples of 'things' which can be added here. Please contribute!!!** 39 | 40 | ## Timings 41 | - **pending_time:** 25 #[OPTIONAL, Number, default 25] Grace time in seconds to allow for exit and entry using Away mode. 42 | - **trigger_time:** 600 #[OPTIONAL, Number, default 600] The time in seconds of the trigger time in which the alarm is firing. before returning previous set alarm state. 43 | 44 | ### [STATES] 45 | - **armed_perimeter:** #[OPTIONAL] 46 | **pending_time:** 10 #[OPTIONAL] State specific setting if not defined inherits from above top level time 47 | **trigger_time:** 300 #[OPTIONAL] State specific setting if not defined inherits from above top level time 48 | #[OPTIONAL however either an immediate or delayed group must exist] Sensors in this group tigger the alarm immediately 49 | immediate: 50 | - binary_sensor.your_sensors 51 | #[OPTIONAL] Sensors in this group start the clock (pending_time) when tripped before the alarm is triggered 52 | delayed: 53 | - binary_sensor.your_sensors 54 | #[OPTIONAL] Use this group to automatically override the warning message on open sensors when arming. (I use this as I have a motion sensor at the front door) 55 | override: 56 | - binary_sensor.your_sensor 57 | 58 | - **armed_home:** #[REQUIRED] 59 | **pending_time:** 10 #[OPTIONAL] State specific setting if not defined inherits from above top level time 60 | **trigger_time:** 300 #[OPTIONAL] State specific setting if not defined inherits from above top level time 61 | #[OPTIONAL however either an immediate or delayed group must exist] Sensors in this group tigger the alarm immediately 62 | immediate: 63 | - binary_sensor.your_sensors 64 | #[OPTIONAL] Sensors in this group start the clock (pending_time) when tripped before the alarm is triggered 65 | delayed: 66 | - binary_sensor.your_sensors 67 | #[OPTIONAL] Use this group to automatically override the warning message on open sensors when arming. (I use this as I have a motion sensor at the front door) 68 | override: 69 | - binary_sensor.your_sensor 70 | 71 | - **armed_away:** #[REQUIRED] 72 | **pending_time:** 25 #[OPTIONAL] State specific setting if not defined inherits from above top level time 73 | **trigger_time:** 600 #[OPTIONAL] State specific setting if not defined inherits from above top level time 74 | #[OPTIONAL however either an immediate or delayed group must exist] Sensors in this group tigger the alarm immediately 75 | immediate: 76 | - binary_sensor.your_sensors 77 | #[OPTIONAL] Sensors in this group start the clock (pending_time) when tripped before the alarm is triggered 78 | delayed: 79 | - binary_sensor.your_sensors 80 | #[OPTIONAL] Use this group to automatically override the warning message on open sensors when arming. (I use this as I have a motion sensor at the front door) 81 | override: 82 | - binary_sensor.your_sensor 83 | 84 | ### [PASSCODE RELATED] 85 | - passcode_attempts: 3 #[OPTIONAL, number] Disabled if commented out. When a value equal or greater than 0 is set, the system will only allow the set amount of password attempts before timing out 86 | - passcode_attempts_timeout: 30 #[OPTIONAL, number] Default 30 seconds. When set with the password attempts option the panel will timeout for the amount of seconds set if the password is entered incorrectly as per the password_attempts option. The system will then reset the allowed password attempts 87 | 88 | ### [MQTT RELATED] 89 | - mqtt: True #[OPTIONAL, boolean] False by default. Settings this to True will enable MQTT Mode. Uncomment options below to use See the README for guidance. 90 | - override_code: True #[OPTIONAL, boolean] False by default. if true allows MQTT commands to disarm the alarm without a valid code. 91 | - state_topic: 'home/alarm' #[OPTIONAL, string] The MQTT topic HA will publish state updates to. 92 | - command_topic: 'home/alarm/set' #[OPTIONAL, string] The MQTT topic HA will subscribe to, to receive commands from a remote device to change the alarm state. 93 | - qos: 0 #[OPTIONAL, number] The maximum QoS level for subscribing and publishing to MQTT messages. Default is 0. 94 | - payload_disarm: "DISARM" #[OPTIONAL, string] The payload to disarm this Alarm Panel. Default is “DISARM”. 95 | - payload_arm_home: "ARM_HOME" #[OPTIONAL, string] The payload to set armed-home mode on this Alarm Panel. Default is “ARM_HOME”. 96 | - payload_arm_away: "ARM_AWAY" #[OPTIONAL, string] The payload to set armed-away mode on this Alarm Panel. Default is “ARM_AWAY”. 97 | - payload_arm_night: "ARM_NIGHT" #[OPTIONAL, string] The payload to set armed-night mode on this Alarm Panel. Default is “ARM_NIGHT”. 98 | 99 | ### [COLOURS] Use any HTML format 100 | - warning_colour: 'orange' #[OPTIONAL, string] 101 | - pending_colour: 'orange' #[OPTIONAL, string] 102 | - disarmed_colour: '#03A9F4' #[OPTIONAL, string] 103 | - armed_home_colour: 'black' #[OPTIONAL, string] 104 | - armed_away_colour: 'black' #[OPTIONAL, string] 105 | - triggered_colour: 'red' #[OPTIONAL, string] 106 | 107 | ### [CUSTOM STATUSES] 108 | -custom_supported_statuses_on: #[OPTIONAL, list of strings] CUSTOM SENSOR STATUSES - These settings allow devices which are not natively supported by this panel to be used. This is to be used when the state of the device is not recognised by the panel. Examples are provided below 109 | - 'running' #EXAMPLE 110 | -custom_supported_statuses_off: 111 | - 'not_running' #EXAMPLE -------------------------------------------------------------------------------- /historic_changelog.md: -------------------------------------------------------------------------------- 1 | - (07/03/18) NEW Feature - Pending and Trigger time can be determined per state. For example arming the alarm in away mode can be set to 60 seconds pending time, yet arming in home mode can be set to 30 seconds. This can also be overriden as per the feature below which has been expanded to include home/perimiter. [See guidance here](guidance/configuration.md#timings) 2 | 3 | - (23/02/18) NEW FEATURE - Override the pending time when arming away mode so that the alarm arms instantly. To do this pass a code parameter of '-1' to the service, this could be in the form of an automation such as: 4 | ``` 5 | alias: '[Alarm] Instantly Arm Away Mode' 6 | trigger: 7 | ... place your trigger here ... 8 | action: 9 | - service: alarm_control_panel.alarm_arm_away 10 | entity_id: alarm_control_panel.house 11 | data: 12 | code: '-1' 13 | ``` 14 | 15 | - (16/02/18) NEW FEATURE - PERSISTENCE!!!!!!!! Enabling persistence in the alarm.yaml file allows this component to save the alarm state everytime it changes. This means if there is a power outage or server crash/failure then upon restart HA will auto load the previous alarm state. This has not been tested in windows but in theory should work. Ensure HA has permission to write to the config folder as this creates a new file named alarm.json. 16 | - (16/02/18) NEW FEATURE - Custom Panel allowing custom html/polymer code!!!!!! Use this to bring any other features you would like to see for example displaying live camera feeds, a rotating image gallery, custom HA buttons and sensors. To use this enable the custom panel in alarm.yaml (custom_panel: True) then ensure you take a copy of custom-element.html and add it to you www/alarm/ folder. Edit the html code between the template tags. I'm have added a custom sample folder where I will upload examples of 'things' which can be added here. Please contribute!!! 17 | - (16/02/18) NEW FEATURE - Added the ability to hide the sensor groups (all sensors, immediate sensors, delayed sensors, inactive sensors) from the display. Open sensors will still appear on the display. (hide_sensor_groups:True) see alarm.yaml for details. 18 | - (16/02/18) NEW FEATURE - Added the ability to mask the passcode during diasarm. The feature will be enabled by deault. See the alarm.yaml for extra information. Credit @mikefero 19 | 20 | - (16/02/18) BUG FIX - Added further code to force the sidebar to hide when the alarm is armed. This prevents a simple refresh showing the sidebar. 21 | - (16/02/18) BUG FIX - Given the time label has caused folk issues I have decided to drop the javascript implementation and use the time derived from HA. Ensure you have a time sensor setup in your sensors.yaml: 22 | ``` 23 | - platform: time_date 24 | display_options: 25 | - 'time' 26 | ``` 27 | - (15/02/18) BUG FIX - Fixed the code disarm issue. 28 | 29 | - (19/01/18) NEW FEATURE - MQTT now allows you to disarm your alarm using the your code. MQTT panels will need to support the format of the payload which is 'DISARM CODE' for example 'DISARM 0000'. To override this so that MQTT can disarm the alarm without passing across the code then set override_code: True in the alarm.yaml. Status feedback to MQTT coming soon... 30 | - (19/01/18) NEW FEATURE - The panel now allows you to hide the sidebar when the alarm is activated preventing in intruder to simply go to configuration and shut down HA. A suitable locked down browser will also be required to prevent the intruder simply changing the URL. You could check out kiosk on android. To activate this feature simply enable hide_sidebar: True in the alarm.yaml NOTE!! Ensure you copy alarm_scripts.js into the appropriate folder 'www/alarm'. This was a tricky feature to implement and future HA updates may break this. If anyone has a better idea on how to code this then be my guest. 31 | 32 | - (19/01/18) ADDITIONAL STATE - Added 'motion_detected' as a supported state 33 | 34 | - (15/01/18) NOTE!!!!!!! - There are a lot of changes, update all files to ensure everything works. (alarm.yaml, panels/alarm.html, custom_components/alarm_control_panel/bwalarm.py, www/lib/countdown360.js, www/lib/jquery-3.2.1.min.js, www/alarm/alarm.css) Also don't forget to clear your browser cache. Raise any issues you come across 35 | 36 | - (15/01/18) NEW FEATURE - Set a maximum number of Passcode attempts with a lockout period. See the alarm.yaml for setup. 37 | - (15/01/18) NEW FEATURE - There is now support for custom and unknown device states. Add your new on/off states into the alarm.yaml file and this component will track them! 38 | 39 | - (15/01/18) NEW GUI - The code panel slides in or out depending on the mode. Let me know what you think. 40 | 41 | - (15/01/18) BUG FIX - Weather alignment on different modes and devices. 42 | - (15/01/18) BUG FIX - Timer mode now appears in warning mode so you know how long before the alarm triggers. 43 | - (15/01/18) BUG FIX - MQTT now allows custom state/command topics via the alarm.yaml. 44 | - (15/01/18) BUG FIX - Re-aligned buttons for smaller devices. 45 | 46 | - (15/01/18) OTHER - Work progressing to clean up the python code. 47 | - (15/01/18) OTHER - CSS is now split from the main http file for readability, still requires a cleanse. 48 | 49 | - (15/01/18) COMING SOON - MQTT Support to valid passcodes. 50 | - (15/01/18) COMING SOON - MQTT Support to provide extended feedback to other alarm interfaces i.e. open sensor arrays, lockout feedback. 51 | - (15/01/18) COMING SOON - Persistance support for rebooted systems. 52 | - (15/01/18) COMING SOON - Disabling of the sidebar when armed. Trickier than I thought due to the use of polymer and its sandboxing. 53 | - (15/01/18) COMING SOON - User Guide/Help pages 54 | - (15/01/18) COMING SOON - Customizable settings such as gui colours, collapsable sensor groups 55 | - (15/01/18) COMING SOON - Screensaver mode 56 | - (15/01/18) COMING SOON - Granular times on home/arm mode 57 | - (15/01/18) COMING SOON - Screenshots + Videos 58 | 59 | - (08/01/17) NEW FEATURE - MQTT Integration. Enable this by setting mqtt to True in the yaml. See alarm.yaml for optional settings. This is based on the [manual mqtt code](https://home-assistant.io/components/alarm_control_panel.manual_mqtt/). [MQTT Needs to be enabled in your HA setup and configured appropriately](https://home-assistant.io/docs/mqtt/broker/#embedded-broker) then you should be able to use [custom panels such as this](https://play.google.com/store/apps/details?id=com.thanksmister.iot.mqtt.alarmpanel&hl=en) 60 | 61 | - (28/12/17) Added a new feature 'Panic Mode' this allows you to set a panic code in the alarm.yaml. When using this code to deactivate the alarm, the alarm is deactivated however a special attribute panic_mode is set to ACTIVE. Use this backed with your automations to trigger custom messages to those who can assist. 62 | - (28/12/17) Added support for override sensors. When sensors are placed in this group any which are open when activing the alarm are ignored. 63 | 64 | - (27/12/17) Added support for devices with open/closed, true/false, locked/unlocked, detected/undetected statuses. There are some heavy changes on the code in readiness for a settings page and an optional screensaver. 65 | 66 | - (19/11/17) Added optional perimeter mode (activates a 'perimeter' group only) which could also ne known as 'Home Day' mode. Added weather sensor into status bar (You must have dark sky weather component enabled specifically sensor.dark_sky_summary), added 0 to code panel. 67 | 68 | - (13/11/17) Added sample automation.yaml. Fixed GUI issues with groups. Outlined base code for 'Perimeter mode' 69 | 70 | - (12/11/17) You can now use either homemodeignore or notathome group title for sensors that need to be ignored during home mode 71 | - (12/11/17) Added a check (displays open sensors in highlighted group) for open sensors when setting alarm (changes button text to override alarm). **NOTE** override in alarm.yaml isn't quite ready yet and you will still need to manually override via the button in the interface for now. -------------------------------------------------------------------------------- /panel_custom.yaml: -------------------------------------------------------------------------------- 1 | - name: alarm 2 | sidebar_title: Alarm 3 | sidebar_icon: mdi:security-home 4 | config: 5 | alarmid: alarm_control_panel.house 6 | -------------------------------------------------------------------------------- /www/alarm/alarm.css: -------------------------------------------------------------------------------- 1 | /* CUSTOM ALARM COMPONENT PANEL CSS 2 | https://github.com/gazoscalvertos/Hass-Custom-Alarm 3 | VERSION: 1.1.4 4 | MODIFIED: 13/11/18 5 | */ 6 | :host { 7 | --countdown-timer-display: none; 8 | --time-display: initial; 9 | 10 | /* THEME - DARK */ 11 | --dark-background: #191A1F; 12 | --dark-inner: #212227; 13 | 14 | --dark-green: #23C48E; 15 | --dark-black: #1A1E22; 16 | 17 | --dark-grey: #7F7F7F; 18 | --dark-grey-text: #BCBCBE; 19 | 20 | --dark-white-text: #FFFFFF; 21 | --dark-greyout-text: #77787A; 22 | 23 | --dark-red: #D3435C; 24 | --dark-light-blue: #04C0F8; 25 | --dark-orange: #ED9D00; 26 | 27 | --dark-indigo: #5771C2; 28 | --dark-indigo-grey: #37446B; 29 | 30 | /* THEME - LIGHT */ 31 | --light-background: white; 32 | --light-inner: #f3f3f3; 33 | 34 | --light-green: #23C48E; 35 | --light-black: #1A1E22; 36 | 37 | --light-grey: #7F7F7F; 38 | --light-grey-text: #BCBCBE; 39 | 40 | --light-white-text: #FFFFFF; 41 | --light-greyout-text: #77787A; 42 | 43 | --light-red: #D3435C; 44 | --light-light-blue: #04C0F8; 45 | --light-orange: #ED9D00; 46 | 47 | --light-indigo: #5771C2; 48 | --light-indigo-grey: #37446B; 49 | 50 | --button-shape: 0%; 51 | 52 | /* GUI SPECIFIC */ 53 | --primary-text-color: 'white'; 54 | 55 | --panel-background-color: var(--dark-background); /*The background color of the main content section.*/ 56 | 57 | /* --panel-outer-background-color: var(--secondary-background-color); */ 58 | --panel-outer-background-color: var(--dark-inner); /*The background color of both the status bar and the menu bar*/ 59 | --panel-text-color: var(--dark-white-text); /*The color of the general text within the panel.*/ 60 | 61 | --header-background-color: var(--dark-background); /*The background color of very top header bar.*/ 62 | --header-text-color: var(--dark-white-text); /*The text color on very top header bar.*/ 63 | 64 | --alarmstatus-text-color: var(--dark-green); /*The text color of the Alarm Status*/ 65 | --time-text-color: var(--dark-white-text); /*The text color of the Time label*/ 66 | --weather-text-color: var(--dark-white-text); /*The text color of the Weather label.*/ 67 | --weather-image-color: var(--dark-light-blue); /*The color of the Weather image*/ 68 | 69 | --info-header-text-color: white; /* The color of the Heading within a particular Section*/ 70 | --info-detail-text-color: var(--dark-green); /*The color of the descriptive text within a particular Section.*/ 71 | 72 | --title-color: var(--dark-orange); /*The color of the Title text within a particular section.*/ 73 | --subtitle-text-color: var(--dark-light-blue); /*The color of the Subtitle text within a particular section.*/ 74 | --openSensors-title-color: var(--dark-orange); /*The color of the Open Sensors Title to draw attention to the open sensor*/ 75 | --button-background-color: var(--dark-background); /*The background color of the alarm buttons.*/ 76 | --cancel-color: var(--dark-red); /*The background color of the cancel button.*/ 77 | --override-color: var(--dark-orange); /* The background color of the override button.*/ 78 | --action-button-border-color: var(--dark-green);/* The border color of the action button.*/ 79 | 80 | --info-panel-buttons-color: var(--dark-light-blue); /*The color of the menu buttons*/ 81 | 82 | --arm-button-border-color: var(--dark-light-blue);/* The border color of the alarm buttons.*/ 83 | --arm-button-text-color: white;/* The color of the text within the alarm buttons.*/ 84 | 85 | --paper-listbox-background-color: blue;/* The background color of the listboxes.*/ 86 | --paper-listbox-color: var(--dark-light-blue);/* The text color within the listboxes.*/ 87 | 88 | --paper-item-selected_-_color: var(--dark-orange);/* The text color of the item selected within a selection box.*/ 89 | --paper-input-container-shared-input-style_-_color: white; 90 | 91 | --font-header: unset; /*Font of the header, time and weather text*/ 92 | 93 | --shadow-effect: below 0 linear-gradient(rgba(255, 255, 255, 0), rgba(255, 255, 255, .1)); 94 | 95 | } 96 | app-toolbar { 97 | background-color: var(--header-background-color); 98 | color: var(--header-text-color); 99 | } 100 | #layout { 101 | background-color: var(--panel-outer-background-color); 102 | display: flex; 103 | flex-direction: column; 104 | height: 100%; 105 | min-height: 100%; 106 | width: 100%; 107 | overflow: hidden; 108 | } 109 | #main-title { 110 | text-align: center !important; 111 | position: relative !important; 112 | display: block !important; 113 | background-color: var(--header-background-color); 114 | padding: 10px; 115 | } 116 | ha-menu-button + [main-title] { 117 | margin-left: -50px; 118 | } 119 | ha-menu-button { 120 | color: var(--header-text-color); 121 | float: left; 122 | } 123 | a, a:link, a:visited, a:active { 124 | color: var(--panel-text-color); 125 | text-decoration: none; 126 | } 127 | 128 | a:hover { 129 | text-decoration: underline; 130 | color: var(--panel-text-color); 131 | } 132 | 133 | .list-item { 134 | cursor: pointer; 135 | font-size: large; 136 | } 137 | .checkbox { 138 | cursor: pointer; 139 | padding: 5px; 140 | font-size: large; 141 | } 142 | 143 | .detail, .show, .hide:target { 144 | display: none; 145 | } 146 | .hide:target + .show, 147 | .hide:target ~ .detail { 148 | display: block; 149 | } 150 | 151 | #modalError { 152 | position: absolute; 153 | z-index: 1; 154 | margin-top: 25vh; 155 | } 156 | 157 | #error-icon { 158 | width: 60%; 159 | text-align: center; 160 | margin: auto; 161 | } 162 | #error-icon iron-icon{ 163 | height: 100px; 164 | width: 100px; 165 | color: var(--title-color); 166 | } 167 | #error-title { 168 | font-size: 40pt; 169 | line-height: 40pt; 170 | color: var(--subtitle-text-color); 171 | margin-bottom: 10px; 172 | } 173 | #error-header { 174 | font-size: x-large; 175 | line-height: 20pt; 176 | color: var(--title-color); 177 | margin-bottom: 10px; 178 | } 179 | #error-detail { 180 | font-size: large; 181 | line-height: 15pt; 182 | color: var(--info-detail-text-color); 183 | margin-bottom: 10px; 184 | } 185 | 186 | 187 | .iron-selected:before{ 188 | content: '✔'; /* Insert content that looks like bullets */ 189 | padding-right: 8px; 190 | color: var(--dark-orange); /* Or a color you prefer */ 191 | } 192 | 193 | #restart{ 194 | background-color: unset; 195 | padding-left: 0; 196 | margin-left: 0; 197 | text-transform: none; 198 | } 199 | 200 | #userDetail, #themeDetail{ 201 | border: var(--primary-text-color); 202 | border-width: 0.5px; 203 | border-style: groove; 204 | padding-left: 20px; 205 | padding-right: 20px; 206 | 207 | } 208 | #settings-alarm, #settings-all, #settings-info, #settings-sensors, #settings-cameras, #settings-floorplan, #settings-mqtt { 209 | /* padding-bottom: 100px; */ 210 | } 211 | .iron-selected { 212 | color: var(--title-color); 213 | } 214 | 215 | .validation-error{ 216 | border: red; 217 | border-style: groove; 218 | border-width: 2px; 219 | } 220 | 221 | .action-button{ 222 | border: 3px solid var(--action-button-border-color) !important; 223 | } 224 | 225 | .dropdown-content{ 226 | width: 90%; 227 | } 228 | 229 | .dropdown-content iron-icon { 230 | padding-right: 10px; 231 | } 232 | .sensors { 233 | padding-right: 20px; 234 | } 235 | 236 | .checkbox{ 237 | padding-left:10px; 238 | padding-right:10px; 239 | } 240 | 241 | .tappable { 242 | cursor: pointer; 243 | margin-top: 2px; 244 | font-size: 1.2em; 245 | } 246 | 247 | #container ::slotted(.content) { 248 | padding: 15px; 249 | } 250 | #warning-button { 251 | color: var(--dark-orange) !important; 252 | } 253 | .shake{ 254 | animation: shake linear 1s; 255 | animation-iteration-count: 1; 256 | transform-origin: 50% 50%; 257 | -webkit-animation: shake linear 1s; 258 | -webkit-animation-iteration-count: 1; 259 | -webkit-transform-origin: 50% 50%; 260 | -moz-animation: shake linear 1s; 261 | -moz-animation-iteration-count: 1; 262 | -moz-transform-origin: 50% 50%; 263 | -o-animation: shake linear 1s; 264 | -o-animation-iteration-count: 1; 265 | -o-transform-origin: 50% 50%; 266 | -ms-animation: shake linear 1s; 267 | -ms-animation-iteration-count: 1; 268 | -ms-transform-origin: 50% 50%; 269 | } 270 | #controls{ 271 | padding-top: 5px; 272 | } 273 | #carousel_main{ 274 | position: relative; 275 | /* height: 60vh; */ 276 | padding: 15px; 277 | 278 | /* transition: all 1s ease-in-out; */ 279 | /* -webkit-transition: all 1s ease-in-out; /** Chrome & Safari **/ 280 | /* -moz-transition: all 1s ease-in-out; /** Firefox **/ 281 | /* -o-transition: all 1s ease-in-out; /** Opera **/ 282 | } 283 | #carousel_settings{ 284 | position: relative; 285 | z-index: 1; 286 | } 287 | 288 | .carousel-slide-in { 289 | /* transform: translate(-1600px,0); */ 290 | /* -webkit-transform: translate(-1600px,0); /** Chrome & Safari **/ 291 | /* -o-transform: translate(-1600px,0); /** Opera **/ 292 | /* -moz-transform: translate(-1600px,0); /** Firefox **/ 293 | 294 | } 295 | 296 | .margin-left { 297 | margin-left: -15px; 298 | } 299 | 300 | .carousel-cell { 301 | /* left: 1600px; */ 302 | top: 0; 303 | /* position: absolute; */ 304 | 305 | /* transition: all 1s ease-in-out; */ 306 | /* -webkit-transition: all 1s ease-in-out; /** Chrome & Safari **/ 307 | /* -moz-transition: all 1s ease-in-out; /** Firefox **/ 308 | /* -o-transition: all 1s ease-in-out; /** Opera **/ 309 | 310 | width: 100%; 311 | /* margin-bottom: 60px; */ 312 | color: var(--panel-text-color); 313 | } 314 | 315 | ha-label-badge { 316 | --ha-label-badge-color: rgb(223, 76, 30); 317 | } 318 | :host ::slotted(.badge-container.ha-label-badge) { 319 | margin: 0px 12px; 320 | } 321 | .actions { 322 | margin-left: 20px; 323 | @apply(--layout-self-center); 324 | } 325 | .flickity-viewport { 326 | /* transition: height 0.5s; */ 327 | } 328 | .box { 329 | display: -webkit-box; 330 | display: -ms-flexbox; 331 | display: flex; 332 | -webkit-box-pack: center; 333 | -ms-flex-pack: center; 334 | justify-content: center; /* align horizontal */ 335 | -webkit-box-align: center; 336 | -ms-flex-align: center; 337 | align-items: center; /* align vertical */ 338 | } 339 | 340 | .box-sensors { 341 | width: 100%; 342 | } 343 | .title{ 344 | font-size: x-large; 345 | } 346 | .sensor-title{ 347 | font-size: x-large; 348 | } 349 | .title, .sensor-title{ 350 | color: var(--subtitle-text-color); 351 | text-align: center; 352 | font-weight: 500; 353 | padding-top: 15px; 354 | } 355 | .openSensors { 356 | margin-left: 8%; 357 | margin-right: 8%; 358 | color: var(--openSensors-title-color); 359 | font-size: x-large; 360 | text-align: center; 361 | font-weight: 500; 362 | padding: 15px 0px; 363 | } 364 | 365 | .summary { 366 | margin: 0px auto 0px; 367 | text-align: center; 368 | } 369 | .cameras { 370 | cursor: pointer; 371 | } 372 | 373 | .camera-caption { 374 | @apply --paper-font-common-nowrap; 375 | position: relative; 376 | bottom: 46px; 377 | /* border-bottom-left-radius: 2px; 378 | border-bottom-right-radius: 2px;*/ 379 | background-color: rgba(0, 0, 0, 0.3); 380 | padding: 10px; 381 | font-size: 16px; 382 | font-weight: 500; 383 | color: white; 384 | } 385 | 386 | .content-box { 387 | display: flex; 388 | flex-direction: column; 389 | justify-content: center; 390 | } 391 | /* ################################################# height > 600px ############################################# */ 392 | @media screen and (min-height: 600px) { 393 | #status-bar { 394 | margin-top: 15px; 395 | margin-bottom: 15px; 396 | height: 87px; 397 | } 398 | } 399 | 400 | /* ################################################# width XS < 767 ############################################# */ 401 | @media screen and (max-width: 767px) { 402 | 403 | #carousel_settings { 404 | margin-right: 10px; 405 | margin-left: 10px; 406 | /*text-align: center;*/ 407 | } 408 | #carousel_settings h1{ 409 | text-align: center; 410 | } 411 | #carousel_settings li{ 412 | text-align: left; 413 | } 414 | .log-item-date { 415 | font-weight: 300; 416 | font-style: italic; 417 | font-size: smaller; 418 | } 419 | .camera-feed { 420 | width: 300px; 421 | } 422 | .content-box { 423 | width: 100%; 424 | } 425 | #code-display { 426 | margin-top: 10px; 427 | } 428 | #main-content { 429 | /* padding-bottom: 30px; */ 430 | } 431 | div.arm paper-button { 432 | width: 36vh; 433 | height: 36vh; 434 | margin: 15px; 435 | font-size: large; 436 | } 437 | #alarm-status span { 438 | font-size: x-large; 439 | } 440 | .sm { 441 | display: none !important; 442 | } 443 | .sign { 444 | font-size: x-large; 445 | } 446 | #clock, #weather { 447 | display: none; 448 | } 449 | #hour, #minute, #middle { 450 | font-size: xx-large; 451 | } 452 | #meridiem { 453 | font-size: 10px; 454 | margin-left: -20px; 455 | margin-right: 3px; 456 | } 457 | #weather-icon img { 458 | width: 40px; 459 | } 460 | .weather { 461 | -webkit-box-pack: center; 462 | -ms-flex-pack: center; 463 | justify-content: center; 464 | -webkit-box-orient: vertical; 465 | -webkit-box-direction: normal; 466 | -ms-flex-direction: column; 467 | flex-direction: column; 468 | display: -webkit-box; 469 | display: -ms-flexbox; 470 | display: flex; 471 | color: var(--weather-text-color); 472 | } 473 | .vLayout { 474 | -webkit-box-orient: vertical; 475 | -webkit-box-direction: normal; 476 | display: -webkit-box; 477 | display: -ms-flexbox; 478 | display: flex; 479 | flex-wrap: wrap; 480 | justify-content: center; 481 | text-align: center; 482 | } 483 | div.arm paper-button { 484 | /* margin-bottom: 10px;*/ 485 | } 486 | div.digits paper-button, div.digits paper-icon-button{ 487 | color: var(--arm-button-text-color); 488 | border-radius: var(--button-shape); 489 | min-width: 11vh; 490 | height: 11vh; 491 | padding: 0; 492 | border: 2px solid var(--arm-button-border-color); 493 | background-color: var(--button-background-color); 494 | margin: 8px; 495 | font-weight: 400; 496 | display: flex; 497 | } 498 | #code-display { 499 | margin-bottom: 15px; 500 | } 501 | } 502 | /* ############## SM > 768 ############### */ 503 | 504 | #info-panel-selection paper-icon-button { 505 | width: 12vw; 506 | height: 12vw; 507 | } 508 | 509 | @media screen and (min-width: 768px) { 510 | .content-box { 511 | /*height: 100%;*/ 512 | width: 100%; 513 | padding-bottom: 20px; 514 | } 515 | .cameras{ 516 | padding-right: 15px; 517 | } 518 | .camera-feed { 519 | width: 400px; 520 | 521 | } 522 | 523 | .log-item{ 524 | display:flex; 525 | flex-direction:row; 526 | width: 100%; 527 | } 528 | 529 | .log-item > span{ 530 | flex-grow:1; 531 | } 532 | 533 | .log-item > div{ 534 | flex-grow:0; 535 | font-size: large; 536 | margin: auto; 537 | font-family: var(--paper-font-body1_-_font-family);; 538 | } 539 | 540 | .log-item-date { 541 | font-weight: 300; 542 | font-style: italic; 543 | text-align: right; 544 | } 545 | #main-content { 546 | /* padding-top: 20px; 547 | padding-bottom: 120px; */ 548 | } 549 | #code-display { 550 | margin-top: 20px; 551 | margin-bottom: 40px; 552 | } 553 | div.arm paper-button { 554 | width: 36vh; 555 | height: 36vh; 556 | /* line-height: 110px;*/ 557 | margin: 20px; 558 | padding: 20px; 559 | font-size: x-large; 560 | } 561 | #alarm-status span { 562 | font-size: xx-large; 563 | } 564 | .xs { 565 | display: none !important; 566 | } 567 | .sign { 568 | font-size: xx-large; 569 | } 570 | #meridiem { 571 | font-size: small; 572 | margin-left: -29px; 573 | } 574 | #weather, #clock { 575 | width: 30%; 576 | } 577 | #alarm-status { 578 | width: 40% 579 | } 580 | #weather-icon img { 581 | width: 70px; 582 | } 583 | .vLayout { 584 | -webkit-box-orient: horizontal; 585 | -webkit-box-direction: normal; 586 | display: -webkit-box; 587 | display: -ms-flexbox; 588 | justify-content: center; 589 | display: flex; 590 | min-width: 50%; 591 | justify-content: center; 592 | flex-wrap: wrap; 593 | text-align: center; 594 | } 595 | div.digits paper-button, div.digits paper-iron-button{ 596 | border-radius: var(--button-shape); 597 | width: 18vh; 598 | height: 18vh; 599 | border: 2px solid var(--arm-button-border-color); 600 | background-color: var(--button-background-color); 601 | display: flex; 602 | vertical-align: middle; 603 | margin: 10px; 604 | padding: 20px; 605 | text-align: center; 606 | font-size: xx-large; 607 | } 608 | } 609 | 610 | /* ######################################################### width => 1024 ################################################# */ 611 | @media screen and (min-width: 1024px) { 612 | #weather, #clock, #alarm-status { 613 | width: 33%; 614 | } 615 | .info-detail { 616 | font-size: large; 617 | } 618 | h1 { 619 | font-size: x-large; 620 | } 621 | 622 | .settings-nav-inner{ 623 | margin-right: 20px; 624 | } 625 | 626 | .settings-nav-inner paper-icon-button { 627 | width: 60px; 628 | height: 60px; 629 | 630 | } 631 | 632 | 633 | #info-panel-selection paper-icon-button { 634 | width: 75px; 635 | height: 75px; 636 | margin-left: 15px; 637 | } 638 | span, li, .settings-nav-text{ 639 | font-size: large; 640 | } 641 | 642 | 643 | } 644 | 645 | /* ######################################################### width => 1200 ################################################# */ 646 | @media screen and (min-width: 1200px) { 647 | .md { 648 | display: none !important; 649 | } 650 | .lg{ 651 | display: flex !important; 652 | } 653 | .initial { 654 | display: flex; 655 | } 656 | 657 | .content-box { 658 | width: 75%; 659 | } 660 | #controls { 661 | /* width: 50% !important; 662 | max-height: 60vh; 663 | height: auto;*/ 664 | } 665 | 666 | 667 | 668 | } 669 | 670 | 671 | /* ######################################################### ALL DEVICES ################################################# */ 672 | #icon { 673 | margin-right: 15px; 674 | } 675 | paper-listbox { 676 | color: var(--info-panel-buttons-color); 677 | background: unset; 678 | } 679 | #content { 680 | display: flex; 681 | flex-direction: column; 682 | flex: 1 1 auto; 683 | width: 100%; 684 | overflow-y: auto; 685 | overflow-x: hidden; 686 | } 687 | h1 { 688 | color: var(--title-color); 689 | } 690 | 691 | #settings-nav { 692 | color: var(--primary-text-color); 693 | justify-content: center; 694 | text-align: center; 695 | padding-top: 10px; 696 | } 697 | #set-login { 698 | text-align: center; 699 | } 700 | .settings-nav-inner { 701 | /* max-width: 40px;*/ 702 | padding: 3px; 703 | } 704 | /*.settings-nav-text { 705 | font-size: x-small; 706 | }*/ 707 | 708 | .info-header { 709 | color: var(--info-header-text-color); 710 | font-size: large; 711 | padding-right: 5px; 712 | } 713 | .setting-outer { 714 | padding-top: 10px; 715 | padding-bottom: 10px; 716 | } 717 | .setting-inner { 718 | padding-bottom: 5px; 719 | width: 100%; 720 | padding-right: 10px; 721 | } 722 | .setting-input > paper-input{ 723 | flex-grow: 3; 724 | } 725 | .setting-input > paper-icon-button{ 726 | flex-grow: 1; 727 | width: 50px; 728 | height: 50px; 729 | } 730 | .setting-toggle { 731 | align-items: center; 732 | } 733 | #info-panel-selection { 734 | color: var(--info-panel-buttons-color); 735 | background: var(--panel-outer-background-color); 736 | position: relative; 737 | display: inline; 738 | flex-wrap: wrap; 739 | align-items: center; 740 | text-align: center; 741 | justify-content: space-between; 742 | padding: .5rem 1rem; 743 | } 744 | 745 | #main-content { 746 | display: flex; 747 | justify-content: center; 748 | flex-direction: row; 749 | background-color: var(--panel-background-color); 750 | } 751 | 752 | .sign { 753 | margin-left: -20px; 754 | } 755 | .ok { 756 | color: green; 757 | } 758 | .warning { 759 | color: orange; 760 | } 761 | .danger { 762 | color: red; 763 | } 764 | 765 | .name { 766 | text-align: center; 767 | } 768 | div.countdown-timer { 769 | display: var(--countdown-timer-display); 770 | } 771 | 772 | div.code { 773 | text-transform: capitalize; 774 | padding-top: 23px; 775 | } 776 | #code-display { 777 | text-align: center; 778 | font-size: xx-large; 779 | letter-spacing: 5px; 780 | color: var(--panel-text-color); 781 | padding-bottom: 10px; 782 | width: 150px; 783 | border-bottom: solid; 784 | border-bottom-width: 1.3px; 785 | margin-top: auto; 786 | margin-right: auto; 787 | margin-left: auto; 788 | height: 20px; 789 | } 790 | 791 | div.code iron-label { 792 | font-size: x-large; 793 | } 794 | 795 | paper-button { 796 | background: #6b8d9c; 797 | color: #ffffff; 798 | } 799 | 800 | 801 | div.arm paper-button { 802 | background-color: var(--button-background-color); 803 | border-radius: var(--button-shape); 804 | border: 3px solid var(--arm-button-border-color); 805 | display: flex; 806 | color: var(--arm-button-text-color); 807 | } 808 | div.arm paper-button div { 809 | border-radius: var(--button-shape); 810 | width: 100px; 811 | height: 100px; 812 | border: 3px solid var(--arm-button-border-color); 813 | display: flex; 814 | line-height: 100px; 815 | margin: 20px; 816 | padding: 20px; 817 | } 818 | div.arm paper-button.big { 819 | width: 68%; 820 | } 821 | div.arm paper-button.cancel { 822 | border: 3px solid var(--cancel-color); 823 | } 824 | div.arm paper-button.override { 825 | border: 3px solid var(--override-color); 826 | } 827 | 828 | div.options { 829 | padding-top: 10px; 830 | } 831 | div.options paper-button{ 832 | width: 96.5%; 833 | } 834 | .tappable { 835 | cursor: pointer; 836 | margin-top: 2px; 837 | font-size: 1.2em; 838 | } 839 | .shadow { 840 | padding-bottom: 6px; 841 | -webkit-box-reflect: var(--shadow-effect); 842 | } 843 | #time, #time span { 844 | color: var( --time-text-color); 845 | text-shadow: 1px 1px #6b8d9b; 846 | display: var(--time-display); 847 | font-size: xx-large; 848 | font-family: var(--font-header); 849 | } 850 | 851 | #middle { 852 | text-align: center; 853 | padding-left: 8px; 854 | 855 | } 856 | #meridiem { 857 | display: inline-block; 858 | vertical-align: middle; 859 | line-height: normal; 860 | height: 100%; 861 | color: #6b8d9b; 862 | } 863 | #alarm-status { 864 | flex-direction: column; 865 | text-transform: capitalize; 866 | } 867 | #alarm-status span { 868 | color: var( --alarmstatus-text-color); 869 | font-weight: bolder; 870 | text-align: center; 871 | } 872 | #weather-icon { 873 | width: 100px; 874 | text-align: center; 875 | fill: var( --weather-image-color); 876 | } 877 | #weather-icon object { 878 | transform: scale(3,3); 879 | } 880 | .weather-summary span { 881 | display: inline-block; 882 | vertical-align: middle; 883 | font-size: x-large; 884 | margin-left: 5px; 885 | padding-right: 5px; 886 | text-align: center; 887 | height: 100%; 888 | font-family: var(--font-header); 889 | color: var( --weather-text-color); 890 | } 891 | 892 | #main-title-text{ 893 | flex-wrap: wrap; 894 | text-align: center; 895 | 896 | font-size: xx-large; 897 | font-family: var(--font-header); 898 | justify-content: space-between; 899 | 900 | padding: .5rem 1rem; 901 | color: var(--header-text-color) 902 | } 903 | .hide-bar { 904 | margin-top: -64px; 905 | height: calc(100vh + 64px); 906 | } 907 | 908 | .view { 909 | position: relative; 910 | top: 24vh; 911 | left: 0; 912 | right: 0; 913 | bottom: 0; 914 | -webkit-perspective: 400; 915 | perspective: 400; 916 | } 917 | #codepanel { 918 | overflow: hidden; 919 | } 920 | 921 | @-webkit-keyframes shake{ 922 | 0% { 923 | -webkit-transform: translate(0px,0px) ; 924 | } 925 | 10% { 926 | -webkit-transform: translate(-10px,0px) ; 927 | } 928 | 20% { 929 | -webkit-transform: translate(10px,0px) ; 930 | } 931 | 30% { 932 | -webkit-transform: translate(-10px,0px) ; 933 | } 934 | 40% { 935 | -webkit-transform: translate(10px,0px) ; 936 | } 937 | 50% { 938 | -webkit-transform: translate(-10px,0px) ; 939 | } 940 | 60% { 941 | -webkit-transform: translate(10px,0px) ; 942 | } 943 | 70% { 944 | -webkit-transform: translate(-10px,0px) ; 945 | } 946 | 80% { 947 | -webkit-transform: translate(10px,0px) ; 948 | } 949 | 90% { 950 | -webkit-transform: translate(-10px,0px) ; 951 | } 952 | 100% { 953 | -webkit-transform: translate(0px,0px) ; 954 | } 955 | } 956 | 957 | @keyframes shake{ 958 | 0% { 959 | transform: translate(0px,0px) ; 960 | } 961 | 10% { 962 | transform: translate(-10px,0px) ; 963 | } 964 | 20% { 965 | transform: translate(10px,0px) ; 966 | } 967 | 30% { 968 | transform: translate(-10px,0px) ; 969 | } 970 | 40% { 971 | transform: translate(10px,0px) ; 972 | } 973 | 50% { 974 | transform: translate(-10px,0px) ; 975 | } 976 | 60% { 977 | transform: translate(10px,0px) ; 978 | } 979 | 70% { 980 | transform: translate(-10px,0px) ; 981 | } 982 | 80% { 983 | transform: translate(10px,0px) ; 984 | } 985 | 90% { 986 | transform: translate(-10px,0px) ; 987 | } 988 | 100% { 989 | transform: translate(0px,0px) ; 990 | } 991 | } 992 | 993 | .remove { 994 | display: none !important; 995 | } 996 | 997 | .info-detail { 998 | color: var(--info-detail-text-color); 999 | padding-top: 2px; 1000 | } 1001 | 1002 | /* The Modal (background) */ 1003 | #modalCamera, #modalDonate { 1004 | display: none; /* Hidden by default */ 1005 | align-items: center; 1006 | position: absolute; /* Stay in place */ 1007 | z-index: 1; /* Sit on top */ 1008 | padding-top: 5vh; /* Location of the box */ 1009 | left: 0; 1010 | top: 0; 1011 | width: 100%; /* Full width */ 1012 | height: calc(100% - 5vh); /* Full height */ 1013 | overflow: hidden; /* Enable scroll if needed */ 1014 | background-color: rgb(0,0,0); /* Fallback color */ 1015 | background-color: rgba(0,0,0,0.9); /* Black w/ opacity */ 1016 | } 1017 | 1018 | 1019 | /* Modal Content (Image) */ 1020 | #imgCamera { 1021 | margin: auto; 1022 | display: block; 1023 | width: 80%; 1024 | max-width: 100vh; 1025 | } 1026 | 1027 | /* Caption of Modal Image (Image Text) - Same Width as the Image */ 1028 | #captionCamera { 1029 | margin: auto; 1030 | display: block; 1031 | width: 100%; 1032 | text-align: center; 1033 | padding: 10px 0; 1034 | height: 150px; 1035 | 1036 | @apply --paper-font-common-nowrap; 1037 | position: relative; 1038 | bottom: 46px; 1039 | border-bottom-left-radius: 2px; 1040 | border-bottom-right-radius: 2px; 1041 | background-color: rgba(0, 0, 0, 0.3); 1042 | font-size: 16px; 1043 | font-weight: 500; 1044 | color: white; 1045 | } 1046 | 1047 | #donateMethod{ 1048 | font-size: xx-large; 1049 | padding-bottom: 20px; 1050 | } 1051 | 1052 | #donateAddress, #donateMethod { 1053 | margin: auto; 1054 | display: block; 1055 | width: 100%; 1056 | text-align: center; 1057 | 1058 | padding-top: 20px; 1059 | 1060 | font-weight: 500; 1061 | } 1062 | 1063 | #donateAddress { 1064 | font-size: large; 1065 | overflow-wrap: break-word; 1066 | } 1067 | 1068 | /* Add Animation - Zoom in the Modal */ 1069 | .modal-camera-content, #captionCamera { 1070 | animation-name: zoomCamera; 1071 | animation-duration: 1.6s; 1072 | } 1073 | 1074 | .qr { 1075 | background-repeat: no-repeat; 1076 | background-size: contain; 1077 | margin: auto; 1078 | display: block; 1079 | 1080 | width: 256px; 1081 | height: 256px; 1082 | } 1083 | 1084 | .logo { 1085 | background-repeat: no-repeat; 1086 | background-size: contain; 1087 | width: 48px; 1088 | height: 48px; 1089 | margin: 10px auto 10px auto; 1090 | } 1091 | 1092 | .donate, .donate-paypal { 1093 | text-align: center; 1094 | padding-top: 10px; 1095 | padding-right: 10px; 1096 | max-width: 200px; 1097 | overflow-wrap: break-word; 1098 | 1099 | } 1100 | 1101 | .logo{ 1102 | -webkit-filter: invert(1); 1103 | filter: invert(1); 1104 | cursor: pointer; 1105 | } 1106 | 1107 | .flexwrap { 1108 | flex-wrap: wrap; 1109 | } 1110 | 1111 | .donate-box { 1112 | max-width: 200px; 1113 | overflow-wrap: break-word; 1114 | text-align: center; 1115 | padding-right: 10px; 1116 | } 1117 | 1118 | .bch .qr{ 1119 | background-image: url(''); 1120 | } 1121 | .btc .qr{ 1122 | background-image: url(''); 1123 | } 1124 | .ltc .qr{ 1125 | background-image: url(''); 1126 | } 1127 | .eth .qr{ 1128 | background-image: url(''); 1129 | } 1130 | .paypal .qr{ 1131 | background-image: url(''); 1132 | } 1133 | .xrp .qr{ 1134 | background-image: url(''); 1135 | } 1136 | #xrp .logo, .xrp .logo{ 1137 | background-image: url(''); 1138 | } 1139 | #btc .logo, .btc .logo{ 1140 | background-image: url(''); 1141 | } 1142 | #bch .logo, .bch .logo{ 1143 | background-image: url(''); 1144 | } 1145 | #ltc .logo, .ltc .logo{ 1146 | background-image: url(''); 1147 | } 1148 | #eth .logo, .eth .logo{ 1149 | background-image: url(''); 1150 | } 1151 | #paypal .logo, .paypal .logo{ 1152 | background-image: url(''); 1153 | } 1154 | -------------------------------------------------------------------------------- /www/alarm/custom-element.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 17 | 18 | 19 | -------------------------------------------------------------------------------- /www/alarm/donate/bch.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/alarm/donate/bch.png -------------------------------------------------------------------------------- /www/alarm/donate/btc.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/alarm/donate/btc.png -------------------------------------------------------------------------------- /www/alarm/donate/eth.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/alarm/donate/eth.png -------------------------------------------------------------------------------- /www/alarm/donate/ltc.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/alarm/donate/ltc.png -------------------------------------------------------------------------------- /www/alarm/donate/paypal.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/alarm/donate/paypal.png -------------------------------------------------------------------------------- /www/alarm/donate/xrp.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/alarm/donate/xrp.png -------------------------------------------------------------------------------- /www/alarm/lobster.woff2: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/alarm/lobster.woff2 -------------------------------------------------------------------------------- /www/images/camera-garage.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/images/camera-garage.jpg -------------------------------------------------------------------------------- /www/images/camera-outdoor.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/images/camera-outdoor.jpg -------------------------------------------------------------------------------- /www/images/camera-pool.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/images/camera-pool.jpg -------------------------------------------------------------------------------- /www/images/camera-tv-room.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/images/camera-tv-room.jpg -------------------------------------------------------------------------------- /www/images/gazos.jpg: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/images/gazos.jpg -------------------------------------------------------------------------------- /www/images/ha.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/gazoscalvertos/Hass-Custom-Alarm/2dd16d6725458ebc8ef8bfd73f1ed062c63803a3/www/images/ha.png -------------------------------------------------------------------------------- /www/lib/countdown360.js: -------------------------------------------------------------------------------- 1 | /* 2 | * Countdown 360 - v0.1.9 3 | * This is a simple attractive circular countdown timer that counts down a number of seconds. The style is configurable and callbacks are supported on completion. 4 | * https://github.com/johnschult/jquery.countdown360 5 | * 6 | * Made by John Schult 7 | * Under MIT License 8 | */ 9 | (function ($, window, document, undefined) { 10 | var pluginName = "countdown360", 11 | defaults = { 12 | radius: 15.5, // radius of arc 13 | strokeStyle: "#477050", // the color of the stroke 14 | strokeWidth: undefined, // the stroke width, dynamically calulated if omitted in options 15 | fillStyle: "#8ac575", // the fill color 16 | fillStyle_0to50: "#8ac575", // the beginning of the timer 0 to 50% 17 | fillStyle_50to75: "orange", // the middle of the timer 50 to 75% 18 | fillStyle_75to100: "red", // the end of the timer 75 to 100% 19 | fontColor: "#477050", // the font color 20 | fontFamily: "sans-serif", // the font family 21 | fontSize: undefined, // the font size, dynamically calulated if omitted in options 22 | fontWeight: 700, // the font weight 23 | autostart: true, // start the countdown automatically 24 | seconds: 10, // the number of seconds to count down 25 | label: ["second", "seconds"], // the label to use or false if none 26 | startOverAfterAdding: true, // Start the timer over after time is added with addSeconds 27 | smooth: false, // should the timer be smooth or stepping 28 | onComplete: function () {} 29 | }; 30 | 31 | function Plugin(element, options) { 32 | this.element = element; 33 | this.settings = $.extend({}, defaults, options); 34 | if (!this.settings.fontSize) { this.settings.fontSize = this.settings.radius/1.2; } 35 | if (!this.settings.strokeWidth) { this.settings.strokeWidth = this.settings.radius/4; } 36 | this._defaults = defaults; 37 | this._name = pluginName; 38 | this._init(); 39 | } 40 | 41 | Plugin.prototype = { 42 | 43 | getTimeRemaining: function() 44 | { 45 | var timeRemaining = this._secondsLeft(this.getElapsedTime()); 46 | return timeRemaining; 47 | }, 48 | getElapsedTime: function() 49 | { 50 | return Math.round((new Date().getTime() - this.startedAt.getTime())/1000); 51 | }, 52 | extendTimer: function (value) { 53 | var seconds = parseInt(value), 54 | secondsElapsed = Math.round((new Date().getTime() - this.startedAt.getTime())/1000); 55 | if ((this._secondsLeft(secondsElapsed) + seconds) <= this.settings.seconds) { 56 | this.startedAt.setSeconds(this.startedAt.getSeconds() + parseInt(value)); 57 | } 58 | }, 59 | addSeconds: function (value) { 60 | var secondsElapsed = Math.round((new Date().getTime() - this.startedAt.getTime())/1000); 61 | if (this.settings.startOverAfterAdding) { 62 | this.settings.seconds = this._secondsLeft(secondsElapsed) + parseInt(value); 63 | this.start(); 64 | } else { 65 | this.settings.seconds += parseInt(value); 66 | } 67 | }, 68 | 69 | start: function () { 70 | this.startedAt = new Date(); 71 | this._drawCountdownShape(Math.PI*3.5, true); 72 | this._drawCountdownLabel(0); 73 | var timerInterval = 1000; 74 | if (this.settings.smooth) { 75 | timerInterval = 16; 76 | } 77 | this.interval = setInterval(jQuery.proxy(this._draw, this), timerInterval); 78 | }, 79 | 80 | stop: function (cb) { 81 | clearInterval(this.interval); 82 | if (cb) { cb(); } 83 | }, 84 | 85 | _init: function () { 86 | this.settings.width = (this.settings.radius * 2) + (this.settings.strokeWidth * 2); 87 | this.settings.height = this.settings.width; 88 | this.settings.arcX = this.settings.radius + this.settings.strokeWidth; 89 | this.settings.arcY = this.settings.arcX; 90 | this._initPen(this._getCanvas()); 91 | if (this.settings.autostart) { this.start(); } 92 | }, 93 | 94 | _getCanvas: function () { 95 | var $canvas = $("" + 98 | ""); 99 | $(this.element).prepend($canvas[0]); 100 | return $canvas[0]; 101 | }, 102 | 103 | _initPen: function (canvas) { 104 | this.pen = canvas.getContext("2d"); 105 | this.pen.lineWidth = this.settings.strokeWidth; 106 | this.pen.strokeStyle = this.settings.strokeStyle; 107 | this.pen.fillStyle = this.settings.fillStyle; 108 | this.pen.textAlign = "center"; 109 | this.pen.textBaseline = "middle"; 110 | this.ariaText = $(canvas).children("#countdown-text"); 111 | this._clearRect(); 112 | }, 113 | 114 | _clearRect: function () { 115 | this.pen.clearRect(0, 0, this.settings.width, this.settings.height); 116 | }, 117 | 118 | _secondsLeft: function(secondsElapsed) { 119 | return this.settings.seconds - secondsElapsed; 120 | }, 121 | 122 | _drawCountdownLabel: function (secondsElapsed) { 123 | this.ariaText.text(secondsLeft); 124 | this.pen.font = this.settings.fontWeight + " " + this.settings.fontSize + "px " + this.settings.fontFamily; 125 | var secondsLeft = this._secondsLeft(secondsElapsed), 126 | label = secondsLeft === 1 ? this.settings.label[0] : this.settings.label[1], 127 | drawLabel = this.settings.label && this.settings.label.length === 2, 128 | x = this.settings.width/2; 129 | if (drawLabel) { 130 | y = this.settings.height/2 - (this.settings.fontSize/6.2); 131 | } else { 132 | y = this.settings.height/2; 133 | } 134 | this.pen.fillStyle = this.settings.fillStyle; 135 | this.pen.fillText(secondsLeft + 1, x, y); 136 | this.pen.fillStyle = this.settings.fontColor; 137 | this.pen.fillText(secondsLeft, x, y); 138 | if (drawLabel) { 139 | this.pen.font = "normal small-caps " + (this.settings.fontSize/3) + "px " + this.settings.fontFamily; 140 | this.pen.fillText(label, this.settings.width/2, this.settings.height/2 + (this.settings.fontSize/2.2)); 141 | } 142 | }, 143 | 144 | _drawCountdownShape: function (endAngle, drawStroke) { 145 | this.pen.fillStyle = this.settings.fillStyle; 146 | this.pen.beginPath(); 147 | this.pen.arc(this.settings.arcX, this.settings.arcY, this.settings.radius, Math.PI*1.5, endAngle, false); 148 | this.pen.fill(); 149 | if (drawStroke) { this.pen.stroke(); } 150 | }, 151 | _draw: function () { 152 | var millisElapsed, secondsElapsed; 153 | millisElapsed = new Date().getTime() - this.startedAt.getTime(); 154 | secondsElapsed = Math.floor((millisElapsed)/1000); 155 | progress = secondsElapsed / this.settings.seconds; 156 | 157 | if (progress > 0.75) { 158 | this.settings.fillStyle = this.settings.fillStyle_75to100; 159 | } else if (progress > 0.5 ){ 160 | this.settings.fillStyle = this.settings.fillStyle_50to75; 161 | } else { 162 | this.settings.fillStyle =this.settings.fillStyle_0to50; 163 | } 164 | 165 | endAngle = (Math.PI*3.5) - (((Math.PI*2)/(this.settings.seconds * 1000)) * millisElapsed); 166 | this._clearRect(); 167 | this._drawCountdownShape(Math.PI*3.5, false); 168 | if (secondsElapsed < this.settings.seconds) { 169 | this._drawCountdownShape(endAngle, true); 170 | this._drawCountdownLabel(secondsElapsed); 171 | } else { 172 | this._drawCountdownLabel(this.settings.seconds); 173 | this.stop(); 174 | this.settings.onComplete(); 175 | } 176 | } 177 | 178 | }; 179 | 180 | $.fn[pluginName] = function (options) { 181 | var plugin = null; 182 | this.each(function() { 183 | plugin = $.data(this, "plugin_" + pluginName); 184 | if (!plugin) { 185 | plugin = new Plugin(this, options); 186 | $.data(this, "plugin_" + pluginName, plugin); 187 | } 188 | }); 189 | return plugin; 190 | }; 191 | 192 | })(jQuery, window, document); 193 | -------------------------------------------------------------------------------- /www/lib/jscolor.js: -------------------------------------------------------------------------------- 1 | /** 2 | * jscolor - JavaScript Color Picker 3 | * 4 | * @link http://jscolor.com 5 | * @license For open source use: GPLv3 6 | * For commercial use: JSColor Commercial License 7 | * @author Jan Odvarko 8 | * @version 2.0.5 9 | * 10 | * See usage examples at http://jscolor.com/examples/ 11 | */ 12 | 13 | 14 | "use strict"; 15 | 16 | 17 | if (!window.jscolor) { window.jscolor = (function () { 18 | 19 | 20 | var jsc = { 21 | 22 | 23 | register : function () { 24 | jsc.attachDOMReadyEvent(jsc.init); 25 | jsc.attachEvent(document, 'mousedown', jsc.onDocumentMouseDown); 26 | jsc.attachEvent(document, 'touchstart', jsc.onDocumentTouchStart); 27 | jsc.attachEvent(window, 'resize', jsc.onWindowResize); 28 | }, 29 | 30 | 31 | init : function () { 32 | if (jsc.jscolor.lookupClass) { 33 | jsc.jscolor.installByClassName(jsc.jscolor.lookupClass); 34 | } 35 | }, 36 | 37 | 38 | tryInstallOnElements : function (elms, className) { 39 | var matchClass = new RegExp('(^|\\s)(' + className + ')(\\s*(\\{[^}]*\\})|\\s|$)', 'i'); 40 | 41 | for (var i = 0; i < elms.length; i += 1) { 42 | if (elms[i].type !== undefined && elms[i].type.toLowerCase() == 'color') { 43 | if (jsc.isColorAttrSupported) { 44 | // skip inputs of type 'color' if supported by the browser 45 | continue; 46 | } 47 | } 48 | var m; 49 | if (!elms[i].jscolor && elms[i].className && (m = elms[i].className.match(matchClass))) { 50 | console.log('test'); 51 | var targetElm = elms[i]; 52 | var optsStr = null; 53 | 54 | var dataOptions = jsc.getDataAttr(targetElm, 'jscolor'); 55 | if (dataOptions !== null) { 56 | optsStr = dataOptions; 57 | } else if (m[4]) { 58 | optsStr = m[4]; 59 | } 60 | 61 | var opts = {}; 62 | if (optsStr) { 63 | try { 64 | opts = (new Function ('return (' + optsStr + ')'))(); 65 | } catch(eParseError) { 66 | jsc.warn('Error parsing jscolor options: ' + eParseError + ':\n' + optsStr); 67 | } 68 | } 69 | targetElm.jscolor = new jsc.jscolor(targetElm, opts); 70 | } 71 | } 72 | }, 73 | 74 | 75 | isColorAttrSupported : (function () { 76 | var elm = document.createElement('input'); 77 | if (elm.setAttribute) { 78 | elm.setAttribute('type', 'color'); 79 | if (elm.type.toLowerCase() == 'color') { 80 | return true; 81 | } 82 | } 83 | return false; 84 | })(), 85 | 86 | 87 | isCanvasSupported : (function () { 88 | var elm = document.createElement('canvas'); 89 | return !!(elm.getContext && elm.getContext('2d')); 90 | })(), 91 | 92 | 93 | fetchElement : function (mixed) { 94 | return typeof mixed === 'string' ? document.getElementById(mixed) : mixed; 95 | }, 96 | 97 | 98 | isElementType : function (elm, type) { 99 | return elm.nodeName.toLowerCase() === type.toLowerCase(); 100 | }, 101 | 102 | 103 | getDataAttr : function (el, name) { 104 | var attrName = 'data-' + name; 105 | var attrValue = el.getAttribute(attrName); 106 | if (attrValue !== null) { 107 | return attrValue; 108 | } 109 | return null; 110 | }, 111 | 112 | 113 | attachEvent : function (el, evnt, func) { 114 | if (el.addEventListener) { 115 | el.addEventListener(evnt, func, false); 116 | } else if (el.attachEvent) { 117 | el.attachEvent('on' + evnt, func); 118 | } 119 | }, 120 | 121 | 122 | detachEvent : function (el, evnt, func) { 123 | if (el.removeEventListener) { 124 | el.removeEventListener(evnt, func, false); 125 | } else if (el.detachEvent) { 126 | el.detachEvent('on' + evnt, func); 127 | } 128 | }, 129 | 130 | 131 | _attachedGroupEvents : {}, 132 | 133 | 134 | attachGroupEvent : function (groupName, el, evnt, func) { 135 | if (!jsc._attachedGroupEvents.hasOwnProperty(groupName)) { 136 | jsc._attachedGroupEvents[groupName] = []; 137 | } 138 | jsc._attachedGroupEvents[groupName].push([el, evnt, func]); 139 | jsc.attachEvent(el, evnt, func); 140 | }, 141 | 142 | 143 | detachGroupEvents : function (groupName) { 144 | if (jsc._attachedGroupEvents.hasOwnProperty(groupName)) { 145 | for (var i = 0; i < jsc._attachedGroupEvents[groupName].length; i += 1) { 146 | var evt = jsc._attachedGroupEvents[groupName][i]; 147 | jsc.detachEvent(evt[0], evt[1], evt[2]); 148 | } 149 | delete jsc._attachedGroupEvents[groupName]; 150 | } 151 | }, 152 | 153 | 154 | attachDOMReadyEvent : function (func) { 155 | var fired = false; 156 | var fireOnce = function () { 157 | if (!fired) { 158 | fired = true; 159 | func(); 160 | } 161 | }; 162 | 163 | if (document.readyState === 'complete') { 164 | setTimeout(fireOnce, 1); // async 165 | return; 166 | } 167 | 168 | if (document.addEventListener) { 169 | document.addEventListener('DOMContentLoaded', fireOnce, false); 170 | 171 | // Fallback 172 | window.addEventListener('load', fireOnce, false); 173 | 174 | } else if (document.attachEvent) { 175 | // IE 176 | document.attachEvent('onreadystatechange', function () { 177 | if (document.readyState === 'complete') { 178 | document.detachEvent('onreadystatechange', arguments.callee); 179 | fireOnce(); 180 | } 181 | }) 182 | 183 | // Fallback 184 | window.attachEvent('onload', fireOnce); 185 | 186 | // IE7/8 187 | if (document.documentElement.doScroll && window == window.top) { 188 | var tryScroll = function () { 189 | if (!document.body) { return; } 190 | try { 191 | document.documentElement.doScroll('left'); 192 | fireOnce(); 193 | } catch (e) { 194 | setTimeout(tryScroll, 1); 195 | } 196 | }; 197 | tryScroll(); 198 | } 199 | } 200 | }, 201 | 202 | 203 | warn : function (msg) { 204 | if (window.console && window.console.warn) { 205 | window.console.warn(msg); 206 | } 207 | }, 208 | 209 | 210 | preventDefault : function (e) { 211 | if (e.preventDefault) { e.preventDefault(); } 212 | e.returnValue = false; 213 | }, 214 | 215 | 216 | captureTarget : function (target) { 217 | // IE 218 | if (target.setCapture) { 219 | jsc._capturedTarget = target; 220 | jsc._capturedTarget.setCapture(); 221 | } 222 | }, 223 | 224 | 225 | releaseTarget : function () { 226 | // IE 227 | if (jsc._capturedTarget) { 228 | jsc._capturedTarget.releaseCapture(); 229 | jsc._capturedTarget = null; 230 | } 231 | }, 232 | 233 | 234 | fireEvent : function (el, evnt) { 235 | if (!el) { 236 | return; 237 | } 238 | if (document.createEvent) { 239 | var ev = document.createEvent('HTMLEvents'); 240 | ev.initEvent(evnt, true, true); 241 | el.dispatchEvent(ev); 242 | } else if (document.createEventObject) { 243 | var ev = document.createEventObject(); 244 | el.fireEvent('on' + evnt, ev); 245 | } else if (el['on' + evnt]) { // alternatively use the traditional event model 246 | el['on' + evnt](); 247 | } 248 | }, 249 | 250 | 251 | classNameToList : function (className) { 252 | return className.replace(/^\s+|\s+$/g, '').split(/\s+/); 253 | }, 254 | 255 | 256 | // The className parameter (str) can only contain a single class name 257 | hasClass : function (elm, className) { 258 | if (!className) { 259 | return false; 260 | } 261 | return -1 != (' ' + elm.className.replace(/\s+/g, ' ') + ' ').indexOf(' ' + className + ' '); 262 | }, 263 | 264 | 265 | // The className parameter (str) can contain multiple class names separated by whitespace 266 | setClass : function (elm, className) { 267 | var classList = jsc.classNameToList(className); 268 | for (var i = 0; i < classList.length; i += 1) { 269 | if (!jsc.hasClass(elm, classList[i])) { 270 | elm.className += (elm.className ? ' ' : '') + classList[i]; 271 | } 272 | } 273 | }, 274 | 275 | 276 | // The className parameter (str) can contain multiple class names separated by whitespace 277 | unsetClass : function (elm, className) { 278 | var classList = jsc.classNameToList(className); 279 | for (var i = 0; i < classList.length; i += 1) { 280 | var repl = new RegExp( 281 | '^\\s*' + classList[i] + '\\s*|' + 282 | '\\s*' + classList[i] + '\\s*$|' + 283 | '\\s+' + classList[i] + '(\\s+)', 284 | 'g' 285 | ); 286 | elm.className = elm.className.replace(repl, '$1'); 287 | } 288 | }, 289 | 290 | 291 | getStyle : function (elm) { 292 | return window.getComputedStyle ? window.getComputedStyle(elm) : elm.currentStyle; 293 | }, 294 | 295 | 296 | setStyle : (function () { 297 | var helper = document.createElement('div'); 298 | var getSupportedProp = function (names) { 299 | for (var i = 0; i < names.length; i += 1) { 300 | if (names[i] in helper.style) { 301 | return names[i]; 302 | } 303 | } 304 | }; 305 | var props = { 306 | borderRadius: getSupportedProp(['borderRadius', 'MozBorderRadius', 'webkitBorderRadius']), 307 | boxShadow: getSupportedProp(['boxShadow', 'MozBoxShadow', 'webkitBoxShadow']) 308 | }; 309 | return function (elm, prop, value) { 310 | switch (prop.toLowerCase()) { 311 | case 'opacity': 312 | var alphaOpacity = Math.round(parseFloat(value) * 100); 313 | elm.style.opacity = value; 314 | elm.style.filter = 'alpha(opacity=' + alphaOpacity + ')'; 315 | break; 316 | default: 317 | elm.style[props[prop]] = value; 318 | break; 319 | } 320 | }; 321 | })(), 322 | 323 | 324 | setBorderRadius : function (elm, value) { 325 | jsc.setStyle(elm, 'borderRadius', value || '0'); 326 | }, 327 | 328 | 329 | setBoxShadow : function (elm, value) { 330 | jsc.setStyle(elm, 'boxShadow', value || 'none'); 331 | }, 332 | 333 | 334 | getElementPos : function (e, relativeToViewport) { 335 | var x=0, y=0; 336 | var rect = e.getBoundingClientRect(); 337 | x = rect.left; 338 | y = rect.top; 339 | if (!relativeToViewport) { 340 | var viewPos = jsc.getViewPos(); 341 | x += viewPos[0]; 342 | y += viewPos[1]; 343 | } 344 | return [x, y]; 345 | }, 346 | 347 | 348 | getElementSize : function (e) { 349 | return [e.offsetWidth, e.offsetHeight]; 350 | }, 351 | 352 | 353 | // get pointer's X/Y coordinates relative to viewport 354 | getAbsPointerPos : function (e) { 355 | if (!e) { e = window.event; } 356 | var x = 0, y = 0; 357 | if (typeof e.changedTouches !== 'undefined' && e.changedTouches.length) { 358 | // touch devices 359 | x = e.changedTouches[0].clientX; 360 | y = e.changedTouches[0].clientY; 361 | } else if (typeof e.clientX === 'number') { 362 | x = e.clientX; 363 | y = e.clientY; 364 | } 365 | return { x: x, y: y }; 366 | }, 367 | 368 | 369 | // get pointer's X/Y coordinates relative to target element 370 | getRelPointerPos : function (e) { 371 | if (!e) { e = window.event; } 372 | var target = e.target || e.srcElement; 373 | var targetRect = target.getBoundingClientRect(); 374 | 375 | var x = 0, y = 0; 376 | 377 | var clientX = 0, clientY = 0; 378 | if (typeof e.changedTouches !== 'undefined' && e.changedTouches.length) { 379 | // touch devices 380 | clientX = e.changedTouches[0].clientX; 381 | clientY = e.changedTouches[0].clientY; 382 | } else if (typeof e.clientX === 'number') { 383 | clientX = e.clientX; 384 | clientY = e.clientY; 385 | } 386 | 387 | x = clientX - targetRect.left; 388 | y = clientY - targetRect.top; 389 | return { x: x, y: y }; 390 | }, 391 | 392 | 393 | getViewPos : function () { 394 | var doc = document.documentElement; 395 | return [ 396 | (window.pageXOffset || doc.scrollLeft) - (doc.clientLeft || 0), 397 | (window.pageYOffset || doc.scrollTop) - (doc.clientTop || 0) 398 | ]; 399 | }, 400 | 401 | 402 | getViewSize : function () { 403 | var doc = document.documentElement; 404 | return [ 405 | (window.innerWidth || doc.clientWidth), 406 | (window.innerHeight || doc.clientHeight), 407 | ]; 408 | }, 409 | 410 | 411 | redrawPosition : function () { 412 | 413 | if (jsc.picker && jsc.picker.owner) { 414 | var thisObj = jsc.picker.owner; 415 | 416 | var tp, vp; 417 | 418 | if (thisObj.fixed) { 419 | // Fixed elements are positioned relative to viewport, 420 | // therefore we can ignore the scroll offset 421 | tp = jsc.getElementPos(thisObj.targetElement, true); // target pos 422 | vp = [0, 0]; // view pos 423 | } else { 424 | tp = jsc.getElementPos(thisObj.targetElement); // target pos 425 | vp = jsc.getViewPos(); // view pos 426 | } 427 | 428 | var ts = jsc.getElementSize(thisObj.targetElement); // target size 429 | var vs = jsc.getViewSize(); // view size 430 | var ps = jsc.getPickerOuterDims(thisObj); // picker size 431 | var a, b, c; 432 | switch (thisObj.position.toLowerCase()) { 433 | case 'left': a=1; b=0; c=-1; break; 434 | case 'right':a=1; b=0; c=1; break; 435 | case 'top': a=0; b=1; c=-1; break; 436 | default: a=0; b=1; c=1; break; 437 | } 438 | var l = (ts[b]+ps[b])/2; 439 | 440 | // compute picker position 441 | if (!thisObj.smartPosition) { 442 | var pp = [ 443 | tp[a], 444 | tp[b]+ts[b]-l+l*c 445 | ]; 446 | } else { 447 | var pp = [ 448 | -vp[a]+tp[a]+ps[a] > vs[a] ? 449 | (-vp[a]+tp[a]+ts[a]/2 > vs[a]/2 && tp[a]+ts[a]-ps[a] >= 0 ? tp[a]+ts[a]-ps[a] : tp[a]) : 450 | tp[a], 451 | -vp[b]+tp[b]+ts[b]+ps[b]-l+l*c > vs[b] ? 452 | (-vp[b]+tp[b]+ts[b]/2 > vs[b]/2 && tp[b]+ts[b]-l-l*c >= 0 ? tp[b]+ts[b]-l-l*c : tp[b]+ts[b]-l+l*c) : 453 | (tp[b]+ts[b]-l+l*c >= 0 ? tp[b]+ts[b]-l+l*c : tp[b]+ts[b]-l-l*c) 454 | ]; 455 | } 456 | 457 | var x = pp[a]; 458 | var y = pp[b]; 459 | var positionValue = thisObj.fixed ? 'fixed' : 'absolute'; 460 | var contractShadow = 461 | (pp[0] + ps[0] > tp[0] || pp[0] < tp[0] + ts[0]) && 462 | (pp[1] + ps[1] < tp[1] + ts[1]); 463 | 464 | jsc._drawPosition(thisObj, x, y, positionValue, contractShadow); 465 | } 466 | }, 467 | 468 | 469 | _drawPosition : function (thisObj, x, y, positionValue, contractShadow) { 470 | var vShadow = contractShadow ? 0 : thisObj.shadowBlur; // px 471 | 472 | jsc.picker.wrap.style.position = positionValue; 473 | jsc.picker.wrap.style.left = x + 'px'; 474 | jsc.picker.wrap.style.top = y + 'px'; 475 | 476 | jsc.setBoxShadow( 477 | jsc.picker.boxS, 478 | thisObj.shadow ? 479 | new jsc.BoxShadow(0, vShadow, thisObj.shadowBlur, 0, thisObj.shadowColor) : 480 | null); 481 | }, 482 | 483 | 484 | getPickerDims : function (thisObj) { 485 | var displaySlider = !!jsc.getSliderComponent(thisObj); 486 | var dims = [ 487 | 2 * thisObj.insetWidth + 2 * thisObj.padding + thisObj.width + 488 | (displaySlider ? 2 * thisObj.insetWidth + jsc.getPadToSliderPadding(thisObj) + thisObj.sliderSize : 0), 489 | 2 * thisObj.insetWidth + 2 * thisObj.padding + thisObj.height + 490 | (thisObj.closable ? 2 * thisObj.insetWidth + thisObj.padding + thisObj.buttonHeight : 0) 491 | ]; 492 | return dims; 493 | }, 494 | 495 | 496 | getPickerOuterDims : function (thisObj) { 497 | var dims = jsc.getPickerDims(thisObj); 498 | return [ 499 | dims[0] + 2 * thisObj.borderWidth, 500 | dims[1] + 2 * thisObj.borderWidth 501 | ]; 502 | }, 503 | 504 | 505 | getPadToSliderPadding : function (thisObj) { 506 | return Math.max(thisObj.padding, 1.5 * (2 * thisObj.pointerBorderWidth + thisObj.pointerThickness)); 507 | }, 508 | 509 | 510 | getPadYComponent : function (thisObj) { 511 | switch (thisObj.mode.charAt(1).toLowerCase()) { 512 | case 'v': return 'v'; break; 513 | } 514 | return 's'; 515 | }, 516 | 517 | 518 | getSliderComponent : function (thisObj) { 519 | if (thisObj.mode.length > 2) { 520 | switch (thisObj.mode.charAt(2).toLowerCase()) { 521 | case 's': return 's'; break; 522 | case 'v': return 'v'; break; 523 | } 524 | } 525 | return null; 526 | }, 527 | 528 | 529 | onDocumentMouseDown : function (e) { 530 | 531 | if (!e) { e = window.event; } 532 | var target = e.explicitOriginalTarget || e.path[0]; 533 | if (target._jscLinkedInstance) { 534 | if (target._jscLinkedInstance.showOnClick) { 535 | target._jscLinkedInstance.show(); 536 | } 537 | } else if (target._jscControlName) { 538 | jsc.onControlPointerStart(e, target, target._jscControlName, 'mouse'); 539 | } else { 540 | // Mouse is outside the picker controls -> hide the color picker! 541 | if (jsc.picker && jsc.picker.owner) { 542 | jsc.picker.owner.hide(); 543 | } 544 | } 545 | }, 546 | 547 | 548 | onDocumentTouchStart : function (e) { 549 | if (!e) { e = window.event; } 550 | 551 | var target = e.explicitOriginalTarget || e.path[0]; 552 | 553 | if (target._jscLinkedInstance) { 554 | if (target._jscLinkedInstance.showOnClick) { 555 | 556 | target._jscLinkedInstance.show(); 557 | } 558 | } else if (target._jscControlName) { 559 | jsc.onControlPointerStart(e, target, target._jscControlName, 'touch'); 560 | } else { 561 | if (jsc.picker && jsc.picker.owner) { 562 | jsc.picker.owner.hide(); 563 | } 564 | } 565 | }, 566 | 567 | 568 | onWindowResize : function (e) { 569 | jsc.redrawPosition(); 570 | }, 571 | 572 | 573 | onParentScroll : function (e) { 574 | // hide the picker when one of the parent elements is scrolled 575 | if (jsc.picker && jsc.picker.owner) { 576 | jsc.picker.owner.hide(); 577 | } 578 | }, 579 | 580 | 581 | _pointerMoveEvent : { 582 | mouse: 'mousemove', 583 | touch: 'touchmove' 584 | }, 585 | _pointerEndEvent : { 586 | mouse: 'mouseup', 587 | touch: 'touchend' 588 | }, 589 | 590 | 591 | _pointerOrigin : null, 592 | _capturedTarget : null, 593 | 594 | 595 | onControlPointerStart : function (e, target, controlName, pointerType) { 596 | var thisObj = target._jscInstance; 597 | 598 | jsc.preventDefault(e); 599 | jsc.captureTarget(target); 600 | 601 | var registerDragEvents = function (doc, offset) { 602 | jsc.attachGroupEvent('drag', doc, jsc._pointerMoveEvent[pointerType], 603 | jsc.onDocumentPointerMove(e, target, controlName, pointerType, offset)); 604 | jsc.attachGroupEvent('drag', doc, jsc._pointerEndEvent[pointerType], 605 | jsc.onDocumentPointerEnd(e, target, controlName, pointerType)); 606 | }; 607 | 608 | registerDragEvents(document, [0, 0]); 609 | 610 | if (window.parent && window.frameElement) { 611 | var rect = window.frameElement.getBoundingClientRect(); 612 | var ofs = [-rect.left, -rect.top]; 613 | registerDragEvents(window.parent.window.document, ofs); 614 | } 615 | 616 | var abs = jsc.getAbsPointerPos(e); 617 | var rel = jsc.getRelPointerPos(e); 618 | jsc._pointerOrigin = { 619 | x: abs.x - rel.x, 620 | y: abs.y - rel.y 621 | }; 622 | 623 | switch (controlName) { 624 | case 'pad': 625 | // if the slider is at the bottom, move it up 626 | switch (jsc.getSliderComponent(thisObj)) { 627 | case 's': if (thisObj.hsv[1] === 0) { thisObj.fromHSV(null, 100, null); }; break; 628 | case 'v': if (thisObj.hsv[2] === 0) { thisObj.fromHSV(null, null, 100); }; break; 629 | } 630 | jsc.setPad(thisObj, e, 0, 0); 631 | break; 632 | 633 | case 'sld': 634 | jsc.setSld(thisObj, e, 0); 635 | break; 636 | } 637 | 638 | jsc.dispatchFineChange(thisObj); 639 | }, 640 | 641 | 642 | onDocumentPointerMove : function (e, target, controlName, pointerType, offset) { 643 | return function (e) { 644 | var thisObj = target._jscInstance; 645 | switch (controlName) { 646 | case 'pad': 647 | if (!e) { e = window.event; } 648 | jsc.setPad(thisObj, e, offset[0], offset[1]); 649 | jsc.dispatchFineChange(thisObj); 650 | break; 651 | 652 | case 'sld': 653 | if (!e) { e = window.event; } 654 | jsc.setSld(thisObj, e, offset[1]); 655 | jsc.dispatchFineChange(thisObj); 656 | break; 657 | } 658 | } 659 | }, 660 | 661 | 662 | onDocumentPointerEnd : function (e, target, controlName, pointerType) { 663 | return function (e) { 664 | var thisObj = target._jscInstance; 665 | jsc.detachGroupEvents('drag'); 666 | jsc.releaseTarget(); 667 | // Always dispatch changes after detaching outstanding mouse handlers, 668 | // in case some user interaction will occur in user's onchange callback 669 | // that would intrude with current mouse events 670 | jsc.dispatchChange(thisObj); 671 | }; 672 | }, 673 | 674 | 675 | dispatchChange : function (thisObj) { 676 | if (thisObj.valueElement) { 677 | if (jsc.isElementType(thisObj.valueElement, 'input')) { 678 | jsc.fireEvent(thisObj.valueElement, 'change'); 679 | } 680 | } 681 | }, 682 | 683 | 684 | dispatchFineChange : function (thisObj) { 685 | if (thisObj.onFineChange) { 686 | var callback; 687 | if (typeof thisObj.onFineChange === 'string') { 688 | callback = new Function (thisObj.onFineChange); 689 | } else { 690 | callback = thisObj.onFineChange; 691 | } 692 | callback.call(thisObj); 693 | } 694 | }, 695 | 696 | 697 | setPad : function (thisObj, e, ofsX, ofsY) { 698 | var pointerAbs = jsc.getAbsPointerPos(e); 699 | var x = ofsX + pointerAbs.x - jsc._pointerOrigin.x - thisObj.padding - thisObj.insetWidth; 700 | var y = ofsY + pointerAbs.y - jsc._pointerOrigin.y - thisObj.padding - thisObj.insetWidth; 701 | 702 | var xVal = x * (360 / (thisObj.width - 1)); 703 | var yVal = 100 - (y * (100 / (thisObj.height - 1))); 704 | 705 | switch (jsc.getPadYComponent(thisObj)) { 706 | case 's': thisObj.fromHSV(xVal, yVal, null, jsc.leaveSld); break; 707 | case 'v': thisObj.fromHSV(xVal, null, yVal, jsc.leaveSld); break; 708 | } 709 | }, 710 | 711 | 712 | setSld : function (thisObj, e, ofsY) { 713 | var pointerAbs = jsc.getAbsPointerPos(e); 714 | var y = ofsY + pointerAbs.y - jsc._pointerOrigin.y - thisObj.padding - thisObj.insetWidth; 715 | 716 | var yVal = 100 - (y * (100 / (thisObj.height - 1))); 717 | 718 | switch (jsc.getSliderComponent(thisObj)) { 719 | case 's': thisObj.fromHSV(null, yVal, null, jsc.leavePad); break; 720 | case 'v': thisObj.fromHSV(null, null, yVal, jsc.leavePad); break; 721 | } 722 | }, 723 | 724 | 725 | _vmlNS : 'jsc_vml_', 726 | _vmlCSS : 'jsc_vml_css_', 727 | _vmlReady : false, 728 | 729 | 730 | initVML : function () { 731 | if (!jsc._vmlReady) { 732 | // init VML namespace 733 | var doc = document; 734 | if (!doc.namespaces[jsc._vmlNS]) { 735 | doc.namespaces.add(jsc._vmlNS, 'urn:schemas-microsoft-com:vml'); 736 | } 737 | if (!doc.styleSheets[jsc._vmlCSS]) { 738 | var tags = ['shape', 'shapetype', 'group', 'background', 'path', 'formulas', 'handles', 'fill', 'stroke', 'shadow', 'textbox', 'textpath', 'imagedata', 'line', 'polyline', 'curve', 'rect', 'roundrect', 'oval', 'arc', 'image']; 739 | var ss = doc.createStyleSheet(); 740 | ss.owningElement.id = jsc._vmlCSS; 741 | for (var i = 0; i < tags.length; i += 1) { 742 | ss.addRule(jsc._vmlNS + '\\:' + tags[i], 'behavior:url(#default#VML);'); 743 | } 744 | } 745 | jsc._vmlReady = true; 746 | } 747 | }, 748 | 749 | 750 | createPalette : function () { 751 | 752 | var paletteObj = { 753 | elm: null, 754 | draw: null 755 | }; 756 | 757 | if (jsc.isCanvasSupported) { 758 | // Canvas implementation for modern browsers 759 | 760 | var canvas = document.createElement('canvas'); 761 | var ctx = canvas.getContext('2d'); 762 | 763 | var drawFunc = function (width, height, type) { 764 | canvas.width = width; 765 | canvas.height = height; 766 | 767 | ctx.clearRect(0, 0, canvas.width, canvas.height); 768 | 769 | var hGrad = ctx.createLinearGradient(0, 0, canvas.width, 0); 770 | hGrad.addColorStop(0 / 6, '#F00'); 771 | hGrad.addColorStop(1 / 6, '#FF0'); 772 | hGrad.addColorStop(2 / 6, '#0F0'); 773 | hGrad.addColorStop(3 / 6, '#0FF'); 774 | hGrad.addColorStop(4 / 6, '#00F'); 775 | hGrad.addColorStop(5 / 6, '#F0F'); 776 | hGrad.addColorStop(6 / 6, '#F00'); 777 | 778 | ctx.fillStyle = hGrad; 779 | ctx.fillRect(0, 0, canvas.width, canvas.height); 780 | 781 | var vGrad = ctx.createLinearGradient(0, 0, 0, canvas.height); 782 | switch (type.toLowerCase()) { 783 | case 's': 784 | vGrad.addColorStop(0, 'rgba(255,255,255,0)'); 785 | vGrad.addColorStop(1, 'rgba(255,255,255,1)'); 786 | break; 787 | case 'v': 788 | vGrad.addColorStop(0, 'rgba(0,0,0,0)'); 789 | vGrad.addColorStop(1, 'rgba(0,0,0,1)'); 790 | break; 791 | } 792 | ctx.fillStyle = vGrad; 793 | ctx.fillRect(0, 0, canvas.width, canvas.height); 794 | }; 795 | 796 | paletteObj.elm = canvas; 797 | paletteObj.draw = drawFunc; 798 | 799 | } else { 800 | // VML fallback for IE 7 and 8 801 | 802 | jsc.initVML(); 803 | 804 | var vmlContainer = document.createElement('div'); 805 | vmlContainer.style.position = 'relative'; 806 | vmlContainer.style.overflow = 'hidden'; 807 | 808 | var hGrad = document.createElement(jsc._vmlNS + ':fill'); 809 | hGrad.type = 'gradient'; 810 | hGrad.method = 'linear'; 811 | hGrad.angle = '90'; 812 | hGrad.colors = '16.67% #F0F, 33.33% #00F, 50% #0FF, 66.67% #0F0, 83.33% #FF0' 813 | 814 | var hRect = document.createElement(jsc._vmlNS + ':rect'); 815 | hRect.style.position = 'absolute'; 816 | hRect.style.left = -1 + 'px'; 817 | hRect.style.top = -1 + 'px'; 818 | hRect.stroked = false; 819 | hRect.appendChild(hGrad); 820 | vmlContainer.appendChild(hRect); 821 | 822 | var vGrad = document.createElement(jsc._vmlNS + ':fill'); 823 | vGrad.type = 'gradient'; 824 | vGrad.method = 'linear'; 825 | vGrad.angle = '180'; 826 | vGrad.opacity = '0'; 827 | 828 | var vRect = document.createElement(jsc._vmlNS + ':rect'); 829 | vRect.style.position = 'absolute'; 830 | vRect.style.left = -1 + 'px'; 831 | vRect.style.top = -1 + 'px'; 832 | vRect.stroked = false; 833 | vRect.appendChild(vGrad); 834 | vmlContainer.appendChild(vRect); 835 | 836 | var drawFunc = function (width, height, type) { 837 | vmlContainer.style.width = width + 'px'; 838 | vmlContainer.style.height = height + 'px'; 839 | 840 | hRect.style.width = 841 | vRect.style.width = 842 | (width + 1) + 'px'; 843 | hRect.style.height = 844 | vRect.style.height = 845 | (height + 1) + 'px'; 846 | 847 | // Colors must be specified during every redraw, otherwise IE won't display 848 | // a full gradient during a subsequential redraw 849 | hGrad.color = '#F00'; 850 | hGrad.color2 = '#F00'; 851 | 852 | switch (type.toLowerCase()) { 853 | case 's': 854 | vGrad.color = vGrad.color2 = '#FFF'; 855 | break; 856 | case 'v': 857 | vGrad.color = vGrad.color2 = '#000'; 858 | break; 859 | } 860 | }; 861 | 862 | paletteObj.elm = vmlContainer; 863 | paletteObj.draw = drawFunc; 864 | } 865 | 866 | return paletteObj; 867 | }, 868 | 869 | 870 | createSliderGradient : function () { 871 | 872 | var sliderObj = { 873 | elm: null, 874 | draw: null 875 | }; 876 | 877 | if (jsc.isCanvasSupported) { 878 | // Canvas implementation for modern browsers 879 | 880 | var canvas = document.createElement('canvas'); 881 | var ctx = canvas.getContext('2d'); 882 | 883 | var drawFunc = function (width, height, color1, color2) { 884 | canvas.width = width; 885 | canvas.height = height; 886 | 887 | ctx.clearRect(0, 0, canvas.width, canvas.height); 888 | 889 | var grad = ctx.createLinearGradient(0, 0, 0, canvas.height); 890 | grad.addColorStop(0, color1); 891 | grad.addColorStop(1, color2); 892 | 893 | ctx.fillStyle = grad; 894 | ctx.fillRect(0, 0, canvas.width, canvas.height); 895 | }; 896 | 897 | sliderObj.elm = canvas; 898 | sliderObj.draw = drawFunc; 899 | 900 | } else { 901 | // VML fallback for IE 7 and 8 902 | 903 | jsc.initVML(); 904 | 905 | var vmlContainer = document.createElement('div'); 906 | vmlContainer.style.position = 'relative'; 907 | vmlContainer.style.overflow = 'hidden'; 908 | 909 | var grad = document.createElement(jsc._vmlNS + ':fill'); 910 | grad.type = 'gradient'; 911 | grad.method = 'linear'; 912 | grad.angle = '180'; 913 | 914 | var rect = document.createElement(jsc._vmlNS + ':rect'); 915 | rect.style.position = 'absolute'; 916 | rect.style.left = -1 + 'px'; 917 | rect.style.top = -1 + 'px'; 918 | rect.stroked = false; 919 | rect.appendChild(grad); 920 | vmlContainer.appendChild(rect); 921 | 922 | var drawFunc = function (width, height, color1, color2) { 923 | vmlContainer.style.width = width + 'px'; 924 | vmlContainer.style.height = height + 'px'; 925 | 926 | rect.style.width = (width + 1) + 'px'; 927 | rect.style.height = (height + 1) + 'px'; 928 | 929 | grad.color = color1; 930 | grad.color2 = color2; 931 | }; 932 | 933 | sliderObj.elm = vmlContainer; 934 | sliderObj.draw = drawFunc; 935 | } 936 | 937 | return sliderObj; 938 | }, 939 | 940 | 941 | leaveValue : 1<<0, 942 | leaveStyle : 1<<1, 943 | leavePad : 1<<2, 944 | leaveSld : 1<<3, 945 | 946 | 947 | BoxShadow : (function () { 948 | var BoxShadow = function (hShadow, vShadow, blur, spread, color, inset) { 949 | this.hShadow = hShadow; 950 | this.vShadow = vShadow; 951 | this.blur = blur; 952 | this.spread = spread; 953 | this.color = color; 954 | this.inset = !!inset; 955 | }; 956 | 957 | BoxShadow.prototype.toString = function () { 958 | var vals = [ 959 | Math.round(this.hShadow) + 'px', 960 | Math.round(this.vShadow) + 'px', 961 | Math.round(this.blur) + 'px', 962 | Math.round(this.spread) + 'px', 963 | this.color 964 | ]; 965 | if (this.inset) { 966 | vals.push('inset'); 967 | } 968 | return vals.join(' '); 969 | }; 970 | 971 | return BoxShadow; 972 | })(), 973 | 974 | 975 | // 976 | // Usage: 977 | // var myColor = new jscolor( [, ]) 978 | // 979 | 980 | jscolor : function (targetElement, options) { 981 | 982 | // General options 983 | // 984 | this.value = null; // initial HEX color. To change it later, use methods fromString(), fromHSV() and fromRGB() 985 | this.valueElement = targetElement; // element that will be used to display and input the color code 986 | this.styleElement = targetElement; // element that will preview the picked color using CSS backgroundColor 987 | this.required = true; // whether the associated text can be left empty 988 | this.refine = true; // whether to refine the entered color code (e.g. uppercase it and remove whitespace) 989 | this.hash = false; // whether to prefix the HEX color code with # symbol 990 | this.uppercase = true; // whether to show the color code in upper case 991 | this.onFineChange = null; // called instantly every time the color changes (value can be either a function or a string with javascript code) 992 | this.activeClass = 'jscolor-active'; // class to be set to the target element when a picker window is open on it 993 | this.overwriteImportant = false; // whether to overwrite colors of styleElement using !important 994 | this.minS = 0; // min allowed saturation (0 - 100) 995 | this.maxS = 100; // max allowed saturation (0 - 100) 996 | this.minV = 0; // min allowed value (brightness) (0 - 100) 997 | this.maxV = 100; // max allowed value (brightness) (0 - 100) 998 | 999 | // Accessing the picked color 1000 | // 1001 | this.hsv = [0, 0, 100]; // read-only [0-360, 0-100, 0-100] 1002 | this.rgb = [255, 255, 255]; // read-only [0-255, 0-255, 0-255] 1003 | 1004 | // Color Picker options 1005 | // 1006 | this.width = 181; // width of color palette (in px) 1007 | this.height = 101; // height of color palette (in px) 1008 | this.showOnClick = true; // whether to display the color picker when user clicks on its target element 1009 | this.mode = 'HSV'; // HSV | HVS | HS | HV - layout of the color picker controls 1010 | this.position = 'bottom'; // left | right | top | bottom - position relative to the target element 1011 | this.smartPosition = true; // automatically change picker position when there is not enough space for it 1012 | this.sliderSize = 16; // px 1013 | this.crossSize = 8; // px 1014 | this.closable = false; // whether to display the Close button 1015 | this.closeText = 'Close'; 1016 | this.buttonColor = '#000000'; // CSS color 1017 | this.buttonHeight = 18; // px 1018 | this.padding = 12; // px 1019 | this.backgroundColor = '#FFFFFF'; // CSS color 1020 | this.borderWidth = 1; // px 1021 | this.borderColor = '#BBBBBB'; // CSS color 1022 | this.borderRadius = 8; // px 1023 | this.insetWidth = 1; // px 1024 | this.insetColor = '#BBBBBB'; // CSS color 1025 | this.shadow = true; // whether to display shadow 1026 | this.shadowBlur = 15; // px 1027 | this.shadowColor = 'rgba(0,0,0,0.2)'; // CSS color 1028 | this.pointerColor = '#4C4C4C'; // px 1029 | this.pointerBorderColor = '#FFFFFF'; // px 1030 | this.pointerBorderWidth = 1; // px 1031 | this.pointerThickness = 2; // px 1032 | this.zIndex = 1000; 1033 | this.container = null; // where to append the color picker (BODY element by default) 1034 | 1035 | 1036 | for (var opt in options) { 1037 | if (options.hasOwnProperty(opt)) { 1038 | this[opt] = options[opt]; 1039 | } 1040 | } 1041 | 1042 | 1043 | this.hide = function () { 1044 | if (isPickerOwner()) { 1045 | detachPicker(); 1046 | } 1047 | }; 1048 | 1049 | 1050 | this.show = function () { 1051 | drawPicker(); 1052 | }; 1053 | 1054 | 1055 | this.redraw = function () { 1056 | if (isPickerOwner()) { 1057 | drawPicker(); 1058 | } 1059 | }; 1060 | 1061 | 1062 | this.importColor = function () { 1063 | if (!this.valueElement) { 1064 | this.exportColor(); 1065 | } else { 1066 | if (jsc.isElementType(this.valueElement, 'input')) { 1067 | if (!this.refine) { 1068 | if (!this.fromString(this.valueElement.value, jsc.leaveValue)) { 1069 | if (this.styleElement) { 1070 | this.styleElement.style.backgroundImage = this.styleElement._jscOrigStyle.backgroundImage; 1071 | this.styleElement.style.backgroundColor = this.styleElement._jscOrigStyle.backgroundColor; 1072 | this.styleElement.style.color = this.styleElement._jscOrigStyle.color; 1073 | } 1074 | this.exportColor(jsc.leaveValue | jsc.leaveStyle); 1075 | } 1076 | } else if (!this.required && /^\s*$/.test(this.valueElement.value)) { 1077 | this.valueElement.value = ''; 1078 | if (this.styleElement) { 1079 | this.styleElement.style.backgroundImage = this.styleElement._jscOrigStyle.backgroundImage; 1080 | this.styleElement.style.backgroundColor = this.styleElement._jscOrigStyle.backgroundColor; 1081 | this.styleElement.style.color = this.styleElement._jscOrigStyle.color; 1082 | } 1083 | this.exportColor(jsc.leaveValue | jsc.leaveStyle); 1084 | 1085 | } else if (this.fromString(this.valueElement.value)) { 1086 | // managed to import color successfully from the value -> OK, don't do anything 1087 | } else { 1088 | this.exportColor(); 1089 | } 1090 | } else { 1091 | // not an input element -> doesn't have any value 1092 | this.exportColor(); 1093 | } 1094 | } 1095 | }; 1096 | 1097 | 1098 | this.exportColor = function (flags) { 1099 | if (!(flags & jsc.leaveValue) && this.valueElement) { 1100 | var value = this.toString(); 1101 | if (this.uppercase) { value = value.toUpperCase(); } 1102 | if (this.hash) { value = '#' + value; } 1103 | 1104 | if (jsc.isElementType(this.valueElement, 'input')) { 1105 | this.valueElement.value = value; 1106 | } else { 1107 | this.valueElement.innerHTML = value; 1108 | } 1109 | } 1110 | if (!(flags & jsc.leaveStyle)) { 1111 | if (this.styleElement) { 1112 | var bgColor = '#' + this.toString(); 1113 | var fgColor = this.isLight() ? '#000' : '#FFF'; 1114 | 1115 | this.styleElement.style.backgroundImage = 'none'; 1116 | this.styleElement.style.backgroundColor = bgColor; 1117 | this.styleElement.style.color = fgColor; 1118 | 1119 | if (this.overwriteImportant) { 1120 | this.styleElement.setAttribute('style', 1121 | 'background: ' + bgColor + ' !important; ' + 1122 | 'color: ' + fgColor + ' !important;' 1123 | ); 1124 | } 1125 | } 1126 | } 1127 | if (!(flags & jsc.leavePad) && isPickerOwner()) { 1128 | redrawPad(); 1129 | } 1130 | if (!(flags & jsc.leaveSld) && isPickerOwner()) { 1131 | redrawSld(); 1132 | } 1133 | }; 1134 | 1135 | 1136 | // h: 0-360 1137 | // s: 0-100 1138 | // v: 0-100 1139 | // 1140 | this.fromHSV = function (h, s, v, flags) { // null = don't change 1141 | if (h !== null) { 1142 | if (isNaN(h)) { return false; } 1143 | h = Math.max(0, Math.min(360, h)); 1144 | } 1145 | if (s !== null) { 1146 | if (isNaN(s)) { return false; } 1147 | s = Math.max(0, Math.min(100, this.maxS, s), this.minS); 1148 | } 1149 | if (v !== null) { 1150 | if (isNaN(v)) { return false; } 1151 | v = Math.max(0, Math.min(100, this.maxV, v), this.minV); 1152 | } 1153 | 1154 | this.rgb = HSV_RGB( 1155 | h===null ? this.hsv[0] : (this.hsv[0]=h), 1156 | s===null ? this.hsv[1] : (this.hsv[1]=s), 1157 | v===null ? this.hsv[2] : (this.hsv[2]=v) 1158 | ); 1159 | 1160 | this.exportColor(flags); 1161 | }; 1162 | 1163 | 1164 | // r: 0-255 1165 | // g: 0-255 1166 | // b: 0-255 1167 | // 1168 | this.fromRGB = function (r, g, b, flags) { // null = don't change 1169 | if (r !== null) { 1170 | if (isNaN(r)) { return false; } 1171 | r = Math.max(0, Math.min(255, r)); 1172 | } 1173 | if (g !== null) { 1174 | if (isNaN(g)) { return false; } 1175 | g = Math.max(0, Math.min(255, g)); 1176 | } 1177 | if (b !== null) { 1178 | if (isNaN(b)) { return false; } 1179 | b = Math.max(0, Math.min(255, b)); 1180 | } 1181 | 1182 | var hsv = RGB_HSV( 1183 | r===null ? this.rgb[0] : r, 1184 | g===null ? this.rgb[1] : g, 1185 | b===null ? this.rgb[2] : b 1186 | ); 1187 | if (hsv[0] !== null) { 1188 | this.hsv[0] = Math.max(0, Math.min(360, hsv[0])); 1189 | } 1190 | if (hsv[2] !== 0) { 1191 | this.hsv[1] = hsv[1]===null ? null : Math.max(0, this.minS, Math.min(100, this.maxS, hsv[1])); 1192 | } 1193 | this.hsv[2] = hsv[2]===null ? null : Math.max(0, this.minV, Math.min(100, this.maxV, hsv[2])); 1194 | 1195 | // update RGB according to final HSV, as some values might be trimmed 1196 | var rgb = HSV_RGB(this.hsv[0], this.hsv[1], this.hsv[2]); 1197 | this.rgb[0] = rgb[0]; 1198 | this.rgb[1] = rgb[1]; 1199 | this.rgb[2] = rgb[2]; 1200 | 1201 | this.exportColor(flags); 1202 | }; 1203 | 1204 | 1205 | this.fromString = function (str, flags) { 1206 | var m; 1207 | if (m = str.match(/^\W*([0-9A-F]{3}([0-9A-F]{3})?)\W*$/i)) { 1208 | // HEX notation 1209 | // 1210 | 1211 | if (m[1].length === 6) { 1212 | // 6-char notation 1213 | this.fromRGB( 1214 | parseInt(m[1].substr(0,2),16), 1215 | parseInt(m[1].substr(2,2),16), 1216 | parseInt(m[1].substr(4,2),16), 1217 | flags 1218 | ); 1219 | } else { 1220 | // 3-char notation 1221 | this.fromRGB( 1222 | parseInt(m[1].charAt(0) + m[1].charAt(0),16), 1223 | parseInt(m[1].charAt(1) + m[1].charAt(1),16), 1224 | parseInt(m[1].charAt(2) + m[1].charAt(2),16), 1225 | flags 1226 | ); 1227 | } 1228 | return true; 1229 | 1230 | } else if (m = str.match(/^\W*rgba?\(([^)]*)\)\W*$/i)) { 1231 | var params = m[1].split(','); 1232 | var re = /^\s*(\d*)(\.\d+)?\s*$/; 1233 | var mR, mG, mB; 1234 | if ( 1235 | params.length >= 3 && 1236 | (mR = params[0].match(re)) && 1237 | (mG = params[1].match(re)) && 1238 | (mB = params[2].match(re)) 1239 | ) { 1240 | var r = parseFloat((mR[1] || '0') + (mR[2] || '')); 1241 | var g = parseFloat((mG[1] || '0') + (mG[2] || '')); 1242 | var b = parseFloat((mB[1] || '0') + (mB[2] || '')); 1243 | this.fromRGB(r, g, b, flags); 1244 | return true; 1245 | } 1246 | } 1247 | return false; 1248 | }; 1249 | 1250 | 1251 | this.toString = function () { 1252 | return ( 1253 | (0x100 | Math.round(this.rgb[0])).toString(16).substr(1) + 1254 | (0x100 | Math.round(this.rgb[1])).toString(16).substr(1) + 1255 | (0x100 | Math.round(this.rgb[2])).toString(16).substr(1) 1256 | ); 1257 | }; 1258 | 1259 | 1260 | this.toHEXString = function () { 1261 | return '#' + this.toString().toUpperCase(); 1262 | }; 1263 | 1264 | 1265 | this.toRGBString = function () { 1266 | return ('rgb(' + 1267 | Math.round(this.rgb[0]) + ',' + 1268 | Math.round(this.rgb[1]) + ',' + 1269 | Math.round(this.rgb[2]) + ')' 1270 | ); 1271 | }; 1272 | 1273 | 1274 | this.isLight = function () { 1275 | return ( 1276 | 0.213 * this.rgb[0] + 1277 | 0.715 * this.rgb[1] + 1278 | 0.072 * this.rgb[2] > 1279 | 255 / 2 1280 | ); 1281 | }; 1282 | 1283 | 1284 | this._processParentElementsInDOM = function () { 1285 | if (this._linkedElementsProcessed) { return; } 1286 | this._linkedElementsProcessed = true; 1287 | 1288 | var elm = this.targetElement; 1289 | do { 1290 | // If the target element or one of its parent nodes has fixed position, 1291 | // then use fixed positioning instead 1292 | // 1293 | // Note: In Firefox, getComputedStyle returns null in a hidden iframe, 1294 | // that's why we need to check if the returned style object is non-empty 1295 | /* var currStyle = jsc.getStyle(elm); 1296 | if (currStyle && currStyle.position.toLowerCase() === 'fixed') { 1297 | this.fixed = true; 1298 | }*/ 1299 | 1300 | if (elm !== this.targetElement) { 1301 | // Ensure to attach onParentScroll only once to each parent element 1302 | // (multiple targetElements can share the same parent nodes) 1303 | // 1304 | // Note: It's not just offsetParents that can be scrollable, 1305 | // that's why we loop through all parent nodes 1306 | if (!elm._jscEventsAttached) { 1307 | jsc.attachEvent(elm, 'scroll', jsc.onParentScroll); 1308 | elm._jscEventsAttached = true; 1309 | } 1310 | } 1311 | } while ((elm = elm.parentNode) && !jsc.isElementType(elm, 'body')); 1312 | }; 1313 | 1314 | 1315 | // r: 0-255 1316 | // g: 0-255 1317 | // b: 0-255 1318 | // 1319 | // returns: [ 0-360, 0-100, 0-100 ] 1320 | // 1321 | function RGB_HSV (r, g, b) { 1322 | r /= 255; 1323 | g /= 255; 1324 | b /= 255; 1325 | var n = Math.min(Math.min(r,g),b); 1326 | var v = Math.max(Math.max(r,g),b); 1327 | var m = v - n; 1328 | if (m === 0) { return [ null, 0, 100 * v ]; } 1329 | var h = r===n ? 3+(b-g)/m : (g===n ? 5+(r-b)/m : 1+(g-r)/m); 1330 | return [ 1331 | 60 * (h===6?0:h), 1332 | 100 * (m/v), 1333 | 100 * v 1334 | ]; 1335 | } 1336 | 1337 | 1338 | // h: 0-360 1339 | // s: 0-100 1340 | // v: 0-100 1341 | // 1342 | // returns: [ 0-255, 0-255, 0-255 ] 1343 | // 1344 | function HSV_RGB (h, s, v) { 1345 | var u = 255 * (v / 100); 1346 | 1347 | if (h === null) { 1348 | return [ u, u, u ]; 1349 | } 1350 | 1351 | h /= 60; 1352 | s /= 100; 1353 | 1354 | var i = Math.floor(h); 1355 | var f = i%2 ? h-i : 1-(h-i); 1356 | var m = u * (1 - s); 1357 | var n = u * (1 - s * f); 1358 | switch (i) { 1359 | case 6: 1360 | case 0: return [u,n,m]; 1361 | case 1: return [n,u,m]; 1362 | case 2: return [m,u,n]; 1363 | case 3: return [m,n,u]; 1364 | case 4: return [n,m,u]; 1365 | case 5: return [u,m,n]; 1366 | } 1367 | } 1368 | 1369 | 1370 | function detachPicker () { 1371 | jsc.unsetClass(THIS.targetElement, THIS.activeClass); 1372 | jsc.picker.wrap.parentNode.removeChild(jsc.picker.wrap); 1373 | delete jsc.picker.owner; 1374 | } 1375 | 1376 | 1377 | function drawPicker () { 1378 | 1379 | // At this point, when drawing the picker, we know what the parent elements are 1380 | // and we can do all related DOM operations, such as registering events on them 1381 | // or checking their positioning 1382 | THIS._processParentElementsInDOM(); 1383 | 1384 | if (!jsc.picker) { 1385 | jsc.picker = { 1386 | owner: null, 1387 | wrap : document.createElement('div'), 1388 | box : document.createElement('div'), 1389 | boxS : document.createElement('div'), // shadow area 1390 | boxB : document.createElement('div'), // border 1391 | pad : document.createElement('div'), 1392 | padB : document.createElement('div'), // border 1393 | padM : document.createElement('div'), // mouse/touch area 1394 | padPal : jsc.createPalette(), 1395 | cross : document.createElement('div'), 1396 | crossBY : document.createElement('div'), // border Y 1397 | crossBX : document.createElement('div'), // border X 1398 | crossLY : document.createElement('div'), // line Y 1399 | crossLX : document.createElement('div'), // line X 1400 | sld : document.createElement('div'), 1401 | sldB : document.createElement('div'), // border 1402 | sldM : document.createElement('div'), // mouse/touch area 1403 | sldGrad : jsc.createSliderGradient(), 1404 | sldPtrS : document.createElement('div'), // slider pointer spacer 1405 | sldPtrIB : document.createElement('div'), // slider pointer inner border 1406 | sldPtrMB : document.createElement('div'), // slider pointer middle border 1407 | sldPtrOB : document.createElement('div'), // slider pointer outer border 1408 | btn : document.createElement('div'), 1409 | btnT : document.createElement('span') // text 1410 | }; 1411 | 1412 | jsc.picker.pad.appendChild(jsc.picker.padPal.elm); 1413 | jsc.picker.padB.appendChild(jsc.picker.pad); 1414 | jsc.picker.cross.appendChild(jsc.picker.crossBY); 1415 | jsc.picker.cross.appendChild(jsc.picker.crossBX); 1416 | jsc.picker.cross.appendChild(jsc.picker.crossLY); 1417 | jsc.picker.cross.appendChild(jsc.picker.crossLX); 1418 | jsc.picker.padB.appendChild(jsc.picker.cross); 1419 | jsc.picker.box.appendChild(jsc.picker.padB); 1420 | jsc.picker.box.appendChild(jsc.picker.padM); 1421 | 1422 | jsc.picker.sld.appendChild(jsc.picker.sldGrad.elm); 1423 | jsc.picker.sldB.appendChild(jsc.picker.sld); 1424 | jsc.picker.sldB.appendChild(jsc.picker.sldPtrOB); 1425 | jsc.picker.sldPtrOB.appendChild(jsc.picker.sldPtrMB); 1426 | jsc.picker.sldPtrMB.appendChild(jsc.picker.sldPtrIB); 1427 | jsc.picker.sldPtrIB.appendChild(jsc.picker.sldPtrS); 1428 | jsc.picker.box.appendChild(jsc.picker.sldB); 1429 | jsc.picker.box.appendChild(jsc.picker.sldM); 1430 | 1431 | jsc.picker.btn.appendChild(jsc.picker.btnT); 1432 | jsc.picker.box.appendChild(jsc.picker.btn); 1433 | 1434 | jsc.picker.boxB.appendChild(jsc.picker.box); 1435 | jsc.picker.wrap.appendChild(jsc.picker.boxS); 1436 | jsc.picker.wrap.appendChild(jsc.picker.boxB); 1437 | } 1438 | 1439 | var p = jsc.picker; 1440 | 1441 | var displaySlider = !!jsc.getSliderComponent(THIS); 1442 | var dims = jsc.getPickerDims(THIS); 1443 | var crossOuterSize = (2 * THIS.pointerBorderWidth + THIS.pointerThickness + 2 * THIS.crossSize); 1444 | var padToSliderPadding = jsc.getPadToSliderPadding(THIS); 1445 | var borderRadius = Math.min( 1446 | THIS.borderRadius, 1447 | Math.round(THIS.padding * Math.PI)); // px 1448 | var padCursor = 'crosshair'; 1449 | 1450 | // wrap 1451 | p.wrap.style.clear = 'both'; 1452 | p.wrap.style.width = (dims[0] + 2 * THIS.borderWidth) + 'px'; 1453 | p.wrap.style.height = (dims[1] + 2 * THIS.borderWidth) + 'px'; 1454 | p.wrap.style.zIndex = THIS.zIndex; 1455 | 1456 | // picker 1457 | p.box.style.width = dims[0] + 'px'; 1458 | p.box.style.height = dims[1] + 'px'; 1459 | 1460 | p.boxS.style.position = 'absolute'; 1461 | p.boxS.style.left = '0'; 1462 | p.boxS.style.top = '0'; 1463 | p.boxS.style.width = '100%'; 1464 | p.boxS.style.height = '100%'; 1465 | jsc.setBorderRadius(p.boxS, borderRadius + 'px'); 1466 | 1467 | // picker border 1468 | p.boxB.style.position = 'relative'; 1469 | p.boxB.style.border = THIS.borderWidth + 'px solid'; 1470 | p.boxB.style.borderColor = THIS.borderColor; 1471 | p.boxB.style.background = THIS.backgroundColor; 1472 | jsc.setBorderRadius(p.boxB, borderRadius + 'px'); 1473 | 1474 | // IE hack: 1475 | // If the element is transparent, IE will trigger the event on the elements under it, 1476 | // e.g. on Canvas or on elements with border 1477 | p.padM.style.background = 1478 | p.sldM.style.background = 1479 | '#FFF'; 1480 | jsc.setStyle(p.padM, 'opacity', '0'); 1481 | jsc.setStyle(p.sldM, 'opacity', '0'); 1482 | 1483 | // pad 1484 | p.pad.style.position = 'relative'; 1485 | p.pad.style.width = THIS.width + 'px'; 1486 | p.pad.style.height = THIS.height + 'px'; 1487 | 1488 | // pad palettes (HSV and HVS) 1489 | p.padPal.draw(THIS.width, THIS.height, jsc.getPadYComponent(THIS)); 1490 | 1491 | // pad border 1492 | p.padB.style.position = 'absolute'; 1493 | p.padB.style.left = THIS.padding + 'px'; 1494 | p.padB.style.top = THIS.padding + 'px'; 1495 | p.padB.style.border = THIS.insetWidth + 'px solid'; 1496 | p.padB.style.borderColor = THIS.insetColor; 1497 | 1498 | // pad mouse area 1499 | p.padM._jscInstance = THIS; 1500 | p.padM._jscControlName = 'pad'; 1501 | p.padM.style.position = 'absolute'; 1502 | p.padM.style.left = '0'; 1503 | p.padM.style.top = '0'; 1504 | p.padM.style.width = (THIS.padding + 2 * THIS.insetWidth + THIS.width + padToSliderPadding / 2) + 'px'; 1505 | p.padM.style.height = dims[1] + 'px'; 1506 | p.padM.style.cursor = padCursor; 1507 | 1508 | // pad cross 1509 | p.cross.style.position = 'absolute'; 1510 | p.cross.style.left = 1511 | p.cross.style.top = 1512 | '0'; 1513 | p.cross.style.width = 1514 | p.cross.style.height = 1515 | crossOuterSize + 'px'; 1516 | 1517 | // pad cross border Y and X 1518 | p.crossBY.style.position = 1519 | p.crossBX.style.position = 1520 | 'absolute'; 1521 | p.crossBY.style.background = 1522 | p.crossBX.style.background = 1523 | THIS.pointerBorderColor; 1524 | p.crossBY.style.width = 1525 | p.crossBX.style.height = 1526 | (2 * THIS.pointerBorderWidth + THIS.pointerThickness) + 'px'; 1527 | p.crossBY.style.height = 1528 | p.crossBX.style.width = 1529 | crossOuterSize + 'px'; 1530 | p.crossBY.style.left = 1531 | p.crossBX.style.top = 1532 | (Math.floor(crossOuterSize / 2) - Math.floor(THIS.pointerThickness / 2) - THIS.pointerBorderWidth) + 'px'; 1533 | p.crossBY.style.top = 1534 | p.crossBX.style.left = 1535 | '0'; 1536 | 1537 | // pad cross line Y and X 1538 | p.crossLY.style.position = 1539 | p.crossLX.style.position = 1540 | 'absolute'; 1541 | p.crossLY.style.background = 1542 | p.crossLX.style.background = 1543 | THIS.pointerColor; 1544 | p.crossLY.style.height = 1545 | p.crossLX.style.width = 1546 | (crossOuterSize - 2 * THIS.pointerBorderWidth) + 'px'; 1547 | p.crossLY.style.width = 1548 | p.crossLX.style.height = 1549 | THIS.pointerThickness + 'px'; 1550 | p.crossLY.style.left = 1551 | p.crossLX.style.top = 1552 | (Math.floor(crossOuterSize / 2) - Math.floor(THIS.pointerThickness / 2)) + 'px'; 1553 | p.crossLY.style.top = 1554 | p.crossLX.style.left = 1555 | THIS.pointerBorderWidth + 'px'; 1556 | 1557 | // slider 1558 | p.sld.style.overflow = 'hidden'; 1559 | p.sld.style.width = THIS.sliderSize + 'px'; 1560 | p.sld.style.height = THIS.height + 'px'; 1561 | 1562 | // slider gradient 1563 | p.sldGrad.draw(THIS.sliderSize, THIS.height, '#000', '#000'); 1564 | 1565 | // slider border 1566 | p.sldB.style.display = displaySlider ? 'block' : 'none'; 1567 | p.sldB.style.position = 'absolute'; 1568 | p.sldB.style.right = THIS.padding + 'px'; 1569 | p.sldB.style.top = THIS.padding + 'px'; 1570 | p.sldB.style.border = THIS.insetWidth + 'px solid'; 1571 | p.sldB.style.borderColor = THIS.insetColor; 1572 | 1573 | // slider mouse area 1574 | p.sldM._jscInstance = THIS; 1575 | p.sldM._jscControlName = 'sld'; 1576 | p.sldM.style.display = displaySlider ? 'block' : 'none'; 1577 | p.sldM.style.position = 'absolute'; 1578 | p.sldM.style.right = '0'; 1579 | p.sldM.style.top = '0'; 1580 | p.sldM.style.width = (THIS.sliderSize + padToSliderPadding / 2 + THIS.padding + 2 * THIS.insetWidth) + 'px'; 1581 | p.sldM.style.height = dims[1] + 'px'; 1582 | p.sldM.style.cursor = 'default'; 1583 | 1584 | // slider pointer inner and outer border 1585 | p.sldPtrIB.style.border = 1586 | p.sldPtrOB.style.border = 1587 | THIS.pointerBorderWidth + 'px solid ' + THIS.pointerBorderColor; 1588 | 1589 | // slider pointer outer border 1590 | p.sldPtrOB.style.position = 'absolute'; 1591 | p.sldPtrOB.style.left = -(2 * THIS.pointerBorderWidth + THIS.pointerThickness) + 'px'; 1592 | p.sldPtrOB.style.top = '0'; 1593 | 1594 | // slider pointer middle border 1595 | p.sldPtrMB.style.border = THIS.pointerThickness + 'px solid ' + THIS.pointerColor; 1596 | 1597 | // slider pointer spacer 1598 | p.sldPtrS.style.width = THIS.sliderSize + 'px'; 1599 | p.sldPtrS.style.height = sliderPtrSpace + 'px'; 1600 | 1601 | // the Close button 1602 | function setBtnBorder () { 1603 | var insetColors = THIS.insetColor.split(/\s+/); 1604 | var outsetColor = insetColors.length < 2 ? insetColors[0] : insetColors[1] + ' ' + insetColors[0] + ' ' + insetColors[0] + ' ' + insetColors[1]; 1605 | p.btn.style.borderColor = outsetColor; 1606 | } 1607 | p.btn.style.display = THIS.closable ? 'block' : 'none'; 1608 | p.btn.style.position = 'absolute'; 1609 | p.btn.style.left = THIS.padding + 'px'; 1610 | p.btn.style.bottom = THIS.padding + 'px'; 1611 | p.btn.style.padding = '0 15px'; 1612 | p.btn.style.height = THIS.buttonHeight + 'px'; 1613 | p.btn.style.border = THIS.insetWidth + 'px solid'; 1614 | setBtnBorder(); 1615 | p.btn.style.color = THIS.buttonColor; 1616 | p.btn.style.font = '12px sans-serif'; 1617 | p.btn.style.textAlign = 'center'; 1618 | try { 1619 | p.btn.style.cursor = 'pointer'; 1620 | } catch(eOldIE) { 1621 | p.btn.style.cursor = 'hand'; 1622 | } 1623 | p.btn.onmousedown = function () { 1624 | THIS.hide(); 1625 | }; 1626 | p.btnT.style.lineHeight = THIS.buttonHeight + 'px'; 1627 | p.btnT.innerHTML = ''; 1628 | p.btnT.appendChild(document.createTextNode(THIS.closeText)); 1629 | 1630 | // place pointers 1631 | redrawPad(); 1632 | redrawSld(); 1633 | 1634 | // If we are changing the owner without first closing the picker, 1635 | // make sure to first deal with the old owner 1636 | if (jsc.picker.owner && jsc.picker.owner !== THIS) { 1637 | jsc.unsetClass(jsc.picker.owner.targetElement, THIS.activeClass); 1638 | } 1639 | 1640 | // Set the new picker owner 1641 | jsc.picker.owner = THIS; 1642 | 1643 | // The redrawPosition() method needs picker.owner to be set, that's why we call it here, 1644 | // after setting the owner 1645 | if (jsc.isElementType(container, 'body')) { 1646 | jsc.redrawPosition(); 1647 | } else { 1648 | jsc._drawPosition(THIS, 0, 0, 'relative', false); 1649 | } 1650 | 1651 | if (p.wrap.parentNode != container) { 1652 | container.appendChild(p.wrap); 1653 | } 1654 | 1655 | jsc.setClass(THIS.targetElement, THIS.activeClass); 1656 | } 1657 | 1658 | 1659 | function redrawPad () { 1660 | // redraw the pad pointer 1661 | switch (jsc.getPadYComponent(THIS)) { 1662 | case 's': var yComponent = 1; break; 1663 | case 'v': var yComponent = 2; break; 1664 | } 1665 | var x = Math.round((THIS.hsv[0] / 360) * (THIS.width - 1)); 1666 | var y = Math.round((1 - THIS.hsv[yComponent] / 100) * (THIS.height - 1)); 1667 | var crossOuterSize = (2 * THIS.pointerBorderWidth + THIS.pointerThickness + 2 * THIS.crossSize); 1668 | var ofs = -Math.floor(crossOuterSize / 2); 1669 | jsc.picker.cross.style.left = (x + ofs) + 'px'; 1670 | jsc.picker.cross.style.top = (y + ofs) + 'px'; 1671 | 1672 | // redraw the slider 1673 | switch (jsc.getSliderComponent(THIS)) { 1674 | case 's': 1675 | var rgb1 = HSV_RGB(THIS.hsv[0], 100, THIS.hsv[2]); 1676 | var rgb2 = HSV_RGB(THIS.hsv[0], 0, THIS.hsv[2]); 1677 | var color1 = 'rgb(' + 1678 | Math.round(rgb1[0]) + ',' + 1679 | Math.round(rgb1[1]) + ',' + 1680 | Math.round(rgb1[2]) + ')'; 1681 | var color2 = 'rgb(' + 1682 | Math.round(rgb2[0]) + ',' + 1683 | Math.round(rgb2[1]) + ',' + 1684 | Math.round(rgb2[2]) + ')'; 1685 | jsc.picker.sldGrad.draw(THIS.sliderSize, THIS.height, color1, color2); 1686 | break; 1687 | case 'v': 1688 | var rgb = HSV_RGB(THIS.hsv[0], THIS.hsv[1], 100); 1689 | var color1 = 'rgb(' + 1690 | Math.round(rgb[0]) + ',' + 1691 | Math.round(rgb[1]) + ',' + 1692 | Math.round(rgb[2]) + ')'; 1693 | var color2 = '#000'; 1694 | jsc.picker.sldGrad.draw(THIS.sliderSize, THIS.height, color1, color2); 1695 | break; 1696 | } 1697 | } 1698 | 1699 | 1700 | function redrawSld () { 1701 | var sldComponent = jsc.getSliderComponent(THIS); 1702 | if (sldComponent) { 1703 | // redraw the slider pointer 1704 | switch (sldComponent) { 1705 | case 's': var yComponent = 1; break; 1706 | case 'v': var yComponent = 2; break; 1707 | } 1708 | var y = Math.round((1 - THIS.hsv[yComponent] / 100) * (THIS.height - 1)); 1709 | jsc.picker.sldPtrOB.style.top = (y - (2 * THIS.pointerBorderWidth + THIS.pointerThickness) - Math.floor(sliderPtrSpace / 2)) + 'px'; 1710 | } 1711 | } 1712 | 1713 | 1714 | function isPickerOwner () { 1715 | return jsc.picker && jsc.picker.owner === THIS; 1716 | } 1717 | 1718 | 1719 | function blurValue () { 1720 | THIS.importColor(); 1721 | } 1722 | 1723 | 1724 | // Find the target element 1725 | if (typeof targetElement === 'string') { 1726 | var id = targetElement; 1727 | var elm = document.getElementById(id); 1728 | if (elm) { 1729 | this.targetElement = elm; 1730 | } else { 1731 | jsc.warn('Could not find target element with ID \'' + id + '\''); 1732 | } 1733 | } else if (targetElement) { 1734 | this.targetElement = targetElement; 1735 | } else { 1736 | jsc.warn('Invalid target element: \'' + targetElement + '\''); 1737 | } 1738 | 1739 | if (this.targetElement._jscLinkedInstance) { 1740 | jsc.warn('Cannot link jscolor twice to the same element. Skipping.'); 1741 | return; 1742 | } 1743 | this.targetElement._jscLinkedInstance = this; 1744 | 1745 | // Find the value element 1746 | this.valueElement = jsc.fetchElement(this.valueElement); 1747 | // Find the style element 1748 | this.styleElement = jsc.fetchElement(this.styleElement); 1749 | 1750 | var THIS = this; 1751 | var container = 1752 | this.container ? 1753 | jsc.fetchElement(this.container) : 1754 | document.getElementsByTagName('body')[0]; 1755 | var sliderPtrSpace = 3; // px 1756 | 1757 | // For BUTTON elements it's important to stop them from sending the form when clicked 1758 | // (e.g. in Safari) 1759 | if (jsc.isElementType(this.targetElement, 'button')) { 1760 | if (this.targetElement.onclick) { 1761 | var origCallback = this.targetElement.onclick; 1762 | this.targetElement.onclick = function (evt) { 1763 | origCallback.call(this, evt); 1764 | return false; 1765 | }; 1766 | } else { 1767 | this.targetElement.onclick = function () { return false; }; 1768 | } 1769 | } 1770 | 1771 | /* 1772 | var elm = this.targetElement; 1773 | do { 1774 | // If the target element or one of its offsetParents has fixed position, 1775 | // then use fixed positioning instead 1776 | // 1777 | // Note: In Firefox, getComputedStyle returns null in a hidden iframe, 1778 | // that's why we need to check if the returned style object is non-empty 1779 | var currStyle = jsc.getStyle(elm); 1780 | if (currStyle && currStyle.position.toLowerCase() === 'fixed') { 1781 | this.fixed = true; 1782 | } 1783 | 1784 | if (elm !== this.targetElement) { 1785 | // attach onParentScroll so that we can recompute the picker position 1786 | // when one of the offsetParents is scrolled 1787 | if (!elm._jscEventsAttached) { 1788 | jsc.attachEvent(elm, 'scroll', jsc.onParentScroll); 1789 | elm._jscEventsAttached = true; 1790 | } 1791 | } 1792 | } while ((elm = elm.offsetParent) && !jsc.isElementType(elm, 'body')); 1793 | */ 1794 | 1795 | // valueElement 1796 | if (this.valueElement) { 1797 | if (jsc.isElementType(this.valueElement, 'input')) { 1798 | var updateField = function () { 1799 | THIS.fromString(THIS.valueElement.value, jsc.leaveValue); 1800 | jsc.dispatchFineChange(THIS); 1801 | }; 1802 | jsc.attachEvent(this.valueElement, 'keyup', updateField); 1803 | jsc.attachEvent(this.valueElement, 'input', updateField); 1804 | jsc.attachEvent(this.valueElement, 'blur', blurValue); 1805 | this.valueElement.setAttribute('autocomplete', 'off'); 1806 | } 1807 | } 1808 | 1809 | // styleElement 1810 | if (this.styleElement) { 1811 | this.styleElement._jscOrigStyle = { 1812 | backgroundImage : this.styleElement.style.backgroundImage, 1813 | backgroundColor : this.styleElement.style.backgroundColor, 1814 | color : this.styleElement.style.color 1815 | }; 1816 | } 1817 | 1818 | if (this.value) { 1819 | // Try to set the color from the .value option and if unsuccessful, 1820 | // export the current color 1821 | this.fromString(this.value) || this.exportColor(); 1822 | } else { 1823 | this.importColor(); 1824 | } 1825 | } 1826 | 1827 | }; 1828 | 1829 | 1830 | //================================ 1831 | // Public properties and methods 1832 | //================================ 1833 | 1834 | 1835 | // By default, search for all elements with class="jscolor" and install a color picker on them. 1836 | // 1837 | // You can change what class name will be looked for by setting the property jscolor.lookupClass 1838 | // anywhere in your HTML document. To completely disable the automatic lookup, set it to null. 1839 | // 1840 | jsc.jscolor.lookupClass = 'jscolor'; 1841 | 1842 | 1843 | jsc.jscolor.installByClassName = function (className) { 1844 | var inputElms = document.getElementsByTagName('input'); 1845 | var buttonElms = document.getElementsByTagName('button'); 1846 | 1847 | jsc.tryInstallOnElements(inputElms, className); 1848 | jsc.tryInstallOnElements(buttonElms, className); 1849 | }; 1850 | 1851 | 1852 | jsc.register(); 1853 | 1854 | 1855 | return jsc.jscolor; 1856 | 1857 | 1858 | })(); } 1859 | -------------------------------------------------------------------------------- /www/lib/sha256.js: -------------------------------------------------------------------------------- 1 | /* 2 | * A JavaScript implementation of the SHA256 hash function. 3 | * 4 | * FILE: sha256.js 5 | * VERSION: 0.8 6 | * AUTHOR: Christoph Bichlmeier 7 | * 8 | * NOTE: This version is not tested thoroughly! 9 | * 10 | * Copyright (c) 2003, Christoph Bichlmeier 11 | * All rights reserved. 12 | * 13 | * Redistribution and use in source and binary forms, with or without 14 | * modification, are permitted provided that the following conditions 15 | * are met: 16 | * 1. Redistributions of source code must retain the above copyright 17 | * notice, this list of conditions and the following disclaimer. 18 | * 2. Redistributions in binary form must reproduce the above copyright 19 | * notice, this list of conditions and the following disclaimer in the 20 | * documentation and/or other materials provided with the distribution. 21 | * 3. Neither the name of the copyright holder nor the names of contributors 22 | * may be used to endorse or promote products derived from this software 23 | * without specific prior written permission. 24 | * 25 | * ====================================================================== 26 | * 27 | * THIS SOFTWARE IS PROVIDED BY THE AUTHORS ''AS IS'' AND ANY EXPRESS 28 | * OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED 29 | * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 30 | * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE 31 | * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 32 | * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 33 | * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR 34 | * BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, 35 | * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE 36 | * OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, 37 | * EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 38 | */ 39 | 40 | /* SHA256 logical functions */ 41 | function rotateRight(n,x) { 42 | return ((x >>> n) | (x << (32 - n))); 43 | } 44 | function choice(x,y,z) { 45 | return ((x & y) ^ (~x & z)); 46 | } 47 | function majority(x,y,z) { 48 | return ((x & y) ^ (x & z) ^ (y & z)); 49 | } 50 | function sha256_Sigma0(x) { 51 | return (rotateRight(2, x) ^ rotateRight(13, x) ^ rotateRight(22, x)); 52 | } 53 | function sha256_Sigma1(x) { 54 | return (rotateRight(6, x) ^ rotateRight(11, x) ^ rotateRight(25, x)); 55 | } 56 | function sha256_sigma0(x) { 57 | return (rotateRight(7, x) ^ rotateRight(18, x) ^ (x >>> 3)); 58 | } 59 | function sha256_sigma1(x) { 60 | return (rotateRight(17, x) ^ rotateRight(19, x) ^ (x >>> 10)); 61 | } 62 | function sha256_expand(W, j) { 63 | return (W[j&0x0f] += sha256_sigma1(W[(j+14)&0x0f]) + W[(j+9)&0x0f] + 64 | sha256_sigma0(W[(j+1)&0x0f])); 65 | } 66 | 67 | /* Hash constant words K: */ 68 | var K256 = new Array( 69 | 0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, 70 | 0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5, 71 | 0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3, 72 | 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174, 73 | 0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc, 74 | 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da, 75 | 0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, 76 | 0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967, 77 | 0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13, 78 | 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85, 79 | 0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, 80 | 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070, 81 | 0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, 82 | 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3, 83 | 0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208, 84 | 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2 85 | ); 86 | 87 | /* global arrays */ 88 | var ihash, count, buffer; 89 | var sha256_hex_digits = "0123456789abcdef"; 90 | 91 | /* Add 32-bit integers with 16-bit operations (bug in some JS-interpreters: 92 | overflow) */ 93 | function safe_add(x, y) 94 | { 95 | var lsw = (x & 0xffff) + (y & 0xffff); 96 | var msw = (x >> 16) + (y >> 16) + (lsw >> 16); 97 | return (msw << 16) | (lsw & 0xffff); 98 | } 99 | 100 | /* Initialise the SHA256 computation */ 101 | function sha256_init() { 102 | ihash = new Array(8); 103 | count = new Array(2); 104 | buffer = new Array(64); 105 | count[0] = count[1] = 0; 106 | ihash[0] = 0x6a09e667; 107 | ihash[1] = 0xbb67ae85; 108 | ihash[2] = 0x3c6ef372; 109 | ihash[3] = 0xa54ff53a; 110 | ihash[4] = 0x510e527f; 111 | ihash[5] = 0x9b05688c; 112 | ihash[6] = 0x1f83d9ab; 113 | ihash[7] = 0x5be0cd19; 114 | } 115 | 116 | /* Transform a 512-bit message block */ 117 | function sha256_transform() { 118 | var a, b, c, d, e, f, g, h, T1, T2; 119 | var W = new Array(16); 120 | 121 | /* Initialize registers with the previous intermediate value */ 122 | a = ihash[0]; 123 | b = ihash[1]; 124 | c = ihash[2]; 125 | d = ihash[3]; 126 | e = ihash[4]; 127 | f = ihash[5]; 128 | g = ihash[6]; 129 | h = ihash[7]; 130 | 131 | /* make 32-bit words */ 132 | for(var i=0; i<16; i++) 133 | W[i] = ((buffer[(i<<2)+3]) | (buffer[(i<<2)+2] << 8) | (buffer[(i<<2)+1] 134 | << 16) | (buffer[i<<2] << 24)); 135 | 136 | for(var j=0; j<64; j++) { 137 | T1 = h + sha256_Sigma1(e) + choice(e, f, g) + K256[j]; 138 | if(j < 16) T1 += W[j]; 139 | else T1 += sha256_expand(W, j); 140 | T2 = sha256_Sigma0(a) + majority(a, b, c); 141 | h = g; 142 | g = f; 143 | f = e; 144 | e = safe_add(d, T1); 145 | d = c; 146 | c = b; 147 | b = a; 148 | a = safe_add(T1, T2); 149 | } 150 | 151 | /* Compute the current intermediate hash value */ 152 | ihash[0] += a; 153 | ihash[1] += b; 154 | ihash[2] += c; 155 | ihash[3] += d; 156 | ihash[4] += e; 157 | ihash[5] += f; 158 | ihash[6] += g; 159 | ihash[7] += h; 160 | } 161 | 162 | /* Read the next chunk of data and update the SHA256 computation */ 163 | function sha256_update(data, inputLen) { 164 | var i, index, curpos = 0; 165 | /* Compute number of bytes mod 64 */ 166 | index = ((count[0] >> 3) & 0x3f); 167 | var remainder = (inputLen & 0x3f); 168 | 169 | /* Update number of bits */ 170 | if ((count[0] += (inputLen << 3)) < (inputLen << 3)) count[1]++; 171 | count[1] += (inputLen >> 29); 172 | 173 | /* Transform as many times as possible */ 174 | for(i=0; i+63> 3) & 0x3f); 189 | buffer[index++] = 0x80; 190 | if(index <= 56) { 191 | for(var i=index; i<56; i++) 192 | buffer[i] = 0; 193 | } else { 194 | for(var i=index; i<64; i++) 195 | buffer[i] = 0; 196 | sha256_transform(); 197 | for(var i=0; i<56; i++) 198 | buffer[i] = 0; 199 | } 200 | buffer[56] = (count[1] >>> 24) & 0xff; 201 | buffer[57] = (count[1] >>> 16) & 0xff; 202 | buffer[58] = (count[1] >>> 8) & 0xff; 203 | buffer[59] = count[1] & 0xff; 204 | buffer[60] = (count[0] >>> 24) & 0xff; 205 | buffer[61] = (count[0] >>> 16) & 0xff; 206 | buffer[62] = (count[0] >>> 8) & 0xff; 207 | buffer[63] = count[0] & 0xff; 208 | sha256_transform(); 209 | } 210 | 211 | /* Split the internal hash values into an array of bytes */ 212 | function sha256_encode_bytes() { 213 | var j=0; 214 | var output = new Array(32); 215 | for(var i=0; i<8; i++) { 216 | output[j++] = ((ihash[i] >>> 24) & 0xff); 217 | output[j++] = ((ihash[i] >>> 16) & 0xff); 218 | output[j++] = ((ihash[i] >>> 8) & 0xff); 219 | output[j++] = (ihash[i] & 0xff); 220 | } 221 | return output; 222 | } 223 | 224 | /* Get the internal hash as a hex string */ 225 | function sha256_encode_hex() { 226 | var output = new String(); 227 | for(var i=0; i<8; i++) { 228 | for(var j=28; j>=0; j-=4) 229 | output += sha256_hex_digits.charAt((ihash[i] >>> j) & 0x0f); 230 | } 231 | return output; 232 | } 233 | 234 | /* Main function: returns a hex string representing the SHA256 value of the 235 | given data */ 236 | function sha256_digest(data) { 237 | sha256_init(); 238 | sha256_update(data, data.length); 239 | sha256_final(); 240 | return sha256_encode_hex(); 241 | } 242 | 243 | /* test if the JS-interpreter is working properly */ 244 | function sha256_self_test() 245 | { 246 | return sha256_digest("message digest") == 247 | "f7846f55cf23e14eebeab5b4e1550cad5b509e3348fbc4efa3a1413d393cb650"; 248 | } 249 | 250 | 251 | --------------------------------------------------------------------------------