444 lines
18 KiB
HTML
444 lines
18 KiB
HTML
<!DOCTYPE html>
|
|
<html lang='en'>
|
|
<head>
|
|
<meta http-equiv='Content-Type' content='text/html; charset=windows-1252'>
|
|
|
|
<title>Sprinkler</title>
|
|
<meta name='viewport' content='width=device-width'>
|
|
<meta name='apple-mobile-web-app-capable' content='yes'>
|
|
<meta name='apple-mobile-web-app-status-bar-style' content='black'>
|
|
<style>
|
|
:root {
|
|
--diameter: 30px;
|
|
}
|
|
html {
|
|
}
|
|
body {
|
|
font-family: 'HelveticaNeue-Light', 'Helvetica Neue Light', 'Helvetica Neue', Helvetica, Arial, 'Lucida Grande', sans-serif;
|
|
font-weight: 300;
|
|
background-color: #f1f1f1;
|
|
color: #000000;
|
|
margin: 0 !important;
|
|
padding: 0 !important;
|
|
}
|
|
table,
|
|
th,
|
|
td {
|
|
border-collapse: collapse;
|
|
padding: 10px;
|
|
text-align: left;
|
|
border-bottom: 1px solid rgb(189, 189, 189);
|
|
width: 100%;
|
|
}
|
|
th {
|
|
width: 100vw;
|
|
padding: 15px;
|
|
text-align: center;
|
|
background-color: #f1f1f1;
|
|
}
|
|
tr {
|
|
background-color: white;
|
|
color: black;
|
|
}
|
|
.button {
|
|
background-color: white;
|
|
border-radius: 15px;
|
|
-webkit-border-radius: 15px;
|
|
-moz-border-radius: 15px;
|
|
height: 30px
|
|
}
|
|
.opaque {
|
|
top: 0px;
|
|
left: 0px;
|
|
background-color: black;
|
|
filter: alpha(opacity=20);
|
|
opacity: 0.2;
|
|
}
|
|
.show {
|
|
top: 0px;
|
|
left: 0px;
|
|
}
|
|
.message {
|
|
position: fixed;
|
|
top: 0px;
|
|
left: 0px;
|
|
height: 100%;
|
|
z-index: 1000;
|
|
display: flex;
|
|
justify-content: center;
|
|
align-items: center;
|
|
width: 100vw;
|
|
}
|
|
.message-text {
|
|
background-color: black;
|
|
color: white;
|
|
height: auto;
|
|
padding: 15px;
|
|
font-size: 20px;
|
|
text-align: center;
|
|
position: relative;
|
|
border-radius: var(--diameter);
|
|
}
|
|
.option_button {
|
|
position: fixed;
|
|
top: 2px;
|
|
left: 2px;
|
|
z-index: 900;
|
|
}
|
|
.options {
|
|
top: 0px;
|
|
left: 0px;
|
|
z-index: 800;
|
|
display: none;
|
|
}
|
|
.input-text {
|
|
font-size: 16px;
|
|
}
|
|
.input-ios {
|
|
font-size: 16px;
|
|
}
|
|
|
|
.status {
|
|
font-size: 14px;
|
|
}
|
|
.small-note {
|
|
font-size: 12px;
|
|
}
|
|
.btn-toggle {
|
|
position: absolute;
|
|
margin-left: -9999px;
|
|
visibility: hidden;
|
|
}
|
|
.btn-toggle+label {
|
|
display: block;
|
|
position: relative;
|
|
cursor: pointer;
|
|
outline: none;
|
|
user-select: none;
|
|
}
|
|
input.btn-toggle-round+label {
|
|
padding: 2px;
|
|
width: calc(2 * var(--diameter));
|
|
height: var(--diameter);
|
|
background-color: #dddddd;
|
|
border-radius: var(--diameter);
|
|
}
|
|
input.btn-toggle-round+label:before,
|
|
input.btn-toggle-round+label:after {
|
|
display: block;
|
|
position: absolute;
|
|
top: 1px;
|
|
left: 1px;
|
|
bottom: 1px;
|
|
content: '';
|
|
}
|
|
input.btn-toggle-round+label:before {
|
|
right: 1px;
|
|
background-color: #f1f1f1;
|
|
border-radius: var(--diameter);
|
|
transition: background 0.4s;
|
|
}
|
|
input.btn-toggle-round+label:after {
|
|
width: var(--diameter);
|
|
background-color: #ffffff;
|
|
border-radius: 100%;
|
|
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.3);
|
|
transition: margin 0.4s;
|
|
}
|
|
input.btn-toggle-round:checked+label:before {
|
|
background-color: rgb(76, 217, 100);
|
|
}
|
|
input.btn-toggle-round:checked+label:after {
|
|
margin-left: var(--diameter);
|
|
}
|
|
</style>
|
|
|
|
<script type='text/javascript'>
|
|
var _poller;
|
|
var _showingConfig=false;
|
|
var _width;
|
|
function setSizeSpecifics() {
|
|
_width = Math.max(document.documentElement.clientWidth, window.innerWidth || 0);
|
|
|
|
/*
|
|
Future change element sizes here if needed
|
|
*/
|
|
}
|
|
function addHeadIcon(rel) {
|
|
var image = '';
|
|
var link = document.createElement('link');
|
|
link.rel = rel;
|
|
link.type= 'image/png';
|
|
link.href = image;
|
|
document.head.appendChild(link);
|
|
}
|
|
function headIcons() {
|
|
addHeadIcon('shortcut icon');
|
|
addHeadIcon('apple-touch-icon');
|
|
addHeadIcon('apple-touch-startup-image');
|
|
}
|
|
function hidemessage() {
|
|
if (_showingConfig == false) {
|
|
document.getElementById('main').className = 'show';
|
|
} else {
|
|
document.getElementById('options').className = 'show';
|
|
}
|
|
document.getElementById('message').style.display = 'none';
|
|
}
|
|
function showmessage() {
|
|
if (_showingConfig == false) {
|
|
document.getElementById('main').className = 'opaque';
|
|
} else {
|
|
document.getElementById('options').className = 'opaque';
|
|
}
|
|
document.getElementById('message').style.display = 'flex';
|
|
}
|
|
function switchOptionsPane() {
|
|
if (_showingConfig == true) {
|
|
document.getElementById('options').style.display = 'none';
|
|
document.getElementById('main').style.display = 'flex';
|
|
_showingConfig = false;
|
|
} else {
|
|
document.getElementById('options').style.display = 'flex';
|
|
document.getElementById('main').style.display = 'none';
|
|
cleanupCalendar();
|
|
_showingConfig = true;
|
|
}
|
|
}
|
|
function calendarhtmlrow(title, daynum, zones) {
|
|
html = '</tr><tr><td>'+title+'</td> \
|
|
<td><input type="time" id="d'+daynum+'-starttime" class="input-ios" onchange="update(this);"></td>';
|
|
for (i=1; i<= zones; i++) {
|
|
html = html+'<td><input type="number" id="d'+daynum+'z'+i+'-runtime" class="input-ios" onchange="update(this);" pattern="[0-9]*" maxlength="2" inputmode="numeric" min="0" max="30" step="1"></td>';
|
|
}
|
|
html = html+'</tr>';
|
|
return html;
|
|
}
|
|
function setzonehtml(zones) {
|
|
html = '';
|
|
|
|
for (i=1; i<= zones; i++) {
|
|
html = html+'<tr> \
|
|
<td>Zone '+i+'</td> \
|
|
<td><label id="z'+i+'-name" class="small-note"></label></td> \
|
|
<td>Runtime</td> \
|
|
<td><input type="number" id="z'+i+'-runtime" onchange="update(this);" class="input-ios" pattern="[0-9]*" maxlength="2" inputmode="numeric" min="1" max="30" step="1"></td> \
|
|
<td> \
|
|
<div class="switch"> \
|
|
<input id="btn-toggle-z'+i+'" name="z'+i+'" class="btn-toggle btn-toggle-round" type="checkbox" onclick="update(this);"> \
|
|
<label for="btn-toggle-z'+i+'"></label> \
|
|
</div> \
|
|
</td> \
|
|
</tr>';
|
|
}
|
|
rnz = document.getElementById('main');
|
|
rnz.innerHTML = rnz.innerHTML.replace("<!-- AUTO HTML -->", html);
|
|
html = '<tr><th colspan="'+(zones+2)+'"> Configuration </th></tr> \
|
|
<tr> \
|
|
<td> </td> \
|
|
<td>Start Time</td>';
|
|
for (i=1; i<= zones; i++) {
|
|
html = html+'<td>Z'+i+'</td>';
|
|
}
|
|
html = html+calendarhtmlrow("Mon", 1, zones);
|
|
html = html+calendarhtmlrow("Tue", 2, zones);
|
|
html = html+calendarhtmlrow("Wed", 3, zones);
|
|
html = html+calendarhtmlrow("Thu", 4, zones);
|
|
html = html+calendarhtmlrow("Fri", 5, zones);
|
|
html = html+calendarhtmlrow("Sat", 6, zones);
|
|
html = html+calendarhtmlrow("Sun", 0, zones);
|
|
rnz = document.getElementById('options');
|
|
rnz.innerHTML = rnz.innerHTML.replace("<!-- AUTO HTML -->", html);
|
|
}
|
|
function cleanupCalendar() {
|
|
for (i = 0; i < 7; i++) {
|
|
st = document.getElementById('d'+i+'-starttime')
|
|
if (st != null) {
|
|
if (st.value != ''){show=true}else{show=false}
|
|
for (j=1; null != (rt = document.getElementById('d'+i+'z'+j+'-runtime'));j++) {
|
|
if (show) {
|
|
rt.classList.remove("opaque");
|
|
if (rt.value == '')
|
|
rt.value = document.getElementById('z'+j+'-runtime').value;
|
|
} else {
|
|
rt.classList.add("opaque");
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
// Need to come back and make the better.
|
|
// if under 375 make sure only 20 character with a space after 10.
|
|
function resize(string) {
|
|
if (_width <= 375)
|
|
return string;
|
|
else
|
|
return "("+string.replace(" ", " ")+")";
|
|
}
|
|
function update(type) {
|
|
var http = new XMLHttpRequest();
|
|
if (http) {
|
|
http.onreadystatechange = function () {
|
|
if (http.readyState === 4) {
|
|
if (http.status == 200 && http.status < 300) {
|
|
if (type == 'firstload') {
|
|
setzonehtml(JSON.parse(http.responseText)['zones']);
|
|
hidemessage();
|
|
}
|
|
var data = JSON.parse(http.responseText);
|
|
for (var obj in data) {
|
|
if (obj == 'title') {
|
|
document.title = data[obj];
|
|
document.getElementById('title').innerHTML = data[obj];
|
|
} if (obj == 'status') {
|
|
if (data[obj].length > 0)
|
|
document.getElementById('status').innerHTML = "Active: "+data[obj];
|
|
else
|
|
document.getElementById('status').innerHTML = "Run now options";
|
|
} else if (obj == '24hdelay-offtime') {
|
|
if (data[obj] > 0) {
|
|
dt = new Date(data[obj] * 1000);
|
|
var options = { weekday: "short", hour: "2-digit", minute: "2-digit"};
|
|
document.getElementById('24hdelay-offtime').innerHTML = '(until '+dt.toLocaleTimeString("en-us", options)+')';
|
|
} else {
|
|
document.getElementById('24hdelay-offtime').innerHTML = '';
|
|
}
|
|
} else if ( (field = document.getElementById(obj)) != null){
|
|
if (obj.substr(-5,5) == '-name')
|
|
field.innerHTML = resize(data[obj]);
|
|
else
|
|
field.value = data[obj];
|
|
} else {
|
|
if ( (btn = document.getElementById('btn-toggle-' + obj)) != null){
|
|
if (data[obj] == 'on')
|
|
btn.checked = true;
|
|
else
|
|
btn.checked = false;
|
|
} else if ( (btn = document.getElementById('btn-' + obj)) != null){
|
|
btn.selectedIndex = data[obj];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
else if(http.status >= 400 || http.status == 0) {
|
|
document.getElementById('message-text').innerHTML = 'Error connecting to server';
|
|
}
|
|
}
|
|
}
|
|
};
|
|
if (type == null ) {
|
|
//http.open('GET', location.href + '?type=read', true);
|
|
http.open('GET', location.href + '.read.json', true);
|
|
//http.open('GET', 'http://raw.githubusercontent.com/sfeakes/sprinklerd/master/demo/index.html.firstload.json', true);
|
|
http.send(null);
|
|
} else if (type == 'firstload') {
|
|
//http.open('GET', location.href + '?type=firstload', true);
|
|
http.open('GET', location.href + '.firstload.json', true);
|
|
http.send(null);
|
|
} else if (type != null) {
|
|
clearTimeout(_poller);
|
|
// compare xx-starttime or xxx-runtime or xxxx-runtime
|
|
if (type.id.substr(-4,4) == 'time') {
|
|
// d1z10-runtime is a calendar config, z11-runtime is a zone default runtime
|
|
if (type.id.substr(0,1) == 'd') {
|
|
http.open('GET', location.href + '?type=calcfg&day='+type.id.substr(1,1)+'&zone='+type.id.substr(3,type.id.lastIndexOf("-")-3)+'&time='+type.value);
|
|
http.send(null);
|
|
cleanupCalendar();
|
|
} else if (type.id.substr(0,1) == 'z') {
|
|
http.open('GET', location.href + '?type=zrtcfg&zone='+type.id.substr(1,type.id.lastIndexOf("-")-1)+'&time='+type.value);
|
|
http.send(null);
|
|
cleanupCalendar();
|
|
}
|
|
} else if (type.id.substr(0,11) == 'btn-toggle-'){
|
|
// If there is a 'xx-runtime' id then it's a zone to start, else it's a cfg button
|
|
if ( (rto = document.getElementById(type.name + '-runtime')) != null){
|
|
http.open('GET', location.href + '?type=zone&zone=' + type.name.substring(1) + '&state=' + ((type.checked == true) ? 'on' : 'off') + '&runtime=' + rto.value, true);
|
|
} else {
|
|
http.open('GET', location.href + '?type=option&option=' + type.name + '&state=' + ((type.checked == true) ? 'on' : 'off'), true);
|
|
}
|
|
// DEMO CLEANUP CRAP
|
|
/*
|
|
if (rto != null && type.checked == true) {
|
|
i = 1;
|
|
while ( (zb = document.getElementById('btn-toggle-z'+i)) != null ) {
|
|
if (zb.name != type.name) {
|
|
zb.checked = false;
|
|
i++;
|
|
}
|
|
}
|
|
}*/
|
|
|
|
http.send(null);
|
|
}
|
|
}
|
|
_poller = setTimeout(update, 5000);
|
|
}
|
|
</script>
|
|
</head>
|
|
|
|
<body onload='setSizeSpecifics(); headIcons(); update("firstload");'>
|
|
<div id='message' class='message' onmousedown='hidemessage()'>
|
|
<div id='message-text' class='message-text'>
|
|
<b>Loading....</b>
|
|
</div>
|
|
</div>
|
|
<div id='option_button' class='option_button' onmousedown='switchOptionsPane()'>
|
|
<img src=''>
|
|
</div>
|
|
<div id='main' class='opaque'>
|
|
<table>
|
|
<tbody>
|
|
<tr>
|
|
<th colspan='5'><label id='title'>Sprinkler</label></th>
|
|
</tr>
|
|
<tr>
|
|
<td colspan='4'>
|
|
Run on calendar schedule
|
|
</td>
|
|
<td>
|
|
<div class='switch'>
|
|
<input id='btn-toggle-calendar' name='calendar' class='btn-toggle btn-toggle-round' type='checkbox' onclick='update(this);'>
|
|
<label for='btn-toggle-calendar'></label>
|
|
</div>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<td colspan='4'>
|
|
24h Rain Delay <label id='24hdelay-offtime' class='small-note'></label>
|
|
</td>
|
|
<td>
|
|
<div class='switch'>
|
|
<input id='btn-toggle-24hdelay' name='24hdelay' class='btn-toggle btn-toggle-round' type='checkbox' onclick='update(this);'>
|
|
<label for='btn-toggle-24hdelay'></label>
|
|
</div>
|
|
</td>
|
|
</tr>
|
|
<tr>
|
|
<th colspan='5' class='status'>
|
|
<label id='status'>Run now option</label>
|
|
<!-- <font size='-1'>Run now option</font> -->
|
|
</th>
|
|
</tr>
|
|
<tr>
|
|
<td colspan='4'>Cycle all Zones</td>
|
|
<td>
|
|
<div class='switch'>
|
|
<input id='btn-toggle-allz' name='allz' class='btn-toggle btn-toggle-round' type='checkbox' onclick='update(this);'>
|
|
<label for='btn-toggle-allz'></label>
|
|
</div>
|
|
</td>
|
|
</tr>
|
|
<!-- AUTO HTML -->
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
<div id='options' class='options'>
|
|
<table>
|
|
<tbody>
|
|
<!-- AUTO HTML -->
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</body>
|
|
</html>
|