|
|
|
@ -1,459 +1,501 @@
|
|
|
|
|
<!DOCTYPE html>
|
|
|
|
|
<html lang="en">
|
|
|
|
|
<head>
|
|
|
|
|
<meta charset="utf-8">
|
|
|
|
|
<meta name="viewport" content="width=500">
|
|
|
|
|
<title>LED Settings</title>
|
|
|
|
|
<script>
|
|
|
|
|
var d=document,laprev=55,maxB=1,maxM=5000,maxPB=4096,bquot=0; //maximum bytes for LED allocation: 5kB for 8266, 32kB for 32
|
|
|
|
|
function H()
|
|
|
|
|
{
|
|
|
|
|
window.open("https://github.com/Aircoookie/WLED/wiki/Settings#led-settings");
|
|
|
|
|
}
|
|
|
|
|
function B()
|
|
|
|
|
{
|
|
|
|
|
window.open("/settings","_self");
|
|
|
|
|
}
|
|
|
|
|
function gId(n){return d.getElementById(n);}
|
|
|
|
|
function off(n){
|
|
|
|
|
d.getElementsByName(n)[0].value = -1;
|
|
|
|
|
}
|
|
|
|
|
var timeout;
|
|
|
|
|
function showToast(text, error = false)
|
|
|
|
|
{
|
|
|
|
|
var x = gId("toast");
|
|
|
|
|
x.innerHTML = text;
|
|
|
|
|
x.className = error ? "error":"show";
|
|
|
|
|
clearTimeout(timeout);
|
|
|
|
|
x.style.animation = 'none';
|
|
|
|
|
timeout = setTimeout(function(){ x.className = x.className.replace("show", ""); }, 2900);
|
|
|
|
|
}
|
|
|
|
|
function bLimits(b,p,m) {
|
|
|
|
|
maxB = b; maxM = m; maxPB = p;
|
|
|
|
|
}
|
|
|
|
|
function pinsOK() {
|
|
|
|
|
var LCs = d.getElementsByTagName("input");
|
|
|
|
|
for (i=0; i<LCs.length; i++) {
|
|
|
|
|
var nm = LCs[i].name.substring(0,2);
|
|
|
|
|
//check for pin conflicts
|
|
|
|
|
if (nm=="L0" || nm=="L1" || nm=="L2" || nm=="L3" || nm=="L4" || nm=="RL" || nm=="BT" || nm=="IR")
|
|
|
|
|
if (LCs[i].value!="" && LCs[i].value!="-1") {
|
|
|
|
|
if (d.um_p && d.um_p.some((e)=>e==parseInt(LCs[i].value,10))) {alert(`Sorry, pins ${JSON.stringify(d.um_p)} can't be used.`);LCs[i].value="";LCs[i].focus();return false;}
|
|
|
|
|
else if (LCs[i].value > 5 && LCs[i].value < 12) {alert("Sorry, pins 6-11 can not be used.");LCs[i].value="";LCs[i].focus();return false;}
|
|
|
|
|
else if (!(nm == "IR" || nm=="BT") && LCs[i].value > 33) {alert("Sorry, pins >33 are input only.");LCs[i].value="";LCs[i].focus();return false;}
|
|
|
|
|
for (j=i+1; j<LCs.length; j++)
|
|
|
|
|
{
|
|
|
|
|
var n2 = LCs[j].name.substring(0,2);
|
|
|
|
|
if (n2=="L0" || n2=="L1" || n2=="L2" || n2=="L3" || n2=="L4" || n2=="RL" || n2=="BT" || n2=="IR")
|
|
|
|
|
if (LCs[j].value!="" && LCs[i].value==LCs[j].value) {alert(`Pin conflict between ${LCs[i].name}/${LCs[j].name}!`);LCs[j].value="";LCs[j].focus();return false;}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
function trySubmit(e) {
|
|
|
|
|
d.Sf.data.value = '';
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
if (!pinsOK()) {e.stopPropagation();return false;} // Prevent form submission and contact with server
|
|
|
|
|
if (bquot > 100) {var msg = "Too many LEDs for me to handle!"; if (maxM < 10000) msg += "\n\rConsider using an ESP32."; alert(msg);}
|
|
|
|
|
if (d.Sf.checkValidity()) d.Sf.submit(); //https://stackoverflow.com/q/37323914
|
|
|
|
|
}
|
|
|
|
|
function S(){GetV();setABL();}
|
|
|
|
|
function enABL()
|
|
|
|
|
{
|
|
|
|
|
var en = gId('able').checked;
|
|
|
|
|
d.Sf.LA.value = (en) ? laprev:0;
|
|
|
|
|
gId('abl').style.display = (en) ? 'inline':'none';
|
|
|
|
|
gId('psu2').style.display = (en) ? 'inline':'none';
|
|
|
|
|
if (d.Sf.LA.value > 0) setABL();
|
|
|
|
|
}
|
|
|
|
|
function enLA()
|
|
|
|
|
{
|
|
|
|
|
var val = d.Sf.LAsel.value;
|
|
|
|
|
d.Sf.LA.value = val;
|
|
|
|
|
gId('LAdis').style.display = (val == 50) ? 'inline':'none';
|
|
|
|
|
UI();
|
|
|
|
|
}
|
|
|
|
|
function setABL()
|
|
|
|
|
{
|
|
|
|
|
gId('able').checked = true;
|
|
|
|
|
d.Sf.LAsel.value = 50;
|
|
|
|
|
switch (parseInt(d.Sf.LA.value)) {
|
|
|
|
|
case 0: gId('able').checked = false; enABL(); break;
|
|
|
|
|
case 30: d.Sf.LAsel.value = 30; break;
|
|
|
|
|
case 35: d.Sf.LAsel.value = 35; break;
|
|
|
|
|
case 55: d.Sf.LAsel.value = 55; break;
|
|
|
|
|
case 255: d.Sf.LAsel.value = 255; break;
|
|
|
|
|
default: gId('LAdis').style.display = 'inline';
|
|
|
|
|
}
|
|
|
|
|
gId('m1').innerHTML = maxM;
|
|
|
|
|
d.getElementsByName("Sf")[0].addEventListener("submit", trySubmit);
|
|
|
|
|
UI();
|
|
|
|
|
}
|
|
|
|
|
//returns mem usage
|
|
|
|
|
function getMem(type, len, p0) {
|
|
|
|
|
if (type < 32) {
|
|
|
|
|
if (maxM < 10000 && p0==3) { //8266 DMA uses 5x the mem
|
|
|
|
|
if (type > 29) return len*20; //RGBW
|
|
|
|
|
return len*15;
|
|
|
|
|
} else if (maxM >= 10000) //ESP32 RMT uses double buffer?
|
|
|
|
|
{
|
|
|
|
|
if (type > 29) return len*8; //RGBW
|
|
|
|
|
return len*6;
|
|
|
|
|
}
|
|
|
|
|
if (type > 29) return len*4; //RGBW
|
|
|
|
|
return len*3;
|
|
|
|
|
}
|
|
|
|
|
if (type > 31 && type < 48) return 5;
|
|
|
|
|
if (type == 44 || type == 45) return len*4; //RGBW
|
|
|
|
|
return len*3;
|
|
|
|
|
}
|
|
|
|
|
function UI(change=false)
|
|
|
|
|
{
|
|
|
|
|
var isRGBW = false, memu = 0;
|
|
|
|
|
|
|
|
|
|
gId('ampwarning').style.display = (d.Sf.MA.value > 7200) ? 'inline':'none';
|
|
|
|
|
|
|
|
|
|
if (d.Sf.LA.value == 255) laprev = 12;
|
|
|
|
|
else if (d.Sf.LA.value > 0) laprev = d.Sf.LA.value;
|
|
|
|
|
|
|
|
|
|
// enable/disable LED fields
|
|
|
|
|
var s = d.getElementsByTagName("select");
|
|
|
|
|
for (i=0; i<s.length; i++) {
|
|
|
|
|
// is the field a LED type?
|
|
|
|
|
if (s[i].name.substring(0,2)=="LT") {
|
|
|
|
|
n=s[i].name.substring(2);
|
|
|
|
|
var type = parseInt(s[i].value,10);
|
|
|
|
|
gId("p0d"+n).innerHTML = (type > 49) ? "Data GPIO:" : (type >41) ? "GPIOs:" : "GPIO:";
|
|
|
|
|
gId("p1d"+n).innerHTML = (type > 49) ? "Clk GPIO:" : "";
|
|
|
|
|
var LK = d.getElementsByName("L1"+n)[0]; // clock pin
|
|
|
|
|
|
|
|
|
|
memu += getMem(type, d.getElementsByName("LC"+n)[0].value, d.getElementsByName("L0"+n)[0].value); // calc memory
|
|
|
|
|
|
|
|
|
|
// enumerate pins
|
|
|
|
|
for (p=1; p<5; p++) {
|
|
|
|
|
var LK = d.getElementsByName("L"+p+n)[0]; // secondary pins
|
|
|
|
|
if (!LK) continue;
|
|
|
|
|
if ((type>49 && p==1) || (type>41 && type < 50 && (p+40 < type))) // TYPE_xxxx values from const.h
|
|
|
|
|
{
|
|
|
|
|
// display pin field
|
|
|
|
|
LK.style.display = "inline";
|
|
|
|
|
LK.required = true;
|
|
|
|
|
} else {
|
|
|
|
|
// hide pin field
|
|
|
|
|
LK.style.display = "none";
|
|
|
|
|
LK.required = false;
|
|
|
|
|
LK.value="";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (change) {
|
|
|
|
|
// // blazoncek experimental extension
|
|
|
|
|
// gId("ew"+n).checked = (type == 30 || type == 31 || type == 44 || type == 45); // RGBW checkbox, TYPE_xxxx values from const.h
|
|
|
|
|
gId("ls"+n).value = n+1; // set LED start
|
|
|
|
|
if (type > 31 && type < 48) d.getElementsByName("LC"+n)[0].value = 1; // for sanity change analog count just to 1 LED
|
|
|
|
|
}
|
|
|
|
|
// // blazoncek experimental extension
|
|
|
|
|
// gId("ew"+n).onclick = (type > 31 && type < 48) ? (function(){return false}) : (function(){}); // prevent change for analog
|
|
|
|
|
// isRGBW |= gId("ew"+n).checked;
|
|
|
|
|
isRGBW |= (type == 30 || type == 31 || (type > 40 && type < 46 && type != 43)); // RGBW checkbox, TYPE_xxxx values from const.h
|
|
|
|
|
gId("co"+n).style.display = (type == 41 || type == 42) ? "none":"inline"; // hide color order for PWM W & WW/CW
|
|
|
|
|
gId("dig"+n+"c").style.display = (type > 40 && type < 48) ? "none":"inline"; // hide skip 1st & count for analog
|
|
|
|
|
gId("dig"+n+"s").style.display = (type > 40 && type < 48) ? "none":"inline"; // hide skip 1st & count for analog
|
|
|
|
|
gId("rev"+n).innerHTML = (type > 40 && type < 48) ? "Inverted":"Reverse (rotated 180°)"; // change reverse text for analog
|
|
|
|
|
gId("psd"+n).innerHTML = (type > 40 && type < 48) ? "Index:":"Start:"; // change analog start description
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// display white channel calculation method
|
|
|
|
|
var myC = d.querySelectorAll('.wc'),
|
|
|
|
|
l = myC.length;
|
|
|
|
|
for (i = 0; i < l; i++) {
|
|
|
|
|
myC[i].style.display = (isRGBW) ? 'inline':'none';
|
|
|
|
|
}
|
|
|
|
|
// check for pin conflicts
|
|
|
|
|
var LCs = d.getElementsByTagName("input");
|
|
|
|
|
var sLC = 0, maxLC = 0;
|
|
|
|
|
for (i=0; i<LCs.length; i++) {
|
|
|
|
|
var nm = LCs[i].name.substring(0,2);
|
|
|
|
|
// do we have a led count field but not total led count
|
|
|
|
|
if (nm=="LC" && LCs[i].name !== "LC") {
|
|
|
|
|
var n=LCs[i].name.substring(2);
|
|
|
|
|
var c=parseInt(LCs[i].value,10);
|
|
|
|
|
/*if(gId("ls"+n).readOnly)*/ gId("ls"+n).value=sLC; // update led start field
|
|
|
|
|
if(c){sLC+=c;if(c>maxLC)maxLC=c;} // increase led count
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
// do we have led pins for digital leds
|
|
|
|
|
if (nm=="L0" || nm=="L1") {
|
|
|
|
|
var lc=d.getElementsByName("LC"+LCs[i].name.substring(2))[0];
|
|
|
|
|
lc.max=maxPB; // update max led count value
|
|
|
|
|
}
|
|
|
|
|
// for pins check conflicts
|
|
|
|
|
if (nm=="L0" || nm=="L1" || nm=="L2" || nm=="L3" || nm=="L4" || nm=="RL" || nm=="BT" || nm=="IR")
|
|
|
|
|
if (LCs[i].value!="" && LCs[i].value!="-1") {
|
|
|
|
|
var p = []; // used pin array
|
|
|
|
|
if (d.um_p && Array.isArray(d.um_p)) for (k=0;k<d.um_p.length;k++) p.push(d.um_p[k]); // fill with reservations
|
|
|
|
|
for (j=0; j<LCs.length; j++) {
|
|
|
|
|
if (i==j) continue;
|
|
|
|
|
var n2 = LCs[j].name.substring(0,2);
|
|
|
|
|
if (n2=="L0" || n2=="L1" || n2=="L2" || n2=="L3" || n2=="L4" || n2=="RL" || n2=="BT" || n2=="IR")
|
|
|
|
|
if (LCs[j].value!="" && LCs[j].value!="-1") p.push(parseInt(LCs[j].value,10)); // add current pin
|
|
|
|
|
}
|
|
|
|
|
// now check for conflicts
|
|
|
|
|
if (p.some((e)=>e==parseInt(LCs[i].value,10))) LCs[i].style.color="red"; else LCs[i].style.color=parseInt(LCs[i].value,10)>33?"orange":"#fff";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// update total led count
|
|
|
|
|
if (gId("LC").readOnly) d.getElementsByName("LC")[0].value = sLC;
|
|
|
|
|
// if we are changing total led count update led count for 1st strip
|
|
|
|
|
if (d.activeElement == d.getElementsByName("LC")[0]) {
|
|
|
|
|
var o = d.getElementsByClassName("iST");
|
|
|
|
|
var i = o.length;
|
|
|
|
|
if (i == 1) d.getElementsByName("LC0")[0].value = d.getElementsByName("LC")[0].value;
|
|
|
|
|
}
|
|
|
|
|
// memory usage and warnings
|
|
|
|
|
gId('m0').innerHTML = memu;
|
|
|
|
|
bquot = memu / maxM * 100;
|
|
|
|
|
gId('dbar').style.background = `linear-gradient(90deg, ${bquot > 60 ? (bquot > 90 ? "red":"orange"):"#ccc"} 0 ${bquot}%%, #444 ${bquot}%% 100%%)`;
|
|
|
|
|
gId('ledwarning').style.display = (sLC > maxPB || maxLC > 800 || bquot > 80) ? 'inline':'none';
|
|
|
|
|
gId('ledwarning').style.color = (sLC > maxPB || maxLC > maxPB || bquot > 100) ? 'red':'orange';
|
|
|
|
|
gId('wreason').innerHTML = (bquot > 80) ? "80% of max. LED memory" +(bquot>100 ? ` (<b>WARNING: Using over ${maxM}B!</b>)` : "") : "800 LEDs per GPIO";
|
|
|
|
|
// calculate power
|
|
|
|
|
var val = Math.ceil((100 + sLC * laprev)/500)/2;
|
|
|
|
|
val = (val > 5) ? Math.ceil(val) : val;
|
|
|
|
|
var s = "";
|
|
|
|
|
var is12V = (d.Sf.LAsel.value == 30);
|
|
|
|
|
var isWS2815 = (d.Sf.LAsel.value == 255);
|
|
|
|
|
if (val < 1.02 && !is12V && !isWS2815)
|
|
|
|
|
{
|
|
|
|
|
s = "ESP 5V pin with 1A USB supply";
|
|
|
|
|
} else
|
|
|
|
|
{
|
|
|
|
|
s += is12V ? "12V ": isWS2815 ? "WS2815 12V " : "5V ";
|
|
|
|
|
s += val;
|
|
|
|
|
s += "A supply connected to LEDs";
|
|
|
|
|
}
|
|
|
|
|
var val2 = Math.ceil((100 + sLC * laprev)/1500)/2;
|
|
|
|
|
val2 = (val2 > 5) ? Math.ceil(val2) : val2;
|
|
|
|
|
var s2 = "(for most effects, ~";
|
|
|
|
|
s2 += val2;
|
|
|
|
|
s2 += "A is enough)<br>";
|
|
|
|
|
gId('psu').innerHTML = s;
|
|
|
|
|
gId('psu2').innerHTML = isWS2815 ? "" : s2;
|
|
|
|
|
gId("json").style.display = d.Sf.IT.value==8 ? "" : "none";
|
|
|
|
|
}
|
|
|
|
|
function lastEnd(i) {
|
|
|
|
|
if (i<1) return 0;
|
|
|
|
|
v = parseInt(d.getElementsByName("LS"+(i-1))[0].value) + parseInt(d.getElementsByName("LC"+(i-1))[0].value);
|
|
|
|
|
var type = parseInt(d.getElementsByName("LT"+(i-1))[0].value);
|
|
|
|
|
if (type > 31 && type < 48) v = 1; //PWM busses
|
|
|
|
|
if (isNaN(v)) return 0;
|
|
|
|
|
return v;
|
|
|
|
|
}
|
|
|
|
|
function addLEDs(n)
|
|
|
|
|
{
|
|
|
|
|
if (n>1) {maxB=n; gId("+").style.display="inline"; return;}
|
|
|
|
|
|
|
|
|
|
var o = d.getElementsByClassName("iST");
|
|
|
|
|
var i = o.length;
|
|
|
|
|
|
|
|
|
|
if ((n==1 && i>=maxB) || (n==-1 && i==0)) return;
|
|
|
|
|
|
|
|
|
|
var f = gId("mLC");
|
|
|
|
|
if (n==1) {
|
|
|
|
|
// npm run build has trouble minimizing spaces inside string
|
|
|
|
|
var cn = `<div class="iST">
|
|
|
|
|
${i>0?'<hr style="width:260px">':''}
|
|
|
|
|
${i+1}:
|
|
|
|
|
<select name="LT${i}" onchange="UI(true)">
|
|
|
|
|
<option value="22">WS281x</option>
|
|
|
|
|
<option value="30">SK6812 RGBW</option>
|
|
|
|
|
<option value="31">TM1814</option>
|
|
|
|
|
<option value="24">400kHz</option>
|
|
|
|
|
<option value="50">WS2801</option>
|
|
|
|
|
<option value="51">APA102</option>
|
|
|
|
|
<option value="52">LPD8806</option>
|
|
|
|
|
<option value="53">P9813</option>
|
|
|
|
|
<option value="41">PWM White</option>
|
|
|
|
|
<option value="42">PWM WWCW</option>
|
|
|
|
|
<option value="43">PWM RGB</option>
|
|
|
|
|
<option value="44">PWM RGBW</option>
|
|
|
|
|
<option value="45">PWM RGBWC</option>
|
|
|
|
|
</select>
|
|
|
|
|
<div id="co${i}" style="display:inline">Color Order:
|
|
|
|
|
<select name="CO${i}">
|
|
|
|
|
<option value="0">GRB</option>
|
|
|
|
|
<option value="1">RGB</option>
|
|
|
|
|
<option value="2">BRG</option>
|
|
|
|
|
<option value="3">RBG</option>
|
|
|
|
|
<option value="4">BGR</option>
|
|
|
|
|
<option value="5">GBR</option>
|
|
|
|
|
</select></div><br>
|
|
|
|
|
<span id="p0d${i}">GPIO:</span><input type="number" name="L0${i}" min="0" max="33" required class="s" onchange="UI()"/>
|
|
|
|
|
<span id="p1d${i}"></span><input type="number" name="L1${i}" min="0" max="33" class="s" onchange="UI()"/>
|
|
|
|
|
<span id="p2d${i}"></span><input type="number" name="L2${i}" min="0" max="33" class="s" onchange="UI()"/>
|
|
|
|
|
<span id="p3d${i}"></span><input type="number" name="L3${i}" min="0" max="33" class="s" onchange="UI()"/>
|
|
|
|
|
<span id="p4d${i}"></span><input type="number" name="L4${i}" min="0" max="33" class="s" onchange="UI()"/>
|
|
|
|
|
<br>
|
|
|
|
|
<span id="psd${i}">Start:</span> <input type="number" name="LS${i}" id="ls${i}" class="l" min="0" max="8191" value="${lastEnd(i)}" readonly required />
|
|
|
|
|
<div id="dig${i}c" style="display:inline">Count: <input type="number" name="LC${i}" class="l" min="0" max="${maxPB}" value="1" required oninput="UI()" /></div><br>
|
|
|
|
|
<span id="rev${i}">Reverse (rotated 180°)</span>: <input type="checkbox" name="CV${i}">
|
|
|
|
|
<div id="dig${i}s" style="display:inline"> Skip 1<sup>st</sup> LED: <input id="sl${i}" type="checkbox" name="SL${i}"></div><br>
|
|
|
|
|
</div>`;
|
|
|
|
|
f.insertAdjacentHTML("beforeend", cn);
|
|
|
|
|
}
|
|
|
|
|
if (n==-1) {
|
|
|
|
|
o[--i].remove();--i;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
gId("+").style.display = (i<maxB-1) ? "inline":"none";
|
|
|
|
|
gId("-").style.display = (i>0) ? "inline":"none";
|
|
|
|
|
|
|
|
|
|
UI();
|
|
|
|
|
}
|
|
|
|
|
function addBtn(i,p,t) {
|
|
|
|
|
var c = gId("btns").innerHTML;
|
|
|
|
|
var bt = "BT" + i;
|
|
|
|
|
var be = "BE" + i;
|
|
|
|
|
c += `Button ${i} GPIO: <input type="number" min="-1" max="40" name="${bt}" onchange="UI()" class="s" value="${p}">`;
|
|
|
|
|
c += `<select name="${be}">`
|
|
|
|
|
c += `<option value="0" ${t==0?"selected":""}>Disabled</option>`;
|
|
|
|
|
c += `<option value="2" ${t==2?"selected":""}>Pushbutton</option>`;
|
|
|
|
|
c += `<option value="3" ${t==3?"selected":""}>Push inverted</option>`;
|
|
|
|
|
c += `<option value="4" ${t==4?"selected":""}>Switch</option>`;
|
|
|
|
|
c += `<option value="5" ${t==5?"selected":""}>PIR sensor</option>`;
|
|
|
|
|
c += `<option value="6" ${t==6?"selected":""}>Touch</option>`;
|
|
|
|
|
c += `<option value="7" ${t==7?"selected":""}>Analog</option>`;
|
|
|
|
|
c += `<option value="8" ${t==8?"selected":""}>Analog inverted</option>`;
|
|
|
|
|
c += `</select>`;
|
|
|
|
|
c += `<span style="cursor: pointer;" onclick="off('${bt}')"> ×</span><br>`;
|
|
|
|
|
gId("btns").innerHTML = c;
|
|
|
|
|
}
|
|
|
|
|
function uploadFile(name) {
|
|
|
|
|
var req = new XMLHttpRequest();
|
|
|
|
|
req.addEventListener('load', function(){showToast(this.responseText,this.status >= 400)});
|
|
|
|
|
req.addEventListener('error', function(e){showToast(e.stack,true);});
|
|
|
|
|
req.open("POST", "/upload");
|
|
|
|
|
var formData = new FormData();
|
|
|
|
|
formData.append("data", d.Sf.data.files[0], name);
|
|
|
|
|
req.send(formData);
|
|
|
|
|
d.Sf.data.value = '';
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
function GetV()
|
|
|
|
|
{
|
|
|
|
|
//values injected by server while sending HTML
|
|
|
|
|
//maxM=5000;maxPB=1536;d.um_p=[1,6,7,8,9,10,11];addLEDs(5);d.Sf.LC.value=250;addLEDs(1);d.Sf.L00.value=2;d.Sf.L10.value=0;d.Sf.LC0.value=250;d.Sf.LT0.value=22;d.Sf.CO0.value=0;d.Sf.LS0.value=0;d.Sf.LS0.checked=0;d.Sf.MA.value=5400;d.Sf.LA.value=55;d.getElementsByClassName("pow")[0].innerHTML="350mA";d.Sf.CA.value=40;d.Sf.AW.value=3;d.Sf.BO.checked=0;d.Sf.BP.value=3;d.Sf.GB.checked=0;d.Sf.GC.checked=1;d.Sf.TF.checked=1;d.Sf.TD.value=700;d.Sf.PF.checked=0;d.Sf.BF.value=64;d.Sf.TB.value=0;d.Sf.TL.value=60;d.Sf.TW.value=1;d.Sf.PB.selectedIndex=0;d.Sf.RL.value=12;d.Sf.RM.checked=0;addBtn(0,0,2);addBtn(1,3,4);addBtn(2,-1,0);d.Sf.IR.value=-1;
|
|
|
|
|
}
|
|
|
|
|
</script>
|
|
|
|
|
<style>@import url("style.css");</style>
|
|
|
|
|
</head>
|
|
|
|
|
<body onload="S()">
|
|
|
|
|
<form id="form_s" name="Sf" method="post">
|
|
|
|
|
<div class="toprow">
|
|
|
|
|
<div class="helpB"><button type="button" onclick="H()">?</button></div>
|
|
|
|
|
<button type="button" onclick="B()">Back</button><button type="submit">Save</button><hr>
|
|
|
|
|
</div>
|
|
|
|
|
<h2>LED & Hardware setup</h2>
|
|
|
|
|
Total LED count: <input name="LC" id="LC" type="number" min="1" max="8192" oninput="UI()" required readonly><br>
|
|
|
|
|
<i>Recommended power supply for brightest white:</i><br>
|
|
|
|
|
<b><span id="psu">?</span></b><br>
|
|
|
|
|
<span id="psu2"><br></span>
|
|
|
|
|
<br>
|
|
|
|
|
Enable automatic brightness limiter: <input type="checkbox" name="ABen" onchange="enABL()" id="able"><br>
|
|
|
|
|
<div id="abl">
|
|
|
|
|
Maximum Current: <input name="MA" type="number" class="l" min="250" max="65000" oninput="UI()" required> mA<br>
|
|
|
|
|
<div id="ampwarning" style="color: orange; display: none;">
|
|
|
|
|
⚠ Your power supply provides high current.<br>
|
|
|
|
|
To improve the safety of your setup,<br>
|
|
|
|
|
please use thick cables,<br>
|
|
|
|
|
multiple power injection points and a fuse!<br>
|
|
|
|
|
</div>
|
|
|
|
|
<i>Automatically limits brightness to stay close to the limit.<br>
|
|
|
|
|
Keep at <1A if powering LEDs directly from the ESP 5V pin!<br>
|
|
|
|
|
If you are using an external power supply, enter its rating.<br>
|
|
|
|
|
(Current estimated usage: <span class="pow">unknown</span>)</i><br><br>
|
|
|
|
|
LED voltage (Max. current for a single LED):<br>
|
|
|
|
|
<select name="LAsel" onchange="enLA()">
|
|
|
|
|
<option value="55" selected>5V default (55mA)</option>
|
|
|
|
|
<option value="35">5V efficient (35mA)</option>
|
|
|
|
|
<option value="30">12V (30mA)</option>
|
|
|
|
|
<option value="255">WS2815 (12mA)</option>
|
|
|
|
|
<option value="50">Custom</option>
|
|
|
|
|
</select><br>
|
|
|
|
|
<span id="LAdis" style="display: none;">Custom max. current per LED: <input name="LA" type="number" min="0" max="255" id="la" oninput="UI()" required> mA<br></span>
|
|
|
|
|
<i>Keep at default if you are unsure about your type of LEDs.</i><br>
|
|
|
|
|
</div>
|
|
|
|
|
<h3>Hardware setup</h3>
|
|
|
|
|
<div id="mLC">LED outputs:</div>
|
|
|
|
|
<button type="button" id="+" onclick="addLEDs(1)">+</button>
|
|
|
|
|
<button type="button" id="-" onclick="addLEDs(-1)">-</button><br>
|
|
|
|
|
LED Memory Usage: <span id="m0">0</span> / <span id="m1">?</span> B<br>
|
|
|
|
|
<div id="dbar" style="display:inline-block; width: 100px; height: 10px; border-radius: 20px;"></div><br>
|
|
|
|
|
<div id="ledwarning" style="color: orange; display: none;">
|
|
|
|
|
⚠ You might run into stability or lag issues.<br>
|
|
|
|
|
Use less than <span id="wreason">800 LEDs per pin</span> for the best experience!<br>
|
|
|
|
|
</div>
|
|
|
|
|
Make a segment for each output: <input type="checkbox" name="MS"> <br>
|
|
|
|
|
<hr style="width:260px">
|
|
|
|
|
<div id="btns"></div>
|
|
|
|
|
Touch threshold: <input type="number" min="0" max="100" name="TT" required><br>
|
|
|
|
|
IR GPIO: <input type="number" min="-1" max="40" name="IR" onchange="UI()" class="s"><select name="IT" onchange="UI()">
|
|
|
|
|
<option value=0>Remote disabled</option>
|
|
|
|
|
<option value=1>24-key RGB</option>
|
|
|
|
|
<option value=2>24-key with CT</option>
|
|
|
|
|
<option value=3>40-key blue</option>
|
|
|
|
|
<option value=4>44-key RGB</option>
|
|
|
|
|
<option value=5>21-key RGB</option>
|
|
|
|
|
<option value=6>6-key black</option>
|
|
|
|
|
<option value=7>9-key red</option>
|
|
|
|
|
<option value=8>JSON remote</option>
|
|
|
|
|
</select><span style="cursor: pointer;" onclick="off('IR')"> ×</span><br>
|
|
|
|
|
<div id="json" style="display:none;">JSON file: <input type="file" name="data" accept=".json"> <input type="button" value="Upload" onclick="uploadFile('/ir.json');"><br></div>
|
|
|
|
|
<div id="toast"></div>
|
|
|
|
|
<a href="https://github.com/Aircoookie/WLED/wiki/Infrared-Control" target="_blank">IR info</a><br>
|
|
|
|
|
Relay GPIO: <input type="number" min="-1" max="33" name="RL" onchange="UI()" class="s"> invert <input type="checkbox" name="RM"><span style="cursor: pointer;" onclick="off('RL')"> ×</span><br>
|
|
|
|
|
<hr style="width:260px">
|
|
|
|
|
<h3>Defaults</h3>
|
|
|
|
|
Turn LEDs on after power up/reset: <input type="checkbox" name="BO"><br>
|
|
|
|
|
Default brightness: <input name="CA" type="number" class="m" min="0" max="255" required> (0-255)<br><br>
|
|
|
|
|
Apply preset <input name="BP" type="number" class="m" min="0" max="250" required> at boot (0 uses defaults)
|
|
|
|
|
<br><br>
|
|
|
|
|
Use Gamma correction for color: <input type="checkbox" name="GC"> (strongly recommended)<br>
|
|
|
|
|
Use Gamma correction for brightness: <input type="checkbox" name="GB"> (not recommended)<br><br>
|
|
|
|
|
Brightness factor: <input name="BF" type="number" class="m" min="1" max="255" required> %
|
|
|
|
|
<h3>Transitions</h3>
|
|
|
|
|
Crossfade: <input type="checkbox" name="TF"><br>
|
|
|
|
|
Transition Time: <input name="TD" type="number" class="xl" min="0" max="65500"> ms<br>
|
|
|
|
|
Enable Palette transitions: <input type="checkbox" name="PF">
|
|
|
|
|
<h3>Timed light</h3>
|
|
|
|
|
Default Duration: <input name="TL" type="number" class="m" min="1" max="255" required> min<br>
|
|
|
|
|
Default Target brightness: <input name="TB" type="number" class="m" min="0" max="255" required><br>
|
|
|
|
|
Mode:
|
|
|
|
|
<select name="TW">
|
|
|
|
|
<option value="0">Wait and set</option>
|
|
|
|
|
<option value="1">Fade</option>
|
|
|
|
|
<option value="2">Fade Color</option>
|
|
|
|
|
<option value="3">Sunrise</option>
|
|
|
|
|
</select>
|
|
|
|
|
<h3>Advanced</h3>
|
|
|
|
|
Palette blending:
|
|
|
|
|
<select name="PB">
|
|
|
|
|
<option value="0">Linear (wrap if moving)</option>
|
|
|
|
|
<option value="1">Linear (always wrap)</option>
|
|
|
|
|
<option value="2">Linear (never wrap)</option>
|
|
|
|
|
<option value="3">None (not recommended)</option>
|
|
|
|
|
</select><br>
|
|
|
|
|
<span class="wc">
|
|
|
|
|
Auto-calculate white channel from RGB:<br>
|
|
|
|
|
<select name="AW">
|
|
|
|
|
<option value=0>None</option>
|
|
|
|
|
<option value=1>Brighter</option>
|
|
|
|
|
<option value=2>Accurate</option>
|
|
|
|
|
<option value=3>Dual</option>
|
|
|
|
|
<option value=4>Legacy</option>
|
|
|
|
|
</select>
|
|
|
|
|
<br></span><hr>
|
|
|
|
|
<button type="button" onclick="B()">Back</button><button type="submit">Save</button>
|
|
|
|
|
</form>
|
|
|
|
|
</body>
|
|
|
|
|
</html>
|
|
|
|
|
<!DOCTYPE html>
|
|
|
|
|
<html lang="en">
|
|
|
|
|
<head>
|
|
|
|
|
<meta charset="utf-8">
|
|
|
|
|
<meta name="viewport" content="width=500">
|
|
|
|
|
<title>LED Settings</title>
|
|
|
|
|
<script>
|
|
|
|
|
var d=document,laprev=55,maxB=1,maxM=5000,maxPB=4096,bquot=0; //maximum bytes for LED allocation: 5kB for 8266, 32kB for 32
|
|
|
|
|
function H()
|
|
|
|
|
{
|
|
|
|
|
window.open("https://github.com/Aircoookie/WLED/wiki/Settings#led-settings");
|
|
|
|
|
}
|
|
|
|
|
function B()
|
|
|
|
|
{
|
|
|
|
|
window.open("/settings","_self");
|
|
|
|
|
}
|
|
|
|
|
function gId(n){return d.getElementById(n);}
|
|
|
|
|
function off(n){
|
|
|
|
|
d.getElementsByName(n)[0].value = -1;
|
|
|
|
|
}
|
|
|
|
|
var timeout;
|
|
|
|
|
function showToast(text, error = false)
|
|
|
|
|
{
|
|
|
|
|
var x = gId("toast");
|
|
|
|
|
x.innerHTML = text;
|
|
|
|
|
x.className = error ? "error":"show";
|
|
|
|
|
clearTimeout(timeout);
|
|
|
|
|
x.style.animation = 'none';
|
|
|
|
|
timeout = setTimeout(function(){ x.className = x.className.replace("show", ""); }, 2900);
|
|
|
|
|
}
|
|
|
|
|
function bLimits(b,p,m) {
|
|
|
|
|
maxB = b; maxM = m; maxPB = p;
|
|
|
|
|
}
|
|
|
|
|
function pinsOK() {
|
|
|
|
|
var LCs = d.getElementsByTagName("input");
|
|
|
|
|
for (i=0; i<LCs.length; i++) {
|
|
|
|
|
var nm = LCs[i].name.substring(0,2);
|
|
|
|
|
// ignore IP address
|
|
|
|
|
if (nm=="L0" || nm=="L1" || nm=="L2" || nm=="L3") {
|
|
|
|
|
var n = LCs[i].name.substring(2);
|
|
|
|
|
var t = parseInt(d.getElementsByName("LT"+n)[0].value, 10); // LED type SELECT
|
|
|
|
|
if (t<16) continue;
|
|
|
|
|
}
|
|
|
|
|
//check for pin conflicts
|
|
|
|
|
if (nm=="L0" || nm=="L1" || nm=="L2" || nm=="L3" || nm=="L4" || nm=="RL" || nm=="BT" || nm=="IR")
|
|
|
|
|
if (LCs[i].value!="" && LCs[i].value!="-1") {
|
|
|
|
|
if (d.um_p && d.um_p.some((e)=>e==parseInt(LCs[i].value,10))) {alert(`Sorry, pins ${JSON.stringify(d.um_p)} can't be used.`);LCs[i].value="";LCs[i].focus();return false;}
|
|
|
|
|
else if (LCs[i].value > 5 && LCs[i].value < 12) {alert("Sorry, pins 6-11 can not be used.");LCs[i].value="";LCs[i].focus();return false;}
|
|
|
|
|
else if (!(nm == "IR" || nm=="BT") && LCs[i].value > 33) {alert("Sorry, pins >33 are input only.");LCs[i].value="";LCs[i].focus();return false;}
|
|
|
|
|
for (j=i+1; j<LCs.length; j++)
|
|
|
|
|
{
|
|
|
|
|
var n2 = LCs[j].name.substring(0,2);
|
|
|
|
|
if (n2=="L0" || n2=="L1" || n2=="L2" || n2=="L3" || n2=="L4" || n2=="RL" || n2=="BT" || n2=="IR") {
|
|
|
|
|
if (n2.substring(0,1)==="L") {
|
|
|
|
|
var m = LCs[j].name.substring(2);
|
|
|
|
|
var t2 = parseInt(d.getElementsByName("LT"+m)[0].value, 10);
|
|
|
|
|
if (t2<16) continue;
|
|
|
|
|
}
|
|
|
|
|
if (LCs[j].value!="" && LCs[i].value==LCs[j].value) {alert(`Pin conflict between ${LCs[i].name}/${LCs[j].name}!`);LCs[j].value="";LCs[j].focus();return false;}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
function trySubmit(e) {
|
|
|
|
|
d.Sf.data.value = '';
|
|
|
|
|
e.preventDefault();
|
|
|
|
|
if (!pinsOK()) {e.stopPropagation();return false;} // Prevent form submission and contact with server
|
|
|
|
|
if (bquot > 100) {var msg = "Too many LEDs for me to handle!"; if (maxM < 10000) msg += "\n\rConsider using an ESP32."; alert(msg);}
|
|
|
|
|
if (d.Sf.checkValidity()) d.Sf.submit(); //https://stackoverflow.com/q/37323914
|
|
|
|
|
}
|
|
|
|
|
function S(){GetV();setABL();}
|
|
|
|
|
function enABL()
|
|
|
|
|
{
|
|
|
|
|
var en = gId('able').checked;
|
|
|
|
|
d.Sf.LA.value = (en) ? laprev:0;
|
|
|
|
|
gId('abl').style.display = (en) ? 'inline':'none';
|
|
|
|
|
gId('psu2').style.display = (en) ? 'inline':'none';
|
|
|
|
|
if (d.Sf.LA.value > 0) setABL();
|
|
|
|
|
}
|
|
|
|
|
function enLA()
|
|
|
|
|
{
|
|
|
|
|
var val = d.Sf.LAsel.value;
|
|
|
|
|
d.Sf.LA.value = val;
|
|
|
|
|
gId('LAdis').style.display = (val == 50) ? 'inline':'none';
|
|
|
|
|
UI();
|
|
|
|
|
}
|
|
|
|
|
function setABL()
|
|
|
|
|
{
|
|
|
|
|
gId('able').checked = true;
|
|
|
|
|
d.Sf.LAsel.value = 50;
|
|
|
|
|
switch (parseInt(d.Sf.LA.value)) {
|
|
|
|
|
case 0: gId('able').checked = false; enABL(); break;
|
|
|
|
|
case 30: d.Sf.LAsel.value = 30; break;
|
|
|
|
|
case 35: d.Sf.LAsel.value = 35; break;
|
|
|
|
|
case 55: d.Sf.LAsel.value = 55; break;
|
|
|
|
|
case 255: d.Sf.LAsel.value = 255; break;
|
|
|
|
|
default: gId('LAdis').style.display = 'inline';
|
|
|
|
|
}
|
|
|
|
|
gId('m1').innerHTML = maxM;
|
|
|
|
|
d.getElementsByName("Sf")[0].addEventListener("submit", trySubmit);
|
|
|
|
|
UI();
|
|
|
|
|
}
|
|
|
|
|
//returns mem usage
|
|
|
|
|
function getMem(t, len, p0) {
|
|
|
|
|
if (t==2 || t==3) {
|
|
|
|
|
return len*(t+1);
|
|
|
|
|
}
|
|
|
|
|
if (t > 15 && t < 32) {
|
|
|
|
|
if (maxM < 10000 && p0==3) { //8266 DMA uses 5x the mem
|
|
|
|
|
if (t > 29) return len*20; //RGBW
|
|
|
|
|
return len*15;
|
|
|
|
|
} else if (maxM >= 10000) //ESP32 RMT uses double buffer?
|
|
|
|
|
{
|
|
|
|
|
if (t > 29) return len*8; //RGBW
|
|
|
|
|
return len*6;
|
|
|
|
|
}
|
|
|
|
|
if (t > 29) return len*4; //RGBW
|
|
|
|
|
return len*3;
|
|
|
|
|
}
|
|
|
|
|
if (t > 31 && t < 48) return 5;
|
|
|
|
|
if (t == 44 || t == 45) return len*4; //RGBW
|
|
|
|
|
return len*3;
|
|
|
|
|
}
|
|
|
|
|
function UI(change=false)
|
|
|
|
|
{
|
|
|
|
|
var isRGBW = false, memu = 0;
|
|
|
|
|
|
|
|
|
|
gId('ampwarning').style.display = (d.Sf.MA.value > 7200) ? 'inline':'none';
|
|
|
|
|
|
|
|
|
|
if (d.Sf.LA.value == 255) laprev = 12;
|
|
|
|
|
else if (d.Sf.LA.value > 0) laprev = d.Sf.LA.value;
|
|
|
|
|
|
|
|
|
|
// enable/disable LED fields
|
|
|
|
|
var s = d.getElementsByTagName("select");
|
|
|
|
|
for (i=0; i<s.length; i++) {
|
|
|
|
|
// is the field a LED type?
|
|
|
|
|
if (s[i].name.substring(0,2)=="LT") {
|
|
|
|
|
var n = s[i].name.substring(2);
|
|
|
|
|
var t = parseInt(s[i].value,10);
|
|
|
|
|
gId("p0d"+n).innerHTML = (t>=10 && t<=15) ? "IP address:" : (t > 49) ? "Data GPIO:" : (t >41) ? "GPIOs:" : "GPIO:";
|
|
|
|
|
gId("p1d"+n).innerHTML = (t > 49) ? "Clk GPIO:" : "";
|
|
|
|
|
var LK = d.getElementsByName("L1"+n)[0]; // clock pin
|
|
|
|
|
|
|
|
|
|
memu += getMem(t, d.getElementsByName("LC"+n)[0].value, d.getElementsByName("L0"+n)[0].value); // calc memory
|
|
|
|
|
|
|
|
|
|
// enumerate pins
|
|
|
|
|
for (p=1; p<5; p++) {
|
|
|
|
|
var LK = d.getElementsByName("L"+p+n)[0]; // secondary pins
|
|
|
|
|
if (!LK) continue;
|
|
|
|
|
if (((t>=10 && t<=15) && p<4) || (t>49 && p==1) || (t>41 && t < 50 && (p+40 < t))) // TYPE_xxxx values from const.h
|
|
|
|
|
{
|
|
|
|
|
// display pin field
|
|
|
|
|
LK.style.display = "inline";
|
|
|
|
|
LK.required = true;
|
|
|
|
|
} else {
|
|
|
|
|
// hide pin field
|
|
|
|
|
LK.style.display = "none";
|
|
|
|
|
LK.required = false;
|
|
|
|
|
LK.value="";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
if (change) {
|
|
|
|
|
// // blazoncek experimental extension
|
|
|
|
|
// gId("ew"+n).checked = (t == 30 || t == 31 || t == 44 || t == 45); // RGBW checkbox, TYPE_xxxx values from const.h
|
|
|
|
|
gId("ls"+n).value = n+1; // set LED start
|
|
|
|
|
if (t > 31 && t < 48) d.getElementsByName("LC"+n)[0].value = 1; // for sanity change analog count just to 1 LED
|
|
|
|
|
}
|
|
|
|
|
// // blazoncek experimental extension
|
|
|
|
|
// gId("ew"+n).onclick = (t > 31 && t < 48) ? (function(){return false}) : (function(){}); // prevent change for analog
|
|
|
|
|
// isRGBW |= gId("ew"+n).checked;
|
|
|
|
|
isRGBW |= (t == 30 || t == 31 || (t > 40 && t < 46 && t != 43)); // RGBW checkbox, TYPE_xxxx values from const.h
|
|
|
|
|
gId("co"+n).style.display = (t<16 || t == 41 || t == 42) ? "none":"inline"; // hide color order for PWM W & WW/CW
|
|
|
|
|
gId("dig"+n+"c").style.display = (t > 40 && t < 48) ? "none":"inline"; // hide count for analog
|
|
|
|
|
gId("dig"+n+"r").style.display = (t<16) ? "none":"inline"; // hide reversed for virtual
|
|
|
|
|
gId("dig"+n+"s").style.display = (t<16 || (t > 40 && t < 48)) ? "none":"inline"; // hide skip 1st for virtual & analog
|
|
|
|
|
gId("rev"+n).innerHTML = (t > 40 && t < 48) ? "Inverted":"Reverse (rotated 180°)"; // change reverse text for analog
|
|
|
|
|
gId("psd"+n).innerHTML = (t > 40 && t < 48) ? "Index:":"Start:"; // change analog start description
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// display white channel calculation method
|
|
|
|
|
var myC = d.querySelectorAll('.wc'),
|
|
|
|
|
l = myC.length;
|
|
|
|
|
for (i = 0; i < l; i++) {
|
|
|
|
|
myC[i].style.display = (isRGBW) ? 'inline':'none';
|
|
|
|
|
}
|
|
|
|
|
// check for pin conflicts
|
|
|
|
|
var LCs = d.getElementsByTagName("input");
|
|
|
|
|
var sLC = 0, maxLC = 0;
|
|
|
|
|
for (i=0; i<LCs.length; i++) {
|
|
|
|
|
var nm = LCs[i].name.substring(0,2); // field name
|
|
|
|
|
var n = LCs[i].name.substring(2); // bus number
|
|
|
|
|
// do we have a led count field but not total led count
|
|
|
|
|
if (nm=="LC" && LCs[i].name !== "LC") {
|
|
|
|
|
var c=parseInt(LCs[i].value,10);
|
|
|
|
|
/*if(gId("ls"+n).readOnly)*/ gId("ls"+n).value=sLC; // update led start field
|
|
|
|
|
if(c){sLC+=c;if(c>maxLC)maxLC=c;} // increase led count
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
// do we have led pins for digital leds
|
|
|
|
|
if (nm=="L0" || nm=="L1") {
|
|
|
|
|
var lc=d.getElementsByName("LC"+n)[0];
|
|
|
|
|
lc.max=maxPB; // update max led count value
|
|
|
|
|
}
|
|
|
|
|
// ignore IP address
|
|
|
|
|
if (nm=="L0" || nm=="L1" || nm=="L2" || nm=="L3") {
|
|
|
|
|
var t = parseInt(d.getElementsByName("LT"+n)[0].value, 10); // LED type SELECT
|
|
|
|
|
if (t<16) {
|
|
|
|
|
LCs[i].max = 255;
|
|
|
|
|
LCs[i].min = 0;
|
|
|
|
|
continue; // do not check conflicts
|
|
|
|
|
} else {
|
|
|
|
|
LCs[i].max = 33;
|
|
|
|
|
LCs[i].min = -1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// check for pin conflicts
|
|
|
|
|
if (nm=="L0" || nm=="L1" || nm=="L2" || nm=="L3" || nm=="L4" || nm=="RL" || nm=="BT" || nm=="IR")
|
|
|
|
|
if (LCs[i].value!="" && LCs[i].value!="-1") {
|
|
|
|
|
var p = []; // used pin array
|
|
|
|
|
if (d.um_p && Array.isArray(d.um_p)) for (k=0;k<d.um_p.length;k++) p.push(d.um_p[k]); // fill with reservations
|
|
|
|
|
for (j=0; j<LCs.length; j++) {
|
|
|
|
|
if (i==j) continue;
|
|
|
|
|
var n2 = LCs[j].name.substring(0,2);
|
|
|
|
|
if (n2=="L0" || n2=="L1" || n2=="L2" || n2=="L3" || n2=="L4" || n2=="RL" || n2=="BT" || n2=="IR") {
|
|
|
|
|
if (n2.substring(0,1)==="L") {
|
|
|
|
|
var m = LCs[j].name.substring(2);
|
|
|
|
|
var t2 = parseInt(d.getElementsByName("LT"+m)[0].value, 10);
|
|
|
|
|
if (t2<16) continue;
|
|
|
|
|
}
|
|
|
|
|
if (LCs[j].value!="" && LCs[j].value!="-1") p.push(parseInt(LCs[j].value,10)); // add current pin
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// now check for conflicts
|
|
|
|
|
if (p.some((e)=>e==parseInt(LCs[i].value,10))) LCs[i].style.color="red"; else LCs[i].style.color=parseInt(LCs[i].value,10)>33?"orange":"#fff";
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
// update total led count
|
|
|
|
|
if (gId("LC").readOnly) d.getElementsByName("LC")[0].value = sLC;
|
|
|
|
|
// if we are changing total led count update led count for 1st strip
|
|
|
|
|
if (d.activeElement == d.getElementsByName("LC")[0]) {
|
|
|
|
|
var o = d.getElementsByClassName("iST");
|
|
|
|
|
var i = o.length;
|
|
|
|
|
if (i == 1) d.getElementsByName("LC0")[0].value = d.getElementsByName("LC")[0].value;
|
|
|
|
|
}
|
|
|
|
|
// memory usage and warnings
|
|
|
|
|
gId('m0').innerHTML = memu;
|
|
|
|
|
bquot = memu / maxM * 100;
|
|
|
|
|
gId('dbar').style.background = `linear-gradient(90deg, ${bquot > 60 ? (bquot > 90 ? "red":"orange"):"#ccc"} 0 ${bquot}%%, #444 ${bquot}%% 100%%)`;
|
|
|
|
|
gId('ledwarning').style.display = (sLC > maxPB || maxLC > 800 || bquot > 80) ? 'inline':'none';
|
|
|
|
|
gId('ledwarning').style.color = (sLC > maxPB || maxLC > maxPB || bquot > 100) ? 'red':'orange';
|
|
|
|
|
gId('wreason').innerHTML = (bquot > 80) ? "80% of max. LED memory" +(bquot>100 ? ` (<b>WARNING: Using over ${maxM}B!</b>)` : "") : "800 LEDs per GPIO";
|
|
|
|
|
// calculate power
|
|
|
|
|
var val = Math.ceil((100 + sLC * laprev)/500)/2;
|
|
|
|
|
val = (val > 5) ? Math.ceil(val) : val;
|
|
|
|
|
var s = "";
|
|
|
|
|
var is12V = (d.Sf.LAsel.value == 30);
|
|
|
|
|
var isWS2815 = (d.Sf.LAsel.value == 255);
|
|
|
|
|
if (val < 1.02 && !is12V && !isWS2815)
|
|
|
|
|
{
|
|
|
|
|
s = "ESP 5V pin with 1A USB supply";
|
|
|
|
|
} else
|
|
|
|
|
{
|
|
|
|
|
s += is12V ? "12V ": isWS2815 ? "WS2815 12V " : "5V ";
|
|
|
|
|
s += val;
|
|
|
|
|
s += "A supply connected to LEDs";
|
|
|
|
|
}
|
|
|
|
|
var val2 = Math.ceil((100 + sLC * laprev)/1500)/2;
|
|
|
|
|
val2 = (val2 > 5) ? Math.ceil(val2) : val2;
|
|
|
|
|
var s2 = "(for most effects, ~";
|
|
|
|
|
s2 += val2;
|
|
|
|
|
s2 += "A is enough)<br>";
|
|
|
|
|
gId('psu').innerHTML = s;
|
|
|
|
|
gId('psu2').innerHTML = isWS2815 ? "" : s2;
|
|
|
|
|
gId("json").style.display = d.Sf.IT.value==8 ? "" : "none";
|
|
|
|
|
}
|
|
|
|
|
function lastEnd(i) {
|
|
|
|
|
if (i<1) return 0;
|
|
|
|
|
v = parseInt(d.getElementsByName("LS"+(i-1))[0].value) + parseInt(d.getElementsByName("LC"+(i-1))[0].value);
|
|
|
|
|
var t = parseInt(d.getElementsByName("LT"+(i-1))[0].value);
|
|
|
|
|
if (t > 31 && t < 48) v = 1; //PWM busses
|
|
|
|
|
if (isNaN(v)) return 0;
|
|
|
|
|
return v;
|
|
|
|
|
}
|
|
|
|
|
function addLEDs(n)
|
|
|
|
|
{
|
|
|
|
|
if (n>1) {maxB=n; gId("+").style.display="inline"; return;}
|
|
|
|
|
|
|
|
|
|
var o = d.getElementsByClassName("iST");
|
|
|
|
|
var i = o.length;
|
|
|
|
|
|
|
|
|
|
if ((n==1 && i>=maxB) || (n==-1 && i==0)) return;
|
|
|
|
|
|
|
|
|
|
var f = gId("mLC");
|
|
|
|
|
if (n==1) {
|
|
|
|
|
// npm run build has trouble minimizing spaces inside string
|
|
|
|
|
var cn = `<div class="iST">
|
|
|
|
|
<hr style="width:260px">
|
|
|
|
|
${i+1}:
|
|
|
|
|
<select name="LT${i}" onchange="UI(true)">
|
|
|
|
|
<option value="22" selected>WS281x</option>
|
|
|
|
|
<option value="30">SK6812 RGBW</option>
|
|
|
|
|
<option value="31">TM1814</option>
|
|
|
|
|
<option value="24">400kHz</option>
|
|
|
|
|
<option value="50">WS2801</option>
|
|
|
|
|
<option value="51">APA102</option>
|
|
|
|
|
<option value="52">LPD8806</option>
|
|
|
|
|
<option value="53">P9813</option>
|
|
|
|
|
<option value="41">PWM White</option>
|
|
|
|
|
<option value="42">PWM WWCW</option>
|
|
|
|
|
<option value="43">PWM RGB</option>
|
|
|
|
|
<option value="44">PWM RGBW</option>
|
|
|
|
|
<option value="45">PWM RGBWC</option>
|
|
|
|
|
<option value="10">DDP RGB (network)</option>
|
|
|
|
|
<option value="11">E1.31 RGB (network)</option>
|
|
|
|
|
<option value="12">ArtNet RGB (network)</option>
|
|
|
|
|
</select>
|
|
|
|
|
<div id="co${i}" style="display:inline">Color Order:
|
|
|
|
|
<select name="CO${i}">
|
|
|
|
|
<option value="0">GRB</option>
|
|
|
|
|
<option value="1">RGB</option>
|
|
|
|
|
<option value="2">BRG</option>
|
|
|
|
|
<option value="3">RBG</option>
|
|
|
|
|
<option value="4">BGR</option>
|
|
|
|
|
<option value="5">GBR</option>
|
|
|
|
|
</select></div>
|
|
|
|
|
<br>
|
|
|
|
|
<span id="p0d${i}">GPIO:</span><input type="number" name="L0${i}" min="0" max="33" required class="s" onchange="UI()"/>
|
|
|
|
|
<span id="p1d${i}"></span><input type="number" name="L1${i}" min="0" max="33" class="s" onchange="UI()"/>
|
|
|
|
|
<span id="p2d${i}"></span><input type="number" name="L2${i}" min="0" max="33" class="s" onchange="UI()"/>
|
|
|
|
|
<span id="p3d${i}"></span><input type="number" name="L3${i}" min="0" max="33" class="s" onchange="UI()"/>
|
|
|
|
|
<span id="p4d${i}"></span><input type="number" name="L4${i}" min="0" max="33" class="s" onchange="UI()"/>
|
|
|
|
|
<br>
|
|
|
|
|
<span id="psd${i}">Start:</span> <input type="number" name="LS${i}" id="ls${i}" class="l" min="0" max="8191" value="${lastEnd(i)}" disabled readonly required />
|
|
|
|
|
<div id="dig${i}c" style="display:inline">Count: <input type="number" name="LC${i}" class="l" min="0" max="${maxPB}" value="1" required oninput="UI()" /></div>
|
|
|
|
|
<br>
|
|
|
|
|
<div id="dig${i}r" style="display:inline"><span id="rev${i}">Reversed</span>: <input type="checkbox" name="CV${i}"></div>
|
|
|
|
|
<div id="dig${i}s" style="display:inline">Skip 1<sup>st</sup> LED: <input id="sl${i}" type="checkbox" name="SL${i}"></div>
|
|
|
|
|
<br>
|
|
|
|
|
</div>`;
|
|
|
|
|
f.insertAdjacentHTML("beforeend", cn);
|
|
|
|
|
}
|
|
|
|
|
if (n==-1) {
|
|
|
|
|
o[--i].remove();--i;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
gId("+").style.display = (i<maxB-1) ? "inline":"none";
|
|
|
|
|
gId("-").style.display = (i>0) ? "inline":"none";
|
|
|
|
|
|
|
|
|
|
UI();
|
|
|
|
|
}
|
|
|
|
|
function addBtn(i,p,t) {
|
|
|
|
|
var c = gId("btns").innerHTML;
|
|
|
|
|
var bt = "BT" + i;
|
|
|
|
|
var be = "BE" + i;
|
|
|
|
|
c += `Button ${i} GPIO: <input type="number" min="-1" max="40" name="${bt}" onchange="UI()" class="s" value="${p}">`;
|
|
|
|
|
c += `<select name="${be}">`
|
|
|
|
|
c += `<option value="0" ${t==0?"selected":""}>Disabled</option>`;
|
|
|
|
|
c += `<option value="2" ${t==2?"selected":""}>Pushbutton</option>`;
|
|
|
|
|
c += `<option value="3" ${t==3?"selected":""}>Push inverted</option>`;
|
|
|
|
|
c += `<option value="4" ${t==4?"selected":""}>Switch</option>`;
|
|
|
|
|
c += `<option value="5" ${t==5?"selected":""}>PIR sensor</option>`;
|
|
|
|
|
c += `<option value="6" ${t==6?"selected":""}>Touch</option>`;
|
|
|
|
|
c += `<option value="7" ${t==7?"selected":""}>Analog</option>`;
|
|
|
|
|
c += `<option value="8" ${t==8?"selected":""}>Analog inverted</option>`;
|
|
|
|
|
c += `</select>`;
|
|
|
|
|
c += `<span style="cursor: pointer;" onclick="off('${bt}')"> ×</span><br>`;
|
|
|
|
|
gId("btns").innerHTML = c;
|
|
|
|
|
}
|
|
|
|
|
function uploadFile(name) {
|
|
|
|
|
var req = new XMLHttpRequest();
|
|
|
|
|
req.addEventListener('load', function(){showToast(this.responseText,this.status >= 400)});
|
|
|
|
|
req.addEventListener('error', function(e){showToast(e.stack,true);});
|
|
|
|
|
req.open("POST", "/upload");
|
|
|
|
|
var formData = new FormData();
|
|
|
|
|
formData.append("data", d.Sf.data.files[0], name);
|
|
|
|
|
req.send(formData);
|
|
|
|
|
d.Sf.data.value = '';
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
function GetV()
|
|
|
|
|
{
|
|
|
|
|
//values injected by server while sending HTML
|
|
|
|
|
//maxM=5000;maxPB=1536;d.um_p=[1,6,7,8,9,10,11];addLEDs(5);d.Sf.LC.value=250;addLEDs(1);d.Sf.L00.value=2;d.Sf.L10.value=0;d.Sf.LC0.value=250;d.Sf.LT0.value=22;d.Sf.CO0.value=0;d.Sf.LS0.value=0;d.Sf.LS0.checked=0;d.Sf.MA.value=5400;d.Sf.LA.value=55;d.getElementsByClassName("pow")[0].innerHTML="350mA";d.Sf.CA.value=40;d.Sf.AW.value=3;d.Sf.BO.checked=0;d.Sf.BP.value=3;d.Sf.GB.checked=0;d.Sf.GC.checked=1;d.Sf.TF.checked=1;d.Sf.TD.value=700;d.Sf.PF.checked=0;d.Sf.BF.value=64;d.Sf.TB.value=0;d.Sf.TL.value=60;d.Sf.TW.value=1;d.Sf.PB.selectedIndex=0;d.Sf.RL.value=12;d.Sf.RM.checked=0;addBtn(0,0,2);addBtn(1,3,4);addBtn(2,-1,0);d.Sf.IR.value=-1;
|
|
|
|
|
}
|
|
|
|
|
</script>
|
|
|
|
|
<style>@import url("style.css");</style>
|
|
|
|
|
</head>
|
|
|
|
|
<body onload="S()">
|
|
|
|
|
<form id="form_s" name="Sf" method="post">
|
|
|
|
|
<div class="toprow">
|
|
|
|
|
<div class="helpB"><button type="button" onclick="H()">?</button></div>
|
|
|
|
|
<button type="button" onclick="B()">Back</button><button type="submit">Save</button><hr>
|
|
|
|
|
</div>
|
|
|
|
|
<h2>LED & Hardware setup</h2>
|
|
|
|
|
Total LED count: <input name="LC" id="LC" type="number" min="1" max="8192" oninput="UI()" disabled required readonly><br>
|
|
|
|
|
<i>Recommended power supply for brightest white:</i><br>
|
|
|
|
|
<b><span id="psu">?</span></b><br>
|
|
|
|
|
<span id="psu2"><br></span>
|
|
|
|
|
<br>
|
|
|
|
|
Enable automatic brightness limiter: <input type="checkbox" name="ABen" onchange="enABL()" id="able"><br>
|
|
|
|
|
<div id="abl">
|
|
|
|
|
Maximum Current: <input name="MA" type="number" class="l" min="250" max="65000" oninput="UI()" required> mA<br>
|
|
|
|
|
<div id="ampwarning" style="color: orange; display: none;">
|
|
|
|
|
⚠ Your power supply provides high current.<br>
|
|
|
|
|
To improve the safety of your setup,<br>
|
|
|
|
|
please use thick cables,<br>
|
|
|
|
|
multiple power injection points and a fuse!<br>
|
|
|
|
|
</div>
|
|
|
|
|
<i>Automatically limits brightness to stay close to the limit.<br>
|
|
|
|
|
Keep at <1A if powering LEDs directly from the ESP 5V pin!<br>
|
|
|
|
|
If you are using an external power supply, enter its rating.<br>
|
|
|
|
|
(Current estimated usage: <span class="pow">unknown</span>)</i><br><br>
|
|
|
|
|
LED voltage (Max. current for a single LED):<br>
|
|
|
|
|
<select name="LAsel" onchange="enLA()">
|
|
|
|
|
<option value="55" selected>5V default (55mA)</option>
|
|
|
|
|
<option value="35">5V efficient (35mA)</option>
|
|
|
|
|
<option value="30">12V (30mA)</option>
|
|
|
|
|
<option value="255">WS2815 (12mA)</option>
|
|
|
|
|
<option value="50">Custom</option>
|
|
|
|
|
</select><br>
|
|
|
|
|
<span id="LAdis" style="display: none;">Custom max. current per LED: <input name="LA" type="number" min="0" max="255" id="la" oninput="UI()" required> mA<br></span>
|
|
|
|
|
<i>Keep at default if you are unsure about your type of LEDs.</i><br>
|
|
|
|
|
</div>
|
|
|
|
|
<h3>Hardware setup</h3>
|
|
|
|
|
<div id="mLC">LED outputs:</div>
|
|
|
|
|
<hr style="width:260px">
|
|
|
|
|
<button type="button" id="+" onclick="addLEDs(1)">+</button>
|
|
|
|
|
<button type="button" id="-" onclick="addLEDs(-1)">-</button><br>
|
|
|
|
|
LED Memory Usage: <span id="m0">0</span> / <span id="m1">?</span> B<br>
|
|
|
|
|
<div id="dbar" style="display:inline-block; width: 100px; height: 10px; border-radius: 20px;"></div><br>
|
|
|
|
|
<div id="ledwarning" style="color: orange; display: none;">
|
|
|
|
|
⚠ You might run into stability or lag issues.<br>
|
|
|
|
|
Use less than <span id="wreason">800 LEDs per pin</span> for the best experience!<br>
|
|
|
|
|
</div>
|
|
|
|
|
<hr style="width:260px">
|
|
|
|
|
Create a segment for each output: <input type="checkbox" name="MS"> <br>
|
|
|
|
|
<hr style="width:260px">
|
|
|
|
|
<div id="btns"></div>
|
|
|
|
|
Touch threshold: <input type="number" min="0" max="100" name="TT" required><br>
|
|
|
|
|
IR GPIO: <input type="number" min="-1" max="40" name="IR" onchange="UI()" class="s"><select name="IT" onchange="UI()">
|
|
|
|
|
<option value=0>Remote disabled</option>
|
|
|
|
|
<option value=1>24-key RGB</option>
|
|
|
|
|
<option value=2>24-key with CT</option>
|
|
|
|
|
<option value=3>40-key blue</option>
|
|
|
|
|
<option value=4>44-key RGB</option>
|
|
|
|
|
<option value=5>21-key RGB</option>
|
|
|
|
|
<option value=6>6-key black</option>
|
|
|
|
|
<option value=7>9-key red</option>
|
|
|
|
|
<option value=8>JSON remote</option>
|
|
|
|
|
</select><span style="cursor: pointer;" onclick="off('IR')"> ×</span><br>
|
|
|
|
|
<div id="json" style="display:none;">JSON file: <input type="file" name="data" accept=".json"> <input type="button" value="Upload" onclick="uploadFile('/ir.json');"><br></div>
|
|
|
|
|
<div id="toast"></div>
|
|
|
|
|
<a href="https://github.com/Aircoookie/WLED/wiki/Infrared-Control" target="_blank">IR info</a><br>
|
|
|
|
|
Relay GPIO: <input type="number" min="-1" max="33" name="RL" onchange="UI()" class="s"> invert <input type="checkbox" name="RM"><span style="cursor: pointer;" onclick="off('RL')"> ×</span><br>
|
|
|
|
|
<hr style="width:260px">
|
|
|
|
|
<h3>Defaults</h3>
|
|
|
|
|
Turn LEDs on after power up/reset: <input type="checkbox" name="BO"><br>
|
|
|
|
|
Default brightness: <input name="CA" type="number" class="m" min="0" max="255" required> (0-255)<br><br>
|
|
|
|
|
Apply preset <input name="BP" type="number" class="m" min="0" max="250" required> at boot (0 uses defaults)
|
|
|
|
|
<br><br>
|
|
|
|
|
Use Gamma correction for color: <input type="checkbox" name="GC"> (strongly recommended)<br>
|
|
|
|
|
Use Gamma correction for brightness: <input type="checkbox" name="GB"> (not recommended)<br><br>
|
|
|
|
|
Brightness factor: <input name="BF" type="number" class="m" min="1" max="255" required> %
|
|
|
|
|
<h3>Transitions</h3>
|
|
|
|
|
Crossfade: <input type="checkbox" name="TF"><br>
|
|
|
|
|
Transition Time: <input name="TD" type="number" class="xl" min="0" max="65500"> ms<br>
|
|
|
|
|
Enable Palette transitions: <input type="checkbox" name="PF">
|
|
|
|
|
<h3>Timed light</h3>
|
|
|
|
|
Default Duration: <input name="TL" type="number" class="m" min="1" max="255" required> min<br>
|
|
|
|
|
Default Target brightness: <input name="TB" type="number" class="m" min="0" max="255" required><br>
|
|
|
|
|
Mode:
|
|
|
|
|
<select name="TW">
|
|
|
|
|
<option value="0">Wait and set</option>
|
|
|
|
|
<option value="1">Fade</option>
|
|
|
|
|
<option value="2">Fade Color</option>
|
|
|
|
|
<option value="3">Sunrise</option>
|
|
|
|
|
</select>
|
|
|
|
|
<h3>Advanced</h3>
|
|
|
|
|
Palette blending:
|
|
|
|
|
<select name="PB">
|
|
|
|
|
<option value="0">Linear (wrap if moving)</option>
|
|
|
|
|
<option value="1">Linear (always wrap)</option>
|
|
|
|
|
<option value="2">Linear (never wrap)</option>
|
|
|
|
|
<option value="3">None (not recommended)</option>
|
|
|
|
|
</select><br>
|
|
|
|
|
<span class="wc">
|
|
|
|
|
Auto-calculate white channel from RGB:<br>
|
|
|
|
|
<select name="AW">
|
|
|
|
|
<option value=0>None</option>
|
|
|
|
|
<option value=1>Brighter</option>
|
|
|
|
|
<option value=2>Accurate</option>
|
|
|
|
|
<option value=3>Dual</option>
|
|
|
|
|
<option value=4>Legacy</option>
|
|
|
|
|
</select>
|
|
|
|
|
<br></span><hr>
|
|
|
|
|
<button type="button" onclick="B()">Back</button><button type="submit">Save</button>
|
|
|
|
|
</form>
|
|
|
|
|
</body>
|
|
|
|
|
</html>
|
|
|
|
|