Initial commit
This commit is contained in:
61
.gitignore
vendored
Normal file
61
.gitignore
vendored
Normal file
@ -0,0 +1,61 @@
|
||||
# Logs
|
||||
logs
|
||||
*.log
|
||||
npm-debug.log*
|
||||
yarn-debug.log*
|
||||
yarn-error.log*
|
||||
|
||||
# Runtime data
|
||||
pids
|
||||
*.pid
|
||||
*.seed
|
||||
*.pid.lock
|
||||
|
||||
# Directory for instrumented libs generated by jscoverage/JSCover
|
||||
lib-cov
|
||||
|
||||
# Coverage directory used by tools like istanbul
|
||||
coverage
|
||||
|
||||
# nyc test coverage
|
||||
.nyc_output
|
||||
|
||||
# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
|
||||
.grunt
|
||||
|
||||
# Bower dependency directory (https://bower.io/)
|
||||
bower_components
|
||||
|
||||
# node-waf configuration
|
||||
.lock-wscript
|
||||
|
||||
# Compiled binary addons (http://nodejs.org/api/addons.html)
|
||||
build/Release
|
||||
|
||||
# Dependency directories
|
||||
node_modules/
|
||||
jspm_packages/
|
||||
|
||||
# Typescript v1 declaration files
|
||||
typings/
|
||||
|
||||
# Optional npm cache directory
|
||||
.npm
|
||||
|
||||
# Optional eslint cache
|
||||
.eslintcache
|
||||
|
||||
# Optional REPL history
|
||||
.node_repl_history
|
||||
|
||||
# Output of 'npm pack'
|
||||
*.tgz
|
||||
|
||||
# Yarn Integrity file
|
||||
.yarn-integrity
|
||||
|
||||
# dotenv environment variables file
|
||||
.env
|
||||
|
||||
# package lock
|
||||
package-lock.json
|
21
LICENSE
Normal file
21
LICENSE
Normal file
@ -0,0 +1,21 @@
|
||||
MIT License
|
||||
|
||||
Copyright (c) 2018
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in all
|
||||
copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
||||
SOFTWARE.
|
167
MMM-ScreenLogic.js
Normal file
167
MMM-ScreenLogic.js
Normal file
@ -0,0 +1,167 @@
|
||||
poolData = {};
|
||||
|
||||
Module.register("MMM-ScreenLogic",{
|
||||
defaults: {
|
||||
showPoolTemp: true,
|
||||
showSpaTemp: true,
|
||||
showPH: true,
|
||||
showOrp: true,
|
||||
showSaltLevel: true,
|
||||
showSaturation: true,
|
||||
colored: true,
|
||||
coldTemp: 84,
|
||||
hotTemp: 90,
|
||||
columns: 3,
|
||||
contentClass: "light"
|
||||
},
|
||||
|
||||
start: function() {
|
||||
this.sendSocketNotification('SCREENLOGIC_CONFIG', this.config);
|
||||
this.sendSocketNotification('SCREENLOGIC_UPDATE');
|
||||
},
|
||||
|
||||
getStyles: function() {
|
||||
return ["screenlogic.css"];
|
||||
},
|
||||
|
||||
getDom: function() {
|
||||
if (!poolData.status) {
|
||||
var wrapper = document.createElement("div");
|
||||
wrapper.innerHTML = 'Loading...';
|
||||
wrapper.className += "dimmed light small";
|
||||
|
||||
return wrapper;
|
||||
} else {
|
||||
var table = document.createElement('table');
|
||||
table.className = "small";
|
||||
if (this.config.colored) {
|
||||
table.className += " colored";
|
||||
}
|
||||
|
||||
var contents = [];
|
||||
|
||||
var row = document.createElement('tr');
|
||||
table.appendChild(row);
|
||||
|
||||
if (this.config.showPoolTemp) {
|
||||
var className = "";
|
||||
if (poolData.status.currentTemp[0] <= this.config.coldTemp) {
|
||||
className += " cold-temp";
|
||||
} else if (poolData.status.currentTemp[0] >= this.config.hotTemp) {
|
||||
className += " hot-temp";
|
||||
}
|
||||
|
||||
contents.push({
|
||||
header: "Pool temp",
|
||||
data: poolData.status.currentTemp[0] + "°" + (!isPoolActive(poolData.status) ? " (last)" : ""),
|
||||
class: this.config.contentClass + className
|
||||
});
|
||||
}
|
||||
if (this.config.showSpaTemp) {
|
||||
var className = "";
|
||||
if (poolData.status.currentTemp[1] <= this.config.coldTemp) {
|
||||
className = " cold-temp";
|
||||
} else if (poolData.status.currentTemp[1] >= this.config.hotTemp) {
|
||||
className = " hot-temp";
|
||||
}
|
||||
|
||||
contents.push({
|
||||
header: "Spa temp",
|
||||
data: poolData.status.currentTemp[1] + "°" + (!isSpaActive(poolData.status) ? " (last)" : ""),
|
||||
class: this.config.contentClass + className
|
||||
});
|
||||
}
|
||||
if (this.config.showPH) {
|
||||
contents.push({
|
||||
header: "pH",
|
||||
data: poolData.status.pH,
|
||||
class: this.config.contentClass
|
||||
});
|
||||
}
|
||||
if (this.config.showOrp) {
|
||||
contents.push({
|
||||
header: "ORP",
|
||||
data: poolData.status.orp,
|
||||
class: this.config.contentClass
|
||||
});
|
||||
}
|
||||
if (this.config.showSaltLevel) {
|
||||
contents.push({
|
||||
header: "Salt PPM",
|
||||
data: poolData.status.saltPPM,
|
||||
class: this.config.contentClass
|
||||
});
|
||||
}
|
||||
if (this.config.showSaturation) {
|
||||
contents.push({
|
||||
header: "Saturation",
|
||||
data: poolData.status.saturation,
|
||||
class: this.config.contentClass
|
||||
});
|
||||
}
|
||||
|
||||
var headerRow = document.createElement('tr');
|
||||
var contentRow = document.createElement('tr');
|
||||
table.appendChild(headerRow);
|
||||
table.appendChild(contentRow);
|
||||
|
||||
var cols = 0;
|
||||
for (var item in contents) {
|
||||
var headerCell = document.createElement('th');
|
||||
headerCell.innerHTML = contents[item].header;
|
||||
headerRow.appendChild(headerCell);
|
||||
|
||||
var contentCell = document.createElement('td');
|
||||
contentCell.innerHTML = contents[item].data;
|
||||
contentCell.className = contents[item].class;
|
||||
contentRow.appendChild(contentCell);
|
||||
|
||||
cols++;
|
||||
if (cols % this.config.columns === 0) {
|
||||
headerRow = document.createElement('tr');
|
||||
contentRow = document.createElement('tr');
|
||||
table.appendChild(headerRow);
|
||||
table.appendChild(contentRow);
|
||||
}
|
||||
}
|
||||
|
||||
return table;
|
||||
}
|
||||
},
|
||||
|
||||
socketNotificationReceived: function(notification, payload) {
|
||||
if (notification === 'SCREENLOGIC_RESULT') {
|
||||
poolData = payload;
|
||||
this.updateDom();
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
const SPA_CIRCUIT_ID = 500;
|
||||
const POOL_CIRCUIT_ID = 505;
|
||||
|
||||
function isPoolActive(status) {
|
||||
for (var i = 0; i < status.circuitArray.length; i++) {
|
||||
if (status.circuitArray[i].id === POOL_CIRCUIT_ID) {
|
||||
return status.circuitArray[i].state === 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function hasSpa(status) {
|
||||
for (var i = 0; i < status.circuitArray.length; i++) {
|
||||
if (status.circuitArray[i].id === SPA_CIRCUIT_ID) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
function isSpaActive(status) {
|
||||
for (var i = 0; i < status.circuitArray.length; i++) {
|
||||
if (status.circuitArray[i].id === SPA_CIRCUIT_ID) {
|
||||
return status.circuitArray[i].state === 1;
|
||||
}
|
||||
}
|
||||
}
|
53
README.md
Normal file
53
README.md
Normal file
@ -0,0 +1,53 @@
|
||||
# MMM-ScreenLogic
|
||||
A <a href="https://github.com/MichMich/MagicMirror">MagicMirror²</a> module used to get real-time values of crypto currencies. Tested with MagicMirror² v2.2.2 server, Chrome 65 on Windows 10 and Midori 0.4.3 on a Raspberry Pi Zero W with Raspbian Jessie. Tested with a Pentair ScreenLogic system running version 5.2 Build 736.0 Rel.
|
||||
|
||||
## Installation
|
||||
1. Navigate into your MagicMirror's `modules` folder and execute `git clone https://github.com/parnic/MMM-ScreenLogic.git`.
|
||||
2. `cd MMM-ScreenLogic`
|
||||
3. Execute `npm install` to install the node dependencies.
|
||||
4. Add the module inside `config.js` placing it where you prefer.
|
||||
|
||||
## Config
|
||||
|Option|Type|Description|Default|
|
||||
|---|---|---|---|
|
||||
|`serverAddress`|String|The IPv4 address of a ScreenLogic unit to connect to. If not set, the system will search for a unit to connect to. If set, `serverPort` must also be set.| |
|
||||
|`serverPort`|Integer|The port of a ScreenLogic unit to connect to (usually 80). If not set, the system will search for a unit to connect to. If set, `serverAddress` must also be set.| |
|
||||
|`showPoolTemp`|Boolean|Whether you'd like to show pool temperature or not.|`true`|
|
||||
|`showSpaTemp`|Boolean|Whether you'd like to show spa temperature or not.|`true`|
|
||||
|`showPH`|Boolean|Whether you'd like to show pH level or not.|`true`|
|
||||
|`showOrp`|Boolean|Whether you'd like to show ORP level or not.|`true`|
|
||||
|`showSaltLevel`|Boolean|Whether you'd like to show salt level (in PPM) or not.|`true`|
|
||||
|`showSaturation`|Boolean|Whether you'd like to show saturation/balance or not.|`true`|
|
||||
|`colored`|Boolean|Whether you'd like colored output or not.|`true`|
|
||||
|`coldTemp`|Integer|Show the temperature colored blue if it's at or below this level for pool/spa (requires option `colored`). This is in whatever scale your system is set to (Fahrenheit/Celsius).|`84`|
|
||||
|`hotTemp`|Integer|Show the temperature colored red if it's at or above this level for pool/spa (requires option `colored`). This is in whatever scale your system is set to (Fahrenheit/Celsius).|`90`|
|
||||
|`columns`|Integer|How many columns to use to display the data before starting a new row.|`3`|
|
||||
|`contentClass`|String|The CSS class used to display content values (beneath the header).|`"light"`|
|
||||
|
||||
Here is an example of an entry in config.js
|
||||
```
|
||||
{
|
||||
module: 'MMM-ScreenLogic',
|
||||
header: 'Pool info',
|
||||
position: 'top_left',
|
||||
config: {
|
||||
showSpaTemp: false,
|
||||
columns: 2,
|
||||
contentClass: 'thin'
|
||||
}
|
||||
},
|
||||
```
|
||||
|
||||
## Screenshot
|
||||
#### With color
|
||||

|
||||
|
||||
## Notes
|
||||
Pull requests are very welcome! If you'd like to see any additional functionality, don't hesitate to let me know.
|
||||
|
||||
This module only works with ScreenLogic controllers on the local network via either a UDP broadcast on 255.255.255.255 or a direct connection if you've specified an address and port in the configuration.
|
||||
|
||||
The data is updated every 30 minutes.
|
||||
|
||||
## Libraries
|
||||
This uses a Node.JS library I created for interfacing with ScreenLogic controllers over the network: <a href="https://github.com/parnic/node-screenlogic">node-screenlogic</a>, so feel free to check that out for more information.
|
66
node_helper.js
Normal file
66
node_helper.js
Normal file
@ -0,0 +1,66 @@
|
||||
var NodeHelper = require('node_helper');
|
||||
var lastResult = {};
|
||||
|
||||
module.exports = NodeHelper.create({
|
||||
start: function() {
|
||||
var self = this;
|
||||
setInterval(function() {
|
||||
self.doUpdate()
|
||||
}, 30 * 60 * 1000);
|
||||
},
|
||||
|
||||
doUpdate: function() {
|
||||
var self = this;
|
||||
getPoolData(this.config, function(poolData) {
|
||||
lastResult = poolData;
|
||||
self.sendSocketNotification('SCREENLOGIC_RESULT', poolData);
|
||||
});
|
||||
},
|
||||
|
||||
socketNotificationReceived: function(notification, payload) {
|
||||
if (notification === 'SCREENLOGIC_CONFIG') {
|
||||
this.config = payload;
|
||||
}
|
||||
if (notification === 'SCREENLOGIC_UPDATE') {
|
||||
this.doUpdate();
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
const ScreenLogic = require('node-screenlogic');
|
||||
|
||||
function getPoolData(config, cb) {
|
||||
if (typeof config === 'undefined' || !config.serverAddress || !config.serverPort) {
|
||||
findServer(cb);
|
||||
} else {
|
||||
populateSystemData(new ScreenLogic.UnitConnection(config.serverPort, config.serverAddress), cb);
|
||||
}
|
||||
}
|
||||
|
||||
function findServer(cb) {
|
||||
var finder = new ScreenLogic.FindUnits();
|
||||
finder.on('serverFound', function(server) {
|
||||
finder.close();
|
||||
populateSystemData(new ScreenLogic.UnitConnection(server), cb);
|
||||
});
|
||||
|
||||
finder.search();
|
||||
}
|
||||
|
||||
function populateSystemData(unit, cb) {
|
||||
var poolData = {};
|
||||
|
||||
unit.on('loggedIn', function() {
|
||||
unit.getControllerConfig();
|
||||
}).on('controllerConfig', function(config) {
|
||||
poolData.degStr = config.degC ? 'C' : 'F';
|
||||
unit.getPoolStatus();
|
||||
}).on('poolStatus', function(status) {
|
||||
poolData.status = status;
|
||||
|
||||
unit.close();
|
||||
cb(poolData);
|
||||
});
|
||||
|
||||
unit.connect();
|
||||
}
|
12
package.json
Normal file
12
package.json
Normal file
@ -0,0 +1,12 @@
|
||||
{
|
||||
"name": "magic-mirror-module-screenlogic",
|
||||
"version": "1.0.0",
|
||||
"description": "Show data from Pentair ScreenLogic systems",
|
||||
"main": "MMM-ScreenLogic.js",
|
||||
"author": "Chris Pickett",
|
||||
"repository": "https://github.com/parnic/MMM-ScreenLogic.git",
|
||||
"license": "MIT",
|
||||
"dependencies": {
|
||||
"node-screenlogic": "~1.0.1"
|
||||
}
|
||||
}
|
7
screenlogic.css
Normal file
7
screenlogic.css
Normal file
@ -0,0 +1,7 @@
|
||||
.MMM-ScreenLogic table.colored .cold-temp {
|
||||
color: #BCDDFF;
|
||||
}
|
||||
|
||||
.MMM-ScreenLogic table.colored .hot-temp {
|
||||
color: #FF8E99;
|
||||
}
|
BIN
screenshot.png
Normal file
BIN
screenshot.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 13 KiB |
Reference in New Issue
Block a user