//page js var loc = false, locip; var noNewSegs = false; var isOn = false, nlA = false, isLv = false, isInfo = false, isNodes = false, syncSend = false, syncTglRecv = true, isRgbw = false, cct = false; var whites = [0,0,0]; var selColors; var expanded = [false]; var powered = [true]; var nlDur = 60, nlTar = 0; var nlMode = false; var selectedFx = 0, prevFx = -1; var selectedPal = 0; var csel = 0; var currentPreset = -1, prevPS = -1; var lastUpdate = 0; var segCount = 0, ledCount = 0, lowestUnused = 0, maxSeg = 0, lSeg = 0; var pcMode = false, pcModeA = false, lastw = 0, wW; var tr = 7; var d = document; var palettesData; var fxdata = []; var pJson = {}, eJson = {}, lJson = {}; var pN = "", pI = 0, pNum = 0; var pmt = 1, pmtLS = 0, pmtLast = 0; var lastinfo = {}; var ws, cpick, ranges; var cfg = { theme:{base:"dark", bg:{url:""}, alpha:{bg:0.6,tab:0.8}, color:{bg:""}}, comp :{colors:{picker: true, rgb: false, quick: true, hex: false}, labels:true, pcmbot:false, pid:true, seglen:false, css:true, hdays:false} }; var hol = [ [0,11,24,4,"https://aircoookie.github.io/xmas.png"], // christmas [0,2,17,1,"https://images.alphacoders.com/491/491123.jpg"], // st. Patrick's day [2022,3,17,2,"https://aircoookie.github.io/easter.png"], [2023,3,9,2,"https://aircoookie.github.io/easter.png"], [2024,2,31,2,"https://aircoookie.github.io/easter.png"] ]; function handleVisibilityChange() {if (!d.hidden && new Date () - lastUpdate > 3000) requestJson();} function sCol(na, col) {d.documentElement.style.setProperty(na, col);} function gId(c) {return d.getElementById(c);} function gEBCN(c) {return d.getElementsByClassName(c);} function isEmpty(o) {return Object.keys(o).length === 0;} function isObj(i) {return (i && typeof i === 'object' && !Array.isArray(i));} function applyCfg() { cTheme(cfg.theme.base === "light"); var bg = cfg.theme.color.bg; if (bg) sCol('--c-1', bg); var ccfg = cfg.comp.colors; gId('hexw').style.display = ccfg.hex ? "block":"none"; gId('picker').style.display = ccfg.picker ? "block":"none"; gId('vwrap').style.display = ccfg.picker ? "block":"none"; gId('kwrap').style.display = ccfg.picker ? "block":"none"; gId('rgbwrap').style.display = ccfg.rgb ? "block":"none"; gId('qcs-w').style.display = ccfg.quick ? "block":"none"; var l = cfg.comp.labels; var e = d.querySelectorAll('.tab-label'); for (var i=0; i=hs && today<=he) img.src = hol[i][4]; } } img.addEventListener('load', (e) => { var a = parseFloat(cfg.theme.alpha.bg); if (isNaN(a)) a = 0.6; bg.style.opacity = a; bg.style.backgroundImage = `url(${img.src})`; img = null; }); } function loadSkinCSS(cId) { if (!gId(cId)) // check if element exists { var h = d.getElementsByTagName('head')[0]; var l = d.createElement('link'); l.id = cId; l.rel = 'stylesheet'; l.type = 'text/css'; l.href = (loc?`http://${locip}`:'.') + '/skin.css'; l.media = 'all'; h.appendChild(l); } } function onLoad() { if (window.location.protocol == "file:") { loc = true; locip = localStorage.getItem('locIp'); if (!locip) { locip = prompt("File Mode. Please enter WLED IP!"); localStorage.setItem('locIp', locip); } } var sett = localStorage.getItem('wledUiCfg'); if (sett) cfg = mergeDeep(cfg, JSON.parse(sett)); resetPUtil(); applyCfg(); if (cfg.comp.hdays) { //load custom holiday list fetch((loc?`http://${locip}`:'.') + "/holidays.json", { // may be loaded from external source method: 'get' }) .then((res)=>{ //if (!res.ok) showErrorToast(); return res.json(); }) .then((json)=>{ if (Array.isArray(json)) hol = json; //TODO: do some parsing first }) .catch((e)=>{ console.log("No array of holidays in holidays.json. Defaults loaded."); }) .finally(()=>{ loadBg(cfg.theme.bg.url); }); } else loadBg(cfg.theme.bg.url); if (cfg.comp.css) loadSkinCSS('skinCss'); var cd = gId('csl').children; for (var i = 0; i < cd.length; i++) cd[i].style.backgroundColor = "rgb(0, 0, 0)"; selectSlot(0); updateTablinks(0); pmtLS = localStorage.getItem('wledPmt'); // Load initial data loadPalettes(()=>{ loadPalettesData(redrawPalPrev); loadFX(()=>{ loadFXData(); setTimeout(()=>{ //ESP8266 can't handle quick requests loadPresets(()=>{ requestJson(); // will create WS }); },100); }); }); resetUtil(); d.addEventListener("visibilitychange", handleVisibilityChange, false); size(); gId("cv").style.opacity=0; if (localStorage.getItem('pcm') == "true") togglePcMode(true); var sls = d.querySelectorAll('input[type="range"]'); for (var sl of sls) { sl.addEventListener('touchstart', toggleBubble); sl.addEventListener('touchend', toggleBubble); } } function updateTablinks(tabI) { var tablinks = gEBCN("tablinks"); for (var i of tablinks) i.classList.remove("active"); if (pcMode) return; tablinks[tabI].classList.add("active"); } function openTab(tabI, force = false) { if (pcMode && !force) return; iSlide = tabI; _C.classList.toggle('smooth', false); _C.style.setProperty('--i', iSlide); updateTablinks(tabI); } var timeout; function showToast(text, error = false) { if (error) gId('connind').style.backgroundColor = "var(--c-r)"; var x = gId("toast"); x.innerHTML = text; x.classList.add(error ? "error":"show"); clearTimeout(timeout); x.style.animation = 'none'; timeout = setTimeout(()=>{ x.classList.remove("show"); }, 2900); if (error) console.log(text); } function showErrorToast() { // if we received a timeout force WS reconnect setTimeout(makeWS,500); showToast('Connection to light failed!', true); } function clearErrorToast() { var x = gId("toast"); if (x.classList.contains("error")) { clearTimeout(timeout); timeout = setTimeout(()=>{ x.classList.remove("show"); x.classList.remove("error"); }, 10000); } } function getRuntimeStr(rt) { var t = parseInt(rt); var days = Math.floor(t/86400); var hrs = Math.floor((t - days*86400)/3600); var mins = Math.floor((t - days*86400 - hrs*3600)/60); var str = days ? (days + " " + (days == 1 ? "day" : "days") + ", ") : ""; str += (hrs || days) ? (hrs + " " + (hrs == 1 ? "hour" : "hours")) : ""; if (!days && hrs) str += ", "; if (t > 59 && !days) str += mins + " min"; if (t < 3600 && t > 59) str += ", "; if (t < 3600) str += (t - mins*60) + " sec"; return str; } function inforow(key, val, unit = "") { return `${key}${val}${unit}`; } function getLowestUnusedP() { var l = 1; for (var key in pJson) if (key == l) l++; if (l > 250) l = 250; return l; } function checkUsed(i) { var id = gId(`p${i}id`).value; if (pJson[id] && (i == 0 || id != i)) gId(`p${i}warn`).innerHTML = `⚠ Overwriting ${pName(id)}!`; else gId(`p${i}warn`).innerHTML = ""; } function pName(i) { var n = "Preset " + i; if (pJson && pJson[i] && pJson[i].n) n = pJson[i].n; return n; } function isPlaylist(i) { return pJson[i].playlist && pJson[i].playlist.ps; } function papiVal(i) { if (!pJson || !pJson[i]) return ""; var o = Object.assign({},pJson[i]); if (o.win) return o.win; delete o.n; delete o.p; delete o.ql; return JSON.stringify(o); } function qlName(i) { if (!pJson || !pJson[i] || !pJson[i].ql) return ""; return pJson[i].ql; } function cpBck() { var copyText = gId("bck"); copyText.select(); copyText.setSelectionRange(0, 999999); d.execCommand("copy"); showToast("Copied to clipboard!"); } function presetError(empty) { var hasBackup = false; var bckstr = ""; try { bckstr = localStorage.getItem("wledP"); if (bckstr.length > 10) hasBackup = true; } catch (e) {} var cn = `
`; if (empty) cn += `You have no presets yet!`; else cn += `Sorry, there was an issue loading your presets!`; if (hasBackup) { cn += `

`; if (empty) cn += `However, there is backup preset data of a previous installation available.
(Saving a preset will hide this and overwrite the backup)`; else cn += `Here is a backup of the last known good state:`; cn += `
`; } cn += `
`; gId('pcont').innerHTML = cn; if (hasBackup) gId('bck').value = bckstr; } function loadPresets(callback = null) { //1st boot (because there is a callback) if (callback && pmt == pmtLS && pmt > 0) { //we have a copy of the presets in local storage and don't need to fetch another one populatePresets(true); pmtLast = pmt; callback(); return; } //afterwards if (!callback && pmt == pmtLast) return; pmtLast = pmt; var url = (loc?`http://${locip}`:'') + '/presets.json'; fetch(url, { method: 'get' }) .then(res => { if (!res.ok) showErrorToast(); return res.json(); }) .then(json => { pJson = json; populatePresets(); }) .catch((e)=>{ showToast(e, true); presetError(false); }) .finally(()=>{ if (callback) setTimeout(callback,99); }); } function loadPalettes(callback = null) { var url = (loc?`http://${locip}`:'') + '/json/palettes'; fetch(url, { method: 'get' }) .then((res)=>{ if (!res.ok) showErrorToast(); return res.json(); }) .then((json)=>{ lJson = Object.entries(json); populatePalettes(); }) .catch((e)=>{ showToast(e, true); presetError(false); }) .finally(()=>{ if (callback) callback(); updateUI(); }); } function loadFX(callback = null) { var url = (loc?`http://${locip}`:'') + '/json/effects'; fetch(url, { method: 'get' }) .then((res)=>{ if (!res.ok) showErrorToast(); return res.json(); }) .then((json)=>{ eJson = Object.entries(json); populateEffects(); }) .catch((e)=>{ showToast(e, true); presetError(false); }) .finally(()=>{ if (callback) callback(); updateUI(); }); } function loadFXData(callback = null) { var url = (loc?`http://${locip}`:'') + '/json/fxdata'; fetch(url, { method: 'get' }) .then((res)=>{ if (!res.ok) showErrorToast(); return res.json(); }) .then((json)=>{ fxdata = json||[]; // add default value for Solid fxdata.shift() fxdata.unshift("@;!;0"); }) .catch((e)=>{ fxdata = []; showToast(e, true); }) .finally(()=>{ if (callback) callback(); updateUI(); }); } var pQL = []; function populateQL() { var cn = ""; if (pQL.length > 0) { pQL.sort((a,b) => (a[0]>b[0])); cn += `

Quick load

`; for (var key of (pQL||[])) { cn += ``; } } gId('pql').innerHTML = cn; } function populatePresets(fromls) { if (fromls) pJson = JSON.parse(localStorage.getItem("wledP")); if (!pJson) {setTimeout(loadPresets,250); return;} delete pJson["0"]; var cn = `

All presets

`; var arr = Object.entries(pJson); arr.sort(cmpP); pQL = []; var is = []; pNum = 0; for (var key of (arr||[])) { if (!isObj(key[1])) continue; let i = parseInt(key[0]); var qll = key[1].ql; if (qll) pQL.push([i, qll, pName(i)]); is.push(i); cn += `
`; if (cfg.comp.pid) cn += `
${i}
`; cn += `
${isPlaylist(i)?"":""}${pName(i)}
`; pNum++; } gId('pcont').innerHTML = cn; if (pNum > 0) { if (pmtLS != pmt && pmt != 0) { localStorage.setItem("wledPmt", pmt); pJson["0"] = {}; localStorage.setItem("wledP", JSON.stringify(pJson)); } pmtLS = pmt; //for (var a = 0; a < is.length; a++) { // let i = is[a]; // if (expanded[i+100]) expand(i+100); //} } else { presetError(true); } updatePA(); populateQL(); } function parseInfo() { var li = lastinfo; var name = li.name; gId('namelabel').innerHTML = name; // if (name === "Dinnerbone") d.documentElement.style.transform = "rotate(180deg)"; if (li.live) name = "(Live) " + name; if (loc) name = "(L) " + name; d.title = name; isRgbw = li.leds.wv; ledCount = li.leds.count; syncTglRecv = li.str; maxSeg = li.leds.maxseg; pmt = li.fs.pmt; cct = li.leds.cct; } function populateInfo(i) { var cn=""; var heap = i.freeheap/1000; heap = heap.toFixed(1); var pwr = i.leds.pwr; var pwru = "Not calculated"; if (pwr > 1000) {pwr /= 1000; pwr = pwr.toFixed((pwr > 10) ? 0 : 1); pwru = pwr + " A";} else if (pwr > 0) {pwr = 50 * Math.round(pwr/50); pwru = pwr + " mA";} var urows=""; if (i.u) { for (const [k, val] of Object.entries(i.u)) { if (val[1]) urows += inforow(k,val[0],val[1]); else urows += inforow(k,val); } } var vcn = "Kuuhaku"; if (i.ver.startsWith("0.13.")) vcn = "Toki"; if (i.ver.includes("-bl")) vcn = "Ryujin"; if (i.cn) vcn = i.cn; cn += `v${i.ver} "${vcn}"

${urows} ${inforow("Build",i.vid)} ${inforow("Signal strength",i.wifi.signal +"% ("+ i.wifi.rssi, " dBm)")} ${inforow("Uptime",getRuntimeStr(i.uptime))} ${inforow("Free heap",heap," kB")} ${i.psram?inforow("Free PSRAM",(i.psram/1024).toFixed(1)," kB"):""} ${inforow("Estimated current",pwru)} ${inforow("Average FPS",i.leds.fps)} ${inforow("MAC address",i.mac)} ${inforow("Filesystem",i.fs.u + "/" + i.fs.t + " kB (" +Math.round(i.fs.u*100/i.fs.t) + "%)")} ${inforow("Environment",i.arch + " " + i.core + " (" + i.lwip + ")")}
`; gId('kv').innerHTML = cn; } function populateSegments(s) { var cn = ""; segCount = 0; lowestUnused = 0; lSeg = 0; for (var y = 0; y < (s.seg||[]).length; y++) { segCount++; var inst = s.seg[y]; let i = parseInt(inst.id); powered[i] = inst.on; if (i == lowestUnused) lowestUnused = i+1; if (i > lSeg) lSeg = i; cn += `
${inst.n ? inst.n : "Segment "+i}
Start LED ${cfg.comp.seglen?"LED count":"Stop LED"} Offset
Grouping Spacing Apply
`; } gId('segcont').innerHTML = cn; if (lowestUnused >= maxSeg) { gId('segutil').innerHTML = 'Maximum number of segments reached.'; noNewSegs = true; } else if (noNewSegs) { resetUtil(); noNewSegs = false; } for (var i = 0; i <= lSeg; i++) { updateLen(i); updateTrail(gId(`seg${i}bri`)); gId(`segr${i}`).style.display = "none"; } if (segCount < 2) gId(`segd${lSeg}`).style.display = "none"; if (!noNewSegs && (cfg.comp.seglen?parseInt(gId(`seg${lSeg}s`).value):0)+parseInt(gId(`seg${lSeg}e`).value) 1) ? "inline":"none"; } function populateEffects() { var effects = eJson; var html = ""; effects.shift(); //remove solid for (let i = 0; i < effects.length; i++) effects[i] = {id: effects[i][0], name:effects[i][1]}; effects.sort((a,b) => (a.name).localeCompare(b.name)); effects.unshift({ "id": 0, "name": "Solid@;!;0" }); for (let i = 0; i < effects.length; i++) { // WLEDSR: add slider and color control to setX (used by requestjson) if (effects[i].name.indexOf("Reserved") < 0) { var posAt = effects[i].name.indexOf("@"); var extra = ''; if (posAt > 0) extra = effects[i].name.substr(posAt); else posAt = 999; html += generateListItemHtml( 'fx', effects[i].id, effects[i].name.substr(0,posAt), 'setX', '','', extra ); } } gId('fxlist').innerHTML=html; } function populatePalettes() { var palettes = lJson; palettes.shift(); //remove default for (let i = 0; i < palettes.length; i++) { palettes[i] = { "id": palettes[i][0], "name": palettes[i][1] }; } palettes.sort((a,b) => (a.name).localeCompare(b.name)); palettes.unshift({ "id": 0, "name": "Default" }); var html = ""; for (let i = 0; i < palettes.length; i++) { html += generateListItemHtml( 'palette', palettes[i].id, palettes[i].name, 'setPalette', `
` ); } gId('pallist').innerHTML=html; } function redrawPalPrev() { let palettes = d.querySelectorAll('#pallist .lstI'); for (var pal of (palettes||[])) { let lP = pal.querySelector('.lstIprev'); if (lP) { lP.style = genPalPrevCss(pal.dataset.id); } } } function genPalPrevCss(id) { if (!palettesData) return; var paletteData = palettesData[id]; if (!paletteData) return 'display: none'; // We need at least two colors for a gradient if (paletteData.length == 1) { paletteData[1] = paletteData[0]; if (Array.isArray(paletteData[1])) { paletteData[1][0] = 255; } } var gradient = []; for (let j = 0; j < paletteData.length; j++) { const element = paletteData[j]; let r; let g; let b; let index = false; if (Array.isArray(element)) { index = element[0]/255*100; r = element[1]; g = element[2]; b = element[3]; } else if (element == 'r') { r = Math.random() * 255; g = Math.random() * 255; b = Math.random() * 255; } else { if (selColors) { let e = element[1] - 1; r = selColors[e][0]; g = selColors[e][1]; b = selColors[e][2]; } } if (index === false) { index = j / paletteData.length * 100; } gradient.push(`rgb(${r},${g},${b}) ${index}%`); } return `background: linear-gradient(to right,${gradient.join()});`; } function generateListItemHtml(listName, id, name, clickAction, extraHtml = '', extraClass = '', extraPar = '') { return `
${name}
${extraHtml}
`; } function btype(b) { switch (b) { case 32: return "ESP32"; case 82: return "ESP8266"; } return "?"; } function bname(o) { if (o.name=="WLED") return o.ip; return o.name; } function populateNodes(i,n) { var cn=""; var urows=""; var nnodes = 0; if (n.nodes) { n.nodes.sort((a,b) => (a.name).localeCompare(b.name)); for (var x=0;x${bname(o)}`; urows += inforow(url,`${btype(o.type)}
${o.vid==0?"N/A":o.vid}`); nnodes++; } } } if (i.ndc < 0) cn += `Instance List is disabled.`; else if (nnodes == 0) cn += `No other instances found.`; cn += ` ${inforow("Current instance:",i.name)} ${urows}
`; gId('kn').innerHTML = cn; } function loadNodes() { var url = (loc?`http://${locip}`:'') + '/json/nodes'; fetch(url, { method: 'get' }) .then((res)=>{ if (!res.ok) showToast('Could not load Node list!', true); return res.json(); }) .then((json)=>{ clearErrorToast(); populateNodes(lastinfo, json); }) .catch((e)=>{ showToast(e, true); }); } //update the 'sliderdisplay' background div of a slider for a visual indication of slider position function updateTrail(e) { if (e==null) return; var max = e.hasAttribute('max') ? e.attributes.max.value : 255; var perc = e.value * 100 / max; perc = parseInt(perc); if (perc < 50) perc += 2; var val = `linear-gradient(90deg, var(--c-f) ${perc}%, var(--c-4) ${perc}%)`; e.parentNode.getElementsByClassName('sliderdisplay')[0].style.background = val; var b = e.parentNode.parentNode.getElementsByTagName('output')[0]; if (b) b.innerHTML = e.value; } //rangetouch slider function function toggleBubble(e) { var b = e.target.parentNode.parentNode.getElementsByTagName('output')[0]; b.classList.toggle('sliderbubbleshow'); } //updates segment length upon input of segment values function updateLen(s) { if (!gId(`seg${s}s`)) return; var start = parseInt(gId(`seg${s}s`).value); var stop = parseInt(gId(`seg${s}e`).value); var len = stop - (cfg.comp.seglen?0:start); var out = "(delete)"; if (len > 1) { out = `${len} LEDs`; } else if (len == 1) { out = "1 LED"; } if (gId(`seg${s}grp`) != null) { var grp = parseInt(gId(`seg${s}grp`).value); var spc = parseInt(gId(`seg${s}spc`).value); if (grp == 0) grp = 1; var virt = Math.ceil(len/(grp + spc)); if (!isNaN(virt) && (grp > 1 || spc > 0)) out += ` (${virt} virtual)`; } gId(`seg${s}len`).innerHTML = out; } //updates background color of currently selected preset function updatePA() { var ps = gEBCN("pres"); for (let i = 0; i < ps.length; i++) { ps[i].classList.remove('selected');; } ps = gEBCN("psts"); for (let i = 0; i < ps.length; i++) { ps[i].classList.remove('selected');; } if (currentPreset > 0) { var acv = gId(`p${currentPreset}o`); if (acv && !expanded[currentPreset+100]) { acv.classList.add('selected'); // scroll selected preset into view (on WS refresh) acv.scrollIntoView({ behavior: 'smooth', block: 'center', }); } acv = gId(`p${currentPreset}qlb`); if (acv) acv.classList.add('selected'); } } function updateUI() { gId('buttonPower').className = (isOn) ? "active":""; gId('buttonNl').className = (nlA) ? "active":""; gId('buttonSync').className = (syncSend) ? "active":""; showNodes(); updateSelectedPalette(); updateSelectedFx(); updateTrail(gId('sliderBri')); updateTrail(gId('sliderSpeed')); updateTrail(gId('sliderIntensity')); updateTrail(gId('sliderC1')); updateTrail(gId('sliderC2')); updateTrail(gId('sliderC3')); gId('wwrap').style.display = (isRgbw) ? "block":"none"; gId("wbal").style.display = (cct) ? "block":"none"; gId('kwrap').style.display = (cct) ? "none":"block"; updatePA(); updatePSliders(); } function updateSelectedPalette() { var parent = gId('pallist'); var selPaletteInput = parent.querySelector(`input[name="palette"][value="${selectedPal}"]`); if (selPaletteInput) selPaletteInput.checked = true; var selElement = parent.querySelector('.selected'); if (selElement) selElement.classList.remove('selected'); var selectedPalette = parent.querySelector(`.lstI[data-id="${selectedPal}"]`); if (selectedPalette) parent.querySelector(`.lstI[data-id="${selectedPal}"]`).classList.add('selected'); } function updateSelectedFx() { var parent = gId('fxlist'); var selEffectInput = parent.querySelector(`input[name="fx"][value="${selectedFx}"]`); if (selEffectInput) selEffectInput.checked = true; var selElement = parent.querySelector('.selected'); if (selElement) selElement.classList.remove('selected'); var selectedEffect = parent.querySelector(`.lstI[data-id="${selectedFx}"]`); if (selectedEffect) { selectedEffect.classList.add('selected'); var fx = (selectedFx != prevFx) && currentPreset==-1; //effect changed & preset==none var ps = (prevPS != currentPreset) && currentPreset==-1; // preset changed & preset==none // WLEDSR: extract the Slider and color control string from the HTML element and set it. setSliderAndColorControl(selectedFx, (fx || ps)); } } function displayRover(i,s) { gId('rover').style.transform = (i.live && s.lor == 0) ? "translateY(0px)":"translateY(100%)"; var sour = i.lip ? i.lip:""; if (sour.length > 2) sour = " from " + sour; gId('lv').innerHTML = `WLED is receiving live ${i.lm} data${sour}`; gId('roverstar').style.display = (i.live && s.lor) ? "block":"none"; } function cmpP(a, b) { if (!a[1].n) return (a[0] > b[0]); // sort playlists first, followed by presets with characters and last presets with special 1st character const c = a[1].n.charCodeAt(0); const d = b[1].n.charCodeAt(0); if ((c>47 && c<58) || (c>64 && c<91) || (c>96 && c<123) || c>255) x = '='; else x = '>'; if ((d>47 && d<58) || (d>64 && d<91) || (d>96 && d<123) || d>255) y = '='; else y = '>'; const n = (a[1].playlist ? '<' : x) + a[1].n; return n.localeCompare((b[1].playlist ? '<' : y) + b[1].n, undefined, {numeric: true}); } function makeWS() { if (ws) { ws.close(); ws=null; } if (lastinfo.ws < 0) return; ws = new WebSocket('ws://'+(loc?locip:window.location.hostname)+'/ws'); ws.binaryType = "arraybuffer"; ws.onmessage = (e)=>{ if (e.data instanceof ArrayBuffer) return; //liveview packet var json = JSON.parse(e.data); if (json.leds) return; //JSON liveview packet clearTimeout(jsonTimeout); jsonTimeout = null; lastUpdate = new Date(); clearErrorToast(); gId('connind').style.backgroundColor = "var(--c-l)"; // json object should contain json.info AND json.state (but may not) var i = json.info; if (i) { lastinfo = i; parseInfo(); showNodes(); if (isInfo) populateInfo(i); } else i = lastinfo; var s = json.state ? json.state : json; displayRover(i, s); readState(s); }; ws.onclose = (e)=>{ gId('connind').style.backgroundColor = "var(--c-r)"; setTimeout(makeWS,500); //retry WS connection } ws.onopen = (e)=>{ //ws.send("{'v':true}"); //unnecessary (https://github.com/Aircoookie/WLED/blob/master/wled00/ws.cpp#L18) reqsLegal = true; } } function readState(s,command=false) { if (!s) return false; isOn = s.on; gId('sliderBri').value= s.bri; nlA = s.nl.on; nlDur = s.nl.dur; nlTar = s.nl.tbri; nlFade = s.nl.fade; syncSend = s.udpn.send; prevPS = currentPreset; if (s.pl<0) currentPreset = s.ps; else currentPreset = s.pl; tr = s.transition; gId('tt').value = tr/10; var selc=0; var ind=0; populateSegments(s); for (let i = 0; i < (s.seg||[]).length; i++) { if(s.seg[i].sel) {selc = ind; break;} ind++; } var i=s.seg[selc]; if (!i) { showToast('No Segments!', true); updateUI(); return; } selColors = i.col; var cd = gId('csl').children; for (let e = cd.length-1; e >= 0; e--) { var r,g,b,w; r = i.col[e][0]; g = i.col[e][1]; b = i.col[e][2]; if (isRgbw) w = i.col[e][3]; cd[e].style.backgroundColor = "rgb(" + r + "," + g + "," + b + ")"; if (isRgbw) whites[e] = parseInt(w); } selectSlot(csel); if (i.cct && i.cct>=0) gId("sliderA").value = i.cct; gId('sliderSpeed').value = i.sx; gId('sliderIntensity').value = i.ix; gId('sliderC1').value = i.f1x ? i.f1x : 0; gId('sliderC2').value = i.f2x ? i.f2x : 0; gId('sliderC3').value = i.f3x ? i.f3x : 0; if (s.error && s.error != 0) { var errstr = ""; switch (s.error) { case 10: errstr = "Could not mount filesystem!"; break; case 11: errstr = "Not enough space to save preset!"; break; case 12: errstr = "Preset not found."; break; case 13: errstr = "Missing ir.json."; break; case 19: errstr = "A filesystem error has occured."; break; } showToast('Error ' + s.error + ": " + errstr, true); } prevFx = selectedFx; selectedPal = i.pal; selectedFx = i.fx; redrawPalPrev(); // if any color changed (random palette did at least) updateUI(); } // WLEDSR: control HTML elements for Slider and Color Control // Technical notes // =============== // If an effect name is followed by an @, slider and color control is effective. // If not effective then: // - For AC effects (id<128) 2 sliders and 3 colors and the palette will be shown // - For SR effects (id>128) 5 sliders and 3 colors and the palette will be shown // If effective (@) // - a ; seperates slider controls (left) from color controls (middle) and palette control (right) // - if left, middle or right is empty no controls are shown // - a , seperates slider controls (max 5) or color controls (max 3). Palette has only one value // - a ! means that the default is used. // - For sliders: Effect speeds, Effect intensity, Custom 1, Custom 2, Custom 3 // - For colors: Fx color, Background color, Custom // - For palette: prompt for color palette OR palette ID if numeric (will hide palette selection) // // Note: If palette is on and no colors are specified 1,2 and 3 is shown in each color circle. // If a color is specified, the 1,2 or 3 is replaced by that specification. // Note: Effects can override default pattern behaviour // - FadeToBlack can override the background setting // - Defining SEGCOL() can override a specific palette using these values (e.g. Color Gradient) function setSliderAndColorControl(idx, applyDef=false) { if (!(Array.isArray(fxdata) && fxdata.length>idx)) return; var topPosition = 0; var controlDefined = (fxdata[idx].substr(0,1) == "@")?true:false; var extra = fxdata[idx].substr(1); var extras = (extra == '')?[]:extra.split(";"); var slOnOff = (extras.length==0 || extras[0]=='')?[]:extras[0].split(","); var coOnOff = (extras.length<2 || extras[1]=='')?[]:extras[1].split(","); var paOnOff = (extras.length<3 || extras[2]=='')?[]:extras[2].split(","); var obj = {"seg":{}}; // set html slider items on/off var nSliders = Math.min(5,Math.floor((gId("Effects").children.length - 1) / 2)); // p (label) & div for each slider + FX list for (let i=0; ii && slOnOff[i] != "")) { label.style.display = "block"; if (slOnOff.length>i && slOnOff[i].indexOf("=")>0) { //embeded default values var dPos = slOnOff[i].indexOf("="); var v = Math.max(0,Math.min(255,parseInt(slOnOff[i].substr(dPos+1)))); if (i==0) { if (applyDef) gId("sliderSpeed").value = v; obj.seg.sx = v; } else if (i==1) { if (applyDef) gId("sliderIntensity").value = v; obj.seg.ix = v; } else { if (applyDef) gId("sliderC"+(i-1)).value = v; obj.seg["C"+(i-1)] = v} slOnOff[i] = slOnOff[i].substring(0,dPos); } if (slOnOff.length>i && slOnOff[i]!="!") label.innerHTML = slOnOff[i]; else if (i==0) label.innerHTML = "Effect speed"; else if (i==1) label.innerHTML = "Effect intensity"; else label.innerHTML = "Custom" + (i-1); label.style.top = "auto"; slider.style.display = "block"; slider.style.top = topPosition + "px"; topPosition += 28; // increase top position for the next control slider.setAttribute('title',label.innerHTML); } else { // disable label and slider slider.style.display = "none"; label.style.display = "none"; } } if (topPosition>0) topPosition += 2; // set top position of the effect list gId("fxFind").style.top = topPosition + "px"; topPosition += 42; var fxList = gId("fxlist"); for (var i=0; ii && coOnOff[i] != "") { btn.style.display = "inline"; if (coOnOff.length>i && coOnOff[i] != "!") { var abbreviation = coOnOff[i].substr(0,2); btn.innerHTML = abbreviation; if (abbreviation != coOnOff[i]) { cslLabel += sep + abbreviation + '=' + coOnOff[i]; sep = ', '; } } else if (i==0) btn.innerHTML = "Fx"; else if (i==1) btn.innerHTML = "Bg"; else btn.innerHTML = "Cs"; hide = false; } else if (!controlDefined /*|| paOnOff.length>0*/) { // if no controls then all buttons should be shown for color 1..3 btn.style.display = "inline"; btn.innerHTML = `${i+1}`; hide = false; } else { btn.style.display = "none"; if (i>0 && csel==i) selectSlot(0); } } gId("cslLabel").innerHTML = cslLabel; // set palette on/off var palw = gId("palw"); // wrapper var pall = gId("pall"); // list // if not controlDefined or palette has a value if ((!controlDefined) || (paOnOff.length>0 && paOnOff[0]!="" && isNaN(paOnOff[0]))) { palw.style.display = "inline-block"; if (paOnOff.length>0 && paOnOff[0].indexOf("=")>0) { //embeded default values var dPos = paOnOff[0].indexOf("="); var v = Math.max(0,Math.min(255,parseInt(paOnOff[0].substr(dPos+1)))); var p = d.querySelector(`#pallist input[name="palette"][value="${v}"]`); if (applyDef && p) { p.checked = true; obj.seg.pal = v; } paOnOff[0] = paOnOff[0].substring(0,dPos); } if (paOnOff.length>0 && paOnOff[0] != "!") pall.innerHTML = paOnOff[0]; else pall.innerHTML = ' Color palette'; } else { // disable label and slider palw.style.display = "none"; // if numeric set as selected palette if (paOnOff.length>0 && paOnOff[0]!="" && !isNaN(paOnOff[0]) && parseInt(paOnOff[0])!=selectedPal) obj.seg.pal = parseInt(paOnOff[0]); } if (!isEmpty(obj.seg) && applyDef) requestJson(obj); //update default values (may need throttling on ESP8266) } var jsonTimeout; var reqsLegal = false; function requestJson(command=null) { gId('connind').style.backgroundColor = "var(--c-r)"; if (command && !reqsLegal) return; //stop post requests from chrome onchange event on page restore if (!jsonTimeout) jsonTimeout = setTimeout(showErrorToast, 3000); var req = null; var url = (loc?`http://${locip}`:'') + '/json/si'; var useWs = (ws && ws.readyState === WebSocket.OPEN); var type = command ? 'post':'get'; if (command) { if (useWs || !command.ps) command.v = true; // force complete /json/si API response (ps is async so no point) command.time = Math.floor(Date.now() / 1000); var t = gId('tt'); if (t.validity.valid && command.transition==null) { var tn = parseInt(t.value*10); if (tn != tr) command.transition = tn; } req = JSON.stringify(command); if (req.length > 1000) useWs = false; //do not send very long requests over websocket }; if (useWs) { ws.send(req?req:'{"v":true}'); return; } else if (command && command.ps) { //refresh UI if we don't use WS (async loading of presets) setTimeout(requestJson,200); } fetch(url, { method: type, headers: { "Content-type": "application/json; charset=UTF-8" }, body: req }) .then(res => { if (!res.ok) showErrorToast(); return res.json(); }) .then(json => { clearTimeout(jsonTimeout); jsonTimeout = null; lastUpdate = new Date(); clearErrorToast(); gId('connind').style.backgroundColor = "var(--c-g)"; if (!json) { showToast('Empty response', true); return; } if (json.success) return; if (json.info) { lastinfo = json.info; parseInfo(); if (isInfo) populateInfo(lastinfo); } var s = json.state ? json.state : json; readState(s); makeWS(); reqsLegal = true; }) .catch((e)=>{ showToast(e, true); console.log(e); }); } function togglePower() { isOn = !isOn; var obj = {"on": isOn}; requestJson(obj); } function toggleNl() { nlA = !nlA; if (nlA) { showToast(`Timer active. Your light will turn ${nlTar > 0 ? "on":"off"} ${nlMode ? "over":"after"} ${nlDur} minutes.`); } else { showToast('Timer deactivated.'); } var obj = {"nl": {"on": nlA}}; requestJson(obj); } function toggleSync() { syncSend = !syncSend; if (syncSend) showToast('Other lights in the network will now sync to this one.'); else showToast('This light and other lights in the network will no longer sync.'); var obj = {"udpn": {"send": syncSend}}; if (syncTglRecv) obj.udpn.recv = syncSend; requestJson(obj); } function toggleLiveview() { isLv = !isLv; gId('liveview').style.display = (isLv) ? "block":"none"; var url = (loc?`http://${locip}`:'') + "/liveview"; gId('liveview').src = (isLv) ? url:"about:blank"; gId('buttonSr').className = (isLv) ? "active":""; if (!isLv && ws && ws.readyState === WebSocket.OPEN) ws.send('{"lv":false}'); size(); } function toggleInfo() { if (isNodes) toggleNodes(); isInfo = !isInfo; if (isInfo) requestJson(); // loadInfo(); gId('info').style.transform = (isInfo) ? "translateY(0px)":"translateY(100%)"; gId('buttonI').className = (isInfo) ? "active":""; } function toggleNodes() { if (isInfo) toggleInfo(); isNodes = !isNodes; if (isNodes) loadNodes(); gId('nodes').style.transform = (isNodes) ? "translateY(0px)":"translateY(100%)"; gId('buttonNodes').className = (isNodes) ? "active":""; } function makeSeg() { var ns = 0; var lu = lowestUnused; if (lu > 0) { var pend = parseInt(gId(`seg${lu -1}e`).value,10) + (cfg.comp.seglen?parseInt(gId(`seg${lu -1}s`).value,10):0); if (pend < ledCount) ns = pend; } gId('segutil').scrollIntoView({ behavior: 'smooth', block: 'start', }); var ct = ledCount-(cfg.comp.seglen?ns:0); var cn = `
Start LED ${cfg.comp.seglen?"LED count":"Stop LED"}
${ledCount - ns} LEDs
`; gId('segutil').innerHTML = cn; } function resetUtil() { gId('segutil').innerHTML = ''; //for (var i=0; i${n}` } return plSelContent; } function refreshPlE(p) { var plEDiv = gId(`ple${p}`); if (!plEDiv) return; var content = "
Playlist entries
"; for (var i = 0; i < plJson[p].ps.length; i++) { content += makePlEntry(p,i); } content += `
`; plEDiv.innerHTML = content; var dels = plEDiv.getElementsByClassName("btn-pl-del"); if (dels.length < 2) dels[0].style.display = "none"; var sels = gId(`seg${p+100}`).getElementsByClassName("sel"); for (var i of sels) { if (i.dataset.val) { if (parseInt(i.dataset.val) > 0) i.value = i.dataset.val; else plJson[p].ps[i.dataset.index] = parseInt(i.value); } } } //p: preset ID, i: ps index function addPl(p,i) { plJson[p].ps.splice(i+1,0,0); plJson[p].dur.splice(i+1,0,plJson[p].dur[i]); plJson[p].transition.splice(i+1,0,plJson[p].transition[i]); refreshPlE(p); } function delPl(p,i) { if (plJson[p].ps.length < 2) return; plJson[p].ps.splice(i,1); plJson[p].dur.splice(i,1); plJson[p].transition.splice(i,1); refreshPlE(p); } function plePs(p,i,field) { plJson[p].ps[i] = parseInt(field.value); } function pleDur(p,i,field) { if (field.validity.valid) plJson[p].dur[i] = Math.floor(field.value*10); } function pleTr(p,i,field) { if (field.validity.valid) plJson[p].transition[i] = Math.floor(field.value*10); } function plR(p) { var pl = plJson[p]; pl.r = gId(`pl${p}rtgl`).checked; if (gId(`pl${p}rptgl`).checked) { //infinite pl.repeat = 0; delete pl.end; gId(`pl${p}o1`).style.display = "none"; } else { pl.repeat = parseInt(gId(`pl${p}rp`).value); pl.end = parseInt(gId(`pl${p}selEnd`).value); gId(`pl${p}o1`).style.display = "block"; } } function makeP(i,pl) { var content = ""; if (pl) { var rep = plJson[i].repeat ? plJson[i].repeat : 0; content = `
Repeat 0?rep:1}> times
End preset:
`; } else { content = ` `; if (Array.isArray(lastinfo.maps) && lastinfo.maps.length>0) { content += `
Ledmap: 
"; } } return `
Quick load label:
(leave empty for no Quick load button)
API command
${content}
Save to ID 0)?i:getLowestUnusedP()}>
${(i>0)?'
${(i>0)? ('
ID ' +i+ '
'):""}`; } function makePUtil() { gId('putil').classList.remove("staytop"); gId('putil').scrollIntoView({ behavior: 'smooth', block: 'start', }); gId('putil').innerHTML = `
${makeP(0)}
`; //for (var i=0; i
Duration Transition #${i+1}
s s
`; } function makePlUtil() { if (pNum < 2) { showToast("You need at least 2 presets to make a playlist!"); //return; } if (plJson[0].transition[0] < 0) plJson[0].transition[0] = tr; gId('putil').classList.remove("staytop"); gId('putil').scrollIntoView({ behavior: 'smooth', block: 'start', }); gId('putil').innerHTML = `
${makeP(0,true)}
`; refreshPlE(0); } function resetPUtil() { gId('putil').classList.add("staytop"); var cn = ``+ ``; gId('putil').innerHTML = cn; } function tglCs(i) { var pss = gId(`p${i}cstgl`).checked; gId(`p${i}o1`).style.display = pss? "block" : "none"; gId(`p${i}o2`).style.display = !pss? "block" : "none"; } function tglSegn(s) { d.gId(`seg${s}t`).style.display = (window.getComputedStyle(d.gId(`seg${s}t`)).display === "none") ? "inline":"none"; } function selSegEx(s) { var obj = {"seg":[]}; for (let i=0; i<=lSeg; i++) obj.seg.push({"sel":(i==s)?true:false}); requestJson(obj); } function selSeg(s) { var sel = gId(`seg${s}sel`).checked; var obj = {"seg": {"id": s, "sel": sel}}; requestJson(obj); } function rptSeg(s) { var name = gId(`seg${s}t`).value; var start = parseInt(gId(`seg${s}s`).value); var stop = parseInt(gId(`seg${s}e`).value); if (stop == 0) {return;} var rev = gId(`seg${s}rev`).checked; var mi = gId(`seg${s}mi`).checked; var sel = gId(`seg${s}sel`).checked; var obj = {"seg": {"id": s, "n": name, "start": start, "stop": (cfg.comp.seglen?start:0)+stop, "rev": rev, "mi": mi, "on": !powered[s], "bri": parseInt(gId(`seg${s}bri`).value), "sel": sel}}; if (gId(`seg${s}grp`)) { var grp = parseInt(gId(`seg${s}grp`).value); var spc = parseInt(gId(`seg${s}spc`).value); var ofs = parseInt(gId(`seg${s}of` ).value); obj.seg.grp = grp; obj.seg.spc = spc; obj.seg.of = ofs; } obj.seg.rpt = true; expand(s); requestJson(obj); } function setSeg(s) { var name = gId(`seg${s}t`).value; var start = parseInt(gId(`seg${s}s`).value); var stop = parseInt(gId(`seg${s}e`).value); if ((cfg.comp.seglen && stop == 0) || (!cfg.comp.seglen && stop <= start)) {delSeg(s); return;} var obj = {"seg": {"id": s, "n": name, "start": start, "stop": (cfg.comp.seglen?start:0)+stop}}; if (gId(`seg${s}grp`)) { var grp = parseInt(gId(`seg${s}grp`).value); var spc = parseInt(gId(`seg${s}spc`).value); var ofs = parseInt(gId(`seg${s}of` ).value); obj.seg.grp = grp; obj.seg.spc = spc; obj.seg.of = ofs; } requestJson(obj); } function delSeg(s) { if (segCount < 2) { showToast("You need to have multiple segments to delete one!"); return; } expanded[s] = false; segCount--; var obj = {"seg": {"id": s, "stop": 0}}; requestJson(obj); } function setRev(s) { var rev = gId(`seg${s}rev`).checked; var obj = {"seg": {"id": s, "rev": rev}}; requestJson(obj); } function setMi(s) { var mi = gId(`seg${s}mi`).checked; var obj = {"seg": {"id": s, "mi": mi}}; requestJson(obj); } function setSegPwr(s) { var obj = {"seg": {"id": s, "on": !powered[s]}}; requestJson(obj); } function setSegBri(s) { var obj = {"seg": {"id": s, "bri": parseInt(gId(`seg${s}bri`).value)}}; requestJson(obj); } function tglFreeze(s=null) { var obj = {"seg": {"frz": "t"}}; // toggle if (s!==null) obj.id = s; requestJson(obj); } function setX(ind = null) { if (ind === null) { ind = parseInt(d.querySelector('#fxlist input[name="fx"]:checked').value); } else { d.querySelector(`#fxlist input[name="fx"][value="${ind}"]`).checked = true; } var selElement = d.querySelector('#fxlist .selected'); if (selElement) selElement.classList.remove('selected'); d.querySelector(`#fxlist .lstI[data-id="${ind}"]`).classList.add('selected'); var obj = {"seg": {"fx": parseInt(ind)}}; requestJson(obj); } function setPalette(paletteId = null) { if (paletteId === null) { paletteId = parseInt(d.querySelector('#pallist input[name="palette"]:checked').value); } else { d.querySelector(`#pallist input[name="palette"][value="${paletteId}"]`).checked = true; } var selElement = d.querySelector('#pallist .selected'); if (selElement) { selElement.classList.remove('selected') } d.querySelector(`#pallist .lstI[data-id="${paletteId}"]`).classList.add('selected'); var obj = {"seg": {"pal": paletteId}}; requestJson(obj); } function setBri() { var obj = {"bri": parseInt(gId('sliderBri').value)}; requestJson(obj); } function setSpeed() { var obj = {"seg": {"sx": parseInt(gId('sliderSpeed').value)}}; requestJson(obj); } function setIntensity() { var obj = {"seg": {"ix": parseInt(gId('sliderIntensity').value)}}; requestJson(obj); } function setLor(i) { var obj = {"lor": i}; requestJson(obj); } function setPreset(i) { var obj = {"ps": i}; if (isPlaylist(i)) obj.on = true; //force on showToast("Loading preset " + pName(i) +" (" + i + ")"); requestJson(obj); } function saveP(i,pl) { pI = parseInt(gId(`p${i}id`).value); if (!pI || pI < 1) pI = (i>0) ? i : getLowestUnusedP(); pN = gId(`p${i}txt`).value; if (pN == "") pN = (pl?"Playlist ":"Preset ") + pI; var obj = {}; if (!gId(`p${i}cstgl`).checked) { var raw = gId(`p${i}api`).value; try { obj = JSON.parse(raw); } catch (e) { obj.win = raw; if (raw.length < 2) { gId(`p${i}warn`).innerHTML = "⚠ Please enter your API command first"; return; } else if (raw.indexOf('{') > -1) { gId(`p${i}warn`).innerHTML = "⚠ Syntax error in custom JSON API command"; return; } else if (raw.indexOf("Please") == 0) { gId(`p${i}warn`).innerHTML = "⚠ Please refresh the page before modifying this preset"; return; } } obj.o = true; } else { if (pl) { obj.playlist = plJson[i]; obj.on = true; obj.o = true; } else { obj.ib = gId(`p${i}ibtgl`).checked; obj.sb = gId(`p${i}sbtgl`).checked; obj.sc = gId(`p${i}sbchk`).checked; if (gId(`p${i}lmp`).value!=="") obj.ledmap = parseInt(gId(`p${i}lmp`).value); } } obj.psave = pI; obj.n = pN; var pQN = gId(`p${i}ql`).value; if (pQN.length > 0) obj.ql = pQN; showToast("Saving " + pN +" (" + pI + ")"); requestJson(obj); if (obj.o) { pJson[pI] = obj; delete pJson[pI].psave; delete pJson[pI].o; delete pJson[pI].v; delete pJson[pI].time; } else { pJson[pI] = {"n":pN, "win":"Please refresh the page to see this newly saved command."}; if (obj.win) pJson[pI].win = obj.win; if (obj.ql) pJson[pI].ql = obj.ql; } populatePresets(); resetPUtil(); if (i>0) expand(pI+100); // collapse edited preset or expand created preset. } function testPl(i,bt) { if (bt.dataset.test == 1) { bt.dataset.test = 0; bt.innerHTML = "Test"; stopPl(); return; } bt.dataset.test = 1; bt.innerHTML = "Stop"; var obj = {}; obj.playlist = plJson[i]; obj.on = true; requestJson(obj); } function stopPl() { requestJson({playlist:{}}) } function delP(i) { var bt = gId(`p${i}del`); if (bt.dataset.cnf == 1) { var obj = {"pdel": i}; requestJson(obj); delete pJson[i]; populatePresets(); } else { bt.style.color = "var(--c-r)"; bt.innerHTML = "Delete!"; bt.dataset.cnf = 1; } } function selectSlot(b) { csel = b; var cd = gId('csl').children; for (let i = 0; i < cd.length; i++) cd[i].classList.remove('xxs-w'); cd[b].classList.add('xxs-w'); setPicker(cd[b].style.backgroundColor); gId('sliderW').value = whites[b]; updatePSliders(); } //set the color from a hex string. Used by quick color selectors var lasth = 0; function pC(col) { if (col == "rnd") { col = {h: 0, s: 0, v: 100}; col.s = Math.floor((Math.random() * 50) + 50); do { col.h = Math.floor(Math.random() * 360); } while (Math.abs(col.h - lasth) < 50); lasth = col.h; } setPicker(col); setColor(0); } function updatePSliders() { //update RGB sliders var col = cpick.color.rgb; gId('sliderR').value = col.r; gId('sliderG').value = col.g; gId('sliderB').value = col.b; //update hex field var str = cpick.color.hexString.substring(1); var w = whites[csel]; if (w > 0) str += w.toString(16); gId('hexc').value = str; gId('hexcnf').style.backgroundColor = "var(--c-3)"; //update value slider var v = gId('sliderV'); v.value = cpick.color.value; //background color as if color had full value var hsv = {"h":cpick.color.hue,"s":cpick.color.saturation,"v":100}; var c = iro.Color.hsvToRgb(hsv); var cs = 'rgb('+c.r+','+c.g+','+c.b+')'; v.nextElementSibling.style.backgroundImage = `linear-gradient(90deg, #000 0%, ${cs})`; //v.parentNode.getElementsByClassName('sliderdisplay')[0].style.setProperty('--bg',cs); //updateTrail(v); //update Kelvin slider gId('sliderK').value = cpick.color.kelvin; } function hexEnter() { if(event.keyCode == 13) fromHex(); } function fromHex() { var str = gId('hexc').value; whites[csel] = parseInt(str.substring(6), 16); try { setPicker("#" + str.substring(0,6)); } catch (e) { setPicker("#ffaa00"); } if (isNaN(whites[csel])) whites[csel] = 0; setColor(2); } function setPicker(rgb) { var c = new iro.Color(rgb); if (c.value > 0) cpick.color.set(c); else cpick.color.setChannel('hsv', 'v', 0); } function fromV() { cpick.color.setChannel('hsv', 'v', d.getElementById('sliderV').value); } function fromK() { cpick.color.set({ kelvin: d.getElementById('sliderK').value }); } function fromRgb() { var r = gId('sliderR').value; var g = gId('sliderG').value; var b = gId('sliderB').value; setPicker(`rgb(${r},${g},${b})`); } //sr 0: from RGB sliders, 1: from picker, 2: from hex function setColor(sr) { var cd = gId('csl').children; // color slots if (sr == 1 && cd[csel].style.backgroundColor == "rgb(0, 0, 0)") cpick.color.setChannel('hsv', 'v', 100); // watch out for spaces!!! cd[csel].style.backgroundColor = cpick.color.rgbString; if (sr != 2) whites[csel] = parseInt(gId('sliderW').value); var col = cpick.color.rgb; var obj = {"seg": {"col": [[col.r, col.g, col.b, whites[csel]],[],[]]}}; if (csel == 1) { obj = {"seg": {"col": [[],[col.r, col.g, col.b, whites[csel]],[]]}}; } else if (csel == 2) { obj = {"seg": {"col": [[],[],[col.r, col.g, col.b, whites[csel]]]}}; } requestJson(obj); } function setBalance(b) { var obj = {"seg": {"cct": parseInt(b)}}; requestJson(obj); } var hc = 0; setInterval(()=>{ if (!isInfo) return; hc+=18; if (hc>300) hc=0; if (hc>200)hc=306; if (hc==144) hc+=36; if (hc==108) hc+=18; gId('heart').style.color = `hsl(${hc}, 100%, 50%)`; }, 910); function openGH() { window.open("https://github.com/Aircoookie/WLED/wiki"); } var cnfr = false; function cnfReset() { if (!cnfr) { var bt = gId('resetbtn'); bt.style.color = "var(--c-r)"; bt.innerHTML = "Confirm Reboot"; cnfr = true; return; } window.location.href = "/reset"; } var cnfrS = false; function rSegs() { var bt = gId('rsbtn'); if (!cnfrS) { bt.style.color = "var(--c-r)"; bt.innerHTML = "Confirm reset"; cnfrS = true; return; } cnfrS = false; bt.style.color = "var(--c-f)"; bt.innerHTML = "Reset segments"; var obj = {"seg":[{"start":0,"stop":ledCount,"sel":true}]}; for (let i=1; i<=lSeg; i++) obj.seg.push({"stop":0}); requestJson(obj); } function loadPalettesData(callback = null) { if (palettesData) return; const lsKey = "wledPalx"; var lsPalData = localStorage.getItem(lsKey); if (lsPalData) { try { var d = JSON.parse(lsPalData); if (d && d.vid == d.vid) { palettesData = d.p; if (callback) callback(); // redrawPalPrev() return; } } catch (e) {} } palettesData = {}; getPalettesData(0, ()=>{ localStorage.setItem(lsKey, JSON.stringify({ p: palettesData, vid: lastinfo.vid })); if (callback) setTimeout(callback, 99); //redrawPalPrev() }); } function getPalettesData(page, callback) { var url = (loc?`http://${locip}`:'') + `/json/palx?page=${page}`; fetch(url, { method: 'get', headers: { "Content-type": "application/json; charset=UTF-8" } }) .then(res => { if (!res.ok) showErrorToast(); return res.json(); }) .then(json => { palettesData = Object.assign({}, palettesData, json.p); if (page < json.m) setTimeout(()=>{ getPalettesData(page + 1, callback); }, 50); else callback(); }) .catch((error)=>{ showToast(error, true); console.log(error); }); } function search(f,l=null) { f.nextElementSibling.style.display=(f.value!=='')?'block':'none'; if (!l) return; var el = gId(l).querySelectorAll('.lstI'); for (i = 0; i < el.length; i++) { var it = el[i]; var itT = it.querySelector('.lstIname').innerText.toUpperCase(); it.style.display = itT.indexOf(f.value.toUpperCase())>-1?'':'none'; } } function clean(c) { c.style.display='none'; var i=c.previousElementSibling; i.value=''; i.focus(); i.dispatchEvent(new Event('input')); } //make sure "dur" and "transition" are arrays with at least the length of "ps" function formatArr(pl) { var l = pl.ps.length; if (!Array.isArray(pl.dur)) { var v = pl.dur; if (isNaN(v)) v = 100; pl.dur = [v]; } var l2 = pl.dur.length; if (l2 < l) { for (var i = 0; i < l - l2; i++) pl.dur.push(pl.dur[l2-1]); } if (!Array.isArray(pl.transition)) { var v = pl.transition; if (isNaN(v)) v = tr; pl.transition = [v]; } var l2 = pl.transition.length; if (l2 < l) { for (var i = 0; i < l - l2; i++) pl.transition.push(pl.transition[l2-1]); } } function expand(i/*,a=false*/) { var seg = gId('seg' +i); let util = i<100?'segutil':'putil'; let stay = i<100?"staybot":"staytop"; /* if (!a) { var j = i>100 ? 100 : 0; var l = i>100 ? expanded.length : 100; for (; j= 100) { var p = i-100; gId(`p${p}o`).classList.toggle('expand'); if (expanded[i]) { if (isPlaylist(p)) { plJson[p] = pJson[p].playlist; //make sure all keys are present in plJson[p] formatArr(plJson[p]); if (isNaN(plJson[p].repeat)) plJson[p].repeat = 0; if (!plJson[p].r) plJson[p].r = false; if (isNaN(plJson[p].end)) plJson[p].end = 0; seg.innerHTML = makeP(p,true); refreshPlE(p); } else { seg.innerHTML = makeP(p); } var papi = papiVal(p); gId(`p${p}api`).value = papi; if (papi.indexOf("Please") == 0) gId(`p${p}cstgl`).checked = false; tglCs(p); } else seg.innerHTML = ""; } seg.parentElement.scrollIntoView({ behavior: 'smooth', block: (expanded[i]?'start':'center'), }); } function unfocusSliders() { gId("sliderBri").blur(); gId("sliderSpeed").blur(); gId("sliderIntensity").blur(); } //sliding UI const _C = d.querySelector('.container'), N = 4; let iSlide = 0, x0 = null, scrollS = 0, locked = false, w; function unify(e) { return e.changedTouches ? e.changedTouches[0] : e; } function hasIroClass(classList) { for (var i = 0; i < classList.length; i++) { var element = classList[i]; if (element.startsWith('Iro')) return true; } return false; } function lock(e) { if (pcMode) return; var l = e.target.classList; var pl = e.target.parentElement.classList; if (l.contains('noslide') || hasIroClass(l) || hasIroClass(pl)) return; x0 = unify(e).clientX; scrollS = gEBCN("tabcontent")[iSlide].scrollTop; _C.classList.toggle('smooth', !(locked = true)); } function move(e) { if(!locked || pcMode) return; var clientX = unify(e).clientX; var dx = clientX - x0; var s = Math.sign(dx); var f = +(s*dx/w).toFixed(2); if((clientX != 0) && (iSlide > 0 || s < 0) && (iSlide < N - 1 || s > 0) && f > 0.12 && gEBCN("tabcontent")[iSlide].scrollTop == scrollS) { _C.style.setProperty('--i', iSlide -= s); f = 1 - f; updateTablinks(iSlide); } _C.style.setProperty('--f', f); _C.classList.toggle('smooth', !(locked = false)); x0 = null; } function showNodes() { gId('buttonNodes').style.display = (lastinfo.ndc > 0 && (wW > 797 || (wW > 539 && wW < 720))) ? "block":"none"; } function size() { wW = window.innerWidth; showNodes(); var h = gId('top').clientHeight; sCol('--th', h + "px"); sCol('--bh', gId('bot').clientHeight + "px"); if (isLv) h -= 4; sCol('--tp', h + "px"); togglePcMode(); } function togglePcMode(fromB = false) { if (fromB) { pcModeA = !pcModeA; localStorage.setItem('pcm', pcModeA); pcMode = pcModeA; } if (wW < 1250 && !pcMode) return; if (!fromB && ((wW < 1250 && lastw < 1250) || (wW >= 1250 && lastw >= 1250))) return; openTab(0, true); if (wW < 1250) {pcMode = false;} else if (pcModeA && !fromB) pcMode = pcModeA; updateTablinks(0); gId('buttonPcm').className = (pcMode) ? "active":""; gId('bot').style.height = (pcMode && !cfg.comp.pcmbot) ? "0":"auto"; sCol('--bh', gId('bot').clientHeight + "px"); _C.style.width = (pcMode)?'100%':'400%'; lastw = wW; } function mergeDeep(target, ...sources) { if (!sources.length) return target; const source = sources.shift(); if (isObj(target) && isObj(source)) { for (const key in source) { if (isObj(source[key])) { if (!target[key]) Object.assign(target, { [key]: {} }); mergeDeep(target[key], source[key]); } else { Object.assign(target, { [key]: source[key] }); } } } return mergeDeep(target, ...sources); } size(); _C.style.setProperty('--n', N); window.addEventListener('resize', size, false); _C.addEventListener('mousedown', lock, false); _C.addEventListener('touchstart', lock, false); _C.addEventListener('mouseout', move, false); _C.addEventListener('mouseup', move, false); _C.addEventListener('touchend', move, false);