//page js var loc = false, locip, locproto = "http:"; var isOn = false, nlA = false, isLv = false, isInfo = false, isNodes = false, syncSend = false/*, syncTglRecv = true*/; var hasWhite = false, hasRGB = false, hasCCT = false; var nlDur = 60, nlTar = 0; var nlMode = false; var segLmax = 0; // size (in pixels) of largest selected segment var selectedFx = 0; var selectedPal = 0; var csel = 0; // selected color slot (0-2) var currentPreset = -1; var lastUpdate = 0; var segCount = 0, ledCount = 0, lowestUnused = 0, maxSeg = 0, lSeg = 0; var pcMode = false, pcModeA = false, lastw = 0, wW; var simplifiedUI = false; var tr = 7; var d = document; const ranges = RangeTouch.setup('input[type="range"]', {}); var retry = false; var palettesData; var fxdata = []; var pJson = {}, eJson = {}, lJson = {}; var plJson = {}; // array of playlists var pN = "", pI = 0, pNum = 0; var pmt = 1, pmtLS = 0, pmtLast = 0; var lastinfo = {}; var isM = false, mw = 0, mh=0; var ws, wsRpt=0; var cfg = { theme:{base:"dark", bg:{url:"", rnd: false, rndGrayscale: false, rndBlur: false}, 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, segpwr:false, segexp:false, css:true, hdays:false, fxdef:true, on:0, off:0, idsort: false} }; // [year, month (0 -> January, 11 -> December), day, duration in days, image url] 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 [2025, 3, 20, 2, "https://aircoookie.github.io/easter.png"], // easter 2025 [2024, 2, 31, 2, "https://aircoookie.github.io/easter.png"], // easter 2024 [0, 6, 4, 1, "https://images.alphacoders.com/516/516792.jpg"], // 4th of July [0, 0, 1, 1, "https://images.alphacoders.com/119/1198800.jpg"] // new year ]; var cpick = new iro.ColorPicker("#picker", { width: 260, wheelLightness: false, wheelAngle: 270, wheelDirection: "clockwise", layout: [{ component: iro.ui.Wheel, options: {} }] }); 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 isNumeric(n) {return !isNaN(parseFloat(n)) && isFinite(n);} // returns true if dataset R, G & B values are 0 function isRgbBlack(a) {return (parseInt(a.r) == 0 && parseInt(a.g) == 0 && parseInt(a.b) == 0);} // returns RGB color from a given dataset function rgbStr(a) {return "rgb(" + a.r + "," + a.g + "," + a.b + ")";} // brightness approximation for selecting white as text color if background bri < 127, and black if higher function rgbBri(a) {return 0.2126*parseInt(a.r) + 0.7152*parseInt(a.g) + 0.0722*parseInt(a.b);} // sets background of color slot selectors function setCSL(cs) { let w = cs.dataset.w ? parseInt(cs.dataset.w) : 0; let hasShadow = getComputedStyle(cs).textShadow !== "none"; if (hasRGB && !isRgbBlack(cs.dataset)) { if (!hasShadow) cs.style.color = rgbBri(cs.dataset) > 127 ? "#000":"#fff"; // if text has no CSS "shadow" cs.style.background = (hasWhite && w > 0) ? `linear-gradient(180deg, ${rgbStr(cs.dataset)} 30%, rgb(${w},${w},${w}))` : rgbStr(cs.dataset); } else { if (hasRGB && !hasWhite) w = 0; cs.style.background = `rgb(${w},${w},${w})`; if (!hasShadow) cs.style.color = w > 127 ? "#000":"#fff"; } } function applyCfg() { cTheme(cfg.theme.base === "light"); gId("Colors").style.paddingTop = cfg.comp.colors.picker ? "0" : "28px"; var bg = cfg.theme.color.bg; if (bg) sCol('--c-1', bg); var l = cfg.comp.labels; sCol('--tbp', l ? "14px 14px 10px 14px":"10px 22px 4px 22px"); sCol('--bbp', l ? "9px 0 7px 0":"10px 0 4px 0"); sCol('--bhd', l ? "block":"none"); // show/hide labels sCol('--bmt', l ? "0px":"5px"); sCol('--t-b', cfg.theme.alpha.tab); sCol('--sgp', !cfg.comp.segpwr ? "block":"none"); // show/hide segment power size(); localStorage.setItem('wledUiCfg', JSON.stringify(cfg)); if (lastinfo.leds) updateUI(); // update component visibility } function tglHex() { cfg.comp.colors.hex = !cfg.comp.colors.hex; applyCfg(); } function tglTheme() { cfg.theme.base = (cfg.theme.base === "light") ? "dark":"light"; applyCfg(); } function tglLabels() { cfg.comp.labels = !cfg.comp.labels; applyCfg(); } function tglRgb() { cfg.comp.colors.rgb = !cfg.comp.colors.rgb; cfg.comp.colors.picker = !cfg.comp.colors.picker; applyCfg(); } function cTheme(light) { if (light) { sCol('--c-1','#eee'); sCol('--c-f','#000'); sCol('--c-2','#ddd'); sCol('--c-3','#bbb'); sCol('--c-4','#aaa'); sCol('--c-5','#999'); sCol('--c-6','#999'); sCol('--c-8','#888'); sCol('--c-b','#444'); sCol('--c-c','#333'); sCol('--c-e','#111'); sCol('--c-d','#222'); sCol('--c-r','#a21'); sCol('--c-g','#2a1'); sCol('--c-l','#26c'); sCol('--c-o','rgba(204, 204, 204, 0.9)'); sCol('--c-sb','#0003'); sCol('--c-sbh','#0006'); sCol('--c-tb','rgba(204, 204, 204, var(--t-b))'); sCol('--c-tba','rgba(170, 170, 170, var(--t-b))'); sCol('--c-tbh','rgba(204, 204, 204, var(--t-b))'); gId('imgw').style.filter = "invert(0.8)"; } else { sCol('--c-1','#111'); sCol('--c-f','#fff'); sCol('--c-2','#222'); sCol('--c-3','#333'); sCol('--c-4','#444'); sCol('--c-5','#555'); sCol('--c-6','#666'); sCol('--c-8','#888'); sCol('--c-b','#bbb'); sCol('--c-c','#ccc'); sCol('--c-e','#eee'); sCol('--c-d','#ddd'); sCol('--c-r','#e42'); sCol('--c-g','#4e2'); sCol('--c-l','#48a'); sCol('--c-o','rgba(34, 34, 34, 0.9)'); sCol('--c-sb','#fff3'); sCol('--c-sbh','#fff5'); sCol('--c-tb','rgba(34, 34, 34, var(--t-b))'); sCol('--c-tba','rgba(102, 102, 102, var(--t-b))'); sCol('--c-tbh','rgba(51, 51, 51, var(--t-b))'); gId('imgw').style.filter = "unset"; } } function loadBg() { const { url: iUrl, rnd: iRnd } = cfg.theme.bg; const bg = gId('bg'); const img = d.createElement("img"); img.src = iUrl; if (!iUrl || iRnd) { const today = new Date(); for (const holiday of (hol || [])) { const year = holiday[0] == 0 ? today.getFullYear() : holiday[0]; const holidayStart = new Date(year, holiday[1], holiday[2]); const holidayEnd = new Date(holidayStart); holidayEnd.setDate(holidayEnd.getDate() + holiday[3]); if (today >= holidayStart && today <= holidayEnd) img.src = holiday[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})`; gId('namelabel').style.color = "var(--c-c)"; // improve namelabel legibility on background image }); } 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 = getURL('/skin.css'); l.media = 'all'; h.appendChild(l); } } function getURL(path) { return (loc ? locproto + "//" + locip : "") + path; } function onLoad() { let l = window.location; if (l.protocol == "file:") { loc = true; locip = localStorage.getItem('locIp'); if (!locip) { locip = prompt("File Mode. Please enter WLED IP!"); localStorage.setItem('locIp', locip); } } else { // detect reverse proxy and/or HTTPS let pathn = l.pathname; let paths = pathn.slice(1,pathn.endsWith('/')?-1:undefined).split("/"); locproto = l.protocol; locip = l.hostname + (l.port ? ":" + l.port : ""); if (paths.length > 0 && paths[0]!=="") { loc = true; locip += "/" + paths.join('/'); } else if (locproto==="https:") { loc = true; } } var sett = localStorage.getItem('wledUiCfg'); if (sett) cfg = mergeDeep(cfg, JSON.parse(sett)); tooltip(); resetPUtil(); initFilters(); if (localStorage.getItem('pcm') == "true" || (!/Mobi/.test(navigator.userAgent) && localStorage.getItem('pcm') == null)) togglePcMode(true); applyCfg(); if (cfg.comp.hdays) { //load custom holiday list fetch(getURL("/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(); }); } else loadBg(); selectSlot(0); updateTablinks(0); cpick.on("input:end", () => {setColor(1);}); cpick.on("color:change", () => {updatePSliders()}); pmtLS = localStorage.getItem('wledPmt'); // Load initial data // Once we figure out why ESP8266 sometimes corrupts JSON responses if they are made in quick succession // we can remove all setTimeout() throttling loadPalettes(()=>{ setTimeout(()=>{ // ESP8266 can't handle quick requests // fill effect extra data array loadFXData(()=>{ setTimeout(()=>{ // ESP8266 can't handle quick requests // load and populate effects loadFX(()=>{ setTimeout(()=>{ // ESP8266 can't handle quick requests loadPalettesData(()=>{ requestJson();// will load presets and create WS if (cfg.comp.css) setTimeout(()=>{loadSkinCSS('skinCss')},100); }); },50); }); },50); }); },50); }); resetUtil(); d.addEventListener("visibilitychange", handleVisibilityChange, false); //size(); gId("cv").style.opacity=0; d.querySelectorAll('input[type="range"]').forEach((sl)=>{ 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) { var x = gId('toast'); //if (error) text += ''; 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() { gId('connind').style.backgroundColor = "var(--c-r)"; showToast('Connection to light failed!', true); } function clearErrorToast(n=5000) { var x = gId('toast'); if (x.classList.contains('error')) { clearTimeout(timeout); timeout = setTimeout(()=>{ x.classList.remove('show'); x.classList.remove('error'); }, n); } } 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 = id>250?"⚠ ID must be 250 or less.":""; } 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 += `
`; } cn += `
`; gId('pcont').innerHTML = cn; if (hasBackup) gId('bck').value = bckstr; } function restore(txt) { 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", getURL("/upload")); var formData = new FormData(); var b = new Blob([txt], {type: "application/json"}); formData.append("data", b, '/presets.json'); req.send(formData); setTimeout(loadPresets, 2000); return false; } 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; fetch(getURL('/presets.json'), { method: 'get' }) .then(res => { if (res.status=="404") return {"0":{}}; //if (!res.ok) showErrorToast(); return res.json(); }) .then(json => { pJson = json; pmtLast = pmt; populatePresets(); }) .catch((e)=>{ //showToast(e, true); presetError(false); }) .finally(()=>{ if (callback) setTimeout(callback,99); }); } function loadPalettes(callback = null) { fetch(getURL('/json/palettes'), { method: 'get' }) .then((res)=>{ if (!res.ok) showErrorToast(); return res.json(); }) .then((json)=>{ lJson = Object.entries(json); populatePalettes(); retry = false; }) .catch((e)=>{ if (!retry) { retry = true; setTimeout(loadPalettes, 500); // retry } showToast(e, true); }) .finally(()=>{ if (callback) callback(); updateUI(); }); } function loadFX(callback = null) { fetch(getURL('/json/effects'), { method: 'get' }) .then((res)=>{ if (!res.ok) showErrorToast(); return res.json(); }) .then((json)=>{ eJson = Object.entries(json); populateEffects(); retry = false; }) .catch((e)=>{ if (!retry) { retry = true; setTimeout(loadFX, 500); // retry } showToast(e, true); }) .finally(()=>{ if (callback) callback(); updateUI(); }); } function loadFXData(callback = null) { fetch(getURL('/json/fxdata'), { method: 'get' }) .then((res)=>{ if (!res.ok) showErrorToast(); return res.json(); }) .then((json)=>{ fxdata = json||[]; // add default value for Solid fxdata.shift() fxdata.unshift(";!;"); retry = false; }) .catch((e)=>{ fxdata = []; if (!retry) { retry = true; setTimeout(loadFXData, 500); // retry } 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').classList.add('expanded'); } else gId('pql').classList.remove('expanded'); 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 = ""; var arr = Object.entries(pJson).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; } else { presetError(true); } updatePA(); populateQL(); } function parseInfo(i) { lastinfo = i; var name = i.name; gId('namelabel').innerHTML = name; if (!name.match(/[\u3040-\u30ff\u3400-\u4dbf\u4e00-\u9fff\uf900-\ufaff\uff66-\uff9f\u3131-\uD79D]/)) gId('namelabel').style.transform = "rotate(180deg)"; // rotate if no CJK characters if (name === "Dinnerbone") d.documentElement.style.transform = "rotate(180deg)"; // Minecraft easter egg if (i.live) name = "(Live) " + name; if (loc) name = "(L) " + name; d.title = name; simplifiedUI = i.simplifiedui; ledCount = i.leds.count; //syncTglRecv = i.str; maxSeg = i.leds.maxseg; pmt = i.fs.pmt; gId('buttonNodes').style.display = lastinfo.ndc > 0 ? null:"none"; // do we have a matrix set-up mw = i.leds.matrix ? i.leds.matrix.w : 0; mh = i.leds.matrix ? i.leds.matrix.h : 0; isM = mw>0 && mh>0; if (!isM) { //gId("filter0D").classList.remove('hide'); //gId("filter1D").classList.add('hide'); gId("filter2D").classList.add('hide'); } else { //gId("filter0D").classList.add('hide'); //gId("filter1D").classList.remove('hide'); gId("filter2D").classList.remove('hide'); } // if (i.noaudio) { // gId("filterVol").classList.add("hide"); // gId("filterFreq").classList.add("hide"); // } // if (!i.u || !i.u.AudioReactive) { // gId("filterVol").classList.add("hide"); hideModes(" ♪"); // hide volume reactive effects // gId("filterFreq").classList.add("hide"); hideModes(" ♫"); // hide frequency reactive effects // } } //https://stackoverflow.com/questions/2592092/executing-script-elements-inserted-with-innerhtml //var setInnerHTML = function(elm, html) { // elm.innerHTML = html; // Array.from(elm.querySelectorAll("script")).forEach( oldScript => { // const newScript = document.createElement("script"); // Array.from(oldScript.attributes) // .forEach( attr => newScript.setAttribute(attr.name, attr.value) ); // newScript.appendChild(document.createTextNode(oldScript.innerHTML)); // oldScript.parentNode.replaceChild(newScript, oldScript); // }); //} //setInnerHTML(obj, html); function populateInfo(i) { var cn=""; var heap = i.freeheap/1024; 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.cn) vcn = i.cn; cn += `v${i.ver} "${vcn}"

${urows} ${urows===""?'':''} ${i.opt&0x100?inforow("Debug",""):''} ${inforow("Build",i.vid)} ${inforow("Signal strength",i.wifi.signal +"% ("+ i.wifi.rssi, " dBm)")} ${inforow("Uptime",getRuntimeStr(i.uptime))} ${inforow("Time",i.time)} ${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; // update all sliders in Info for (let sd of (d.querySelectorAll('#kv .sliderdisplay')||[])) { let s = sd.previousElementSibling; if (s) updateTrail(s); } } function populateSegments(s) { var cn = ""; let li = lastinfo; segCount = 0; lowestUnused = 0; lSeg = 0; for (var inst of (s.seg||[])) { segCount++; let i = parseInt(inst.id); if (i == lowestUnused) lowestUnused = i+1; if (i > lSeg) lSeg = i; let sg = gId(`seg${i}`); let exp = sg ? (sg.classList.contains('expanded') || (i===0 && cfg.comp.segexp)) : false; // segment set icon color let cG = "var(--c-b)"; switch (inst.set) { case 1: cG = "var(--c-r)"; break; case 2: cG = "var(--c-g)"; break; case 3: cG = "var(--c-l)"; break; } let segp = `
`+ ``+ `
`+ ``+ `
`+ `
`+ `
`; let staX = inst.start; let stoX = inst.stop; let staY = inst.startY; let stoY = inst.stopY; let isMSeg = isM && staXReverse ${isM?'':'direction'}`; let miXck = ``; let rvYck = "", miYck =""; let smpl = simplifiedUI ? 'hide' : ''; if (isMSeg) { rvYck = ``; miYck = ``; } let map2D = `
Expand 1D FX
`+ `
`+ `
`; let sndSim = `
Sound sim
`+ `
`+ `
`; cn += `
`+ ``+ `
`+ `&#x${inst.frz ? (li.live && li.liveseg==i?'e410':'e0e8') : 'e325'};`+ (inst.n ? inst.n : "Segment "+i) + `
`+ `ɸ${String.fromCharCode(inst.set+"A".charCodeAt(0))};`+ `
`+ `
`+ ``+ `
`+ ``+ (cfg.comp.segpwr ? segp : '') + `
`+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ (isMSeg ? ''+ ''+ ''+ ''+ ''+ '' : '') + ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ `
${isMSeg?'Start X':'Start LED'}${isMSeg?(cfg.comp.seglen?"Width":"Stop X"):(cfg.comp.seglen?"LED count":"Stop LED")}${isMSeg?'':'Offset'}
${isMSeg?miXck+'
'+rvXck:''}
Start Y'+(cfg.comp.seglen?'Height':'Stop Y')+'
'+miYck+'
'+rvYck+'
GroupingSpacing
`+ `
`+ (!isMSeg ? rvXck : '') + (isMSeg&&stoY-staY>1&&stoX-staX>1 ? map2D : '') + (s.AudioReactive && s.AudioReactive.on ? "" : sndSim) + ``+ `
`+ ``+ ``+ `
`+ `
`+ (cfg.comp.segpwr ? '' : segp) + `
`; } gId('segcont').innerHTML = cn; gId("segcont").classList.remove("hide"); let noNewSegs = (lowestUnused >= maxSeg); resetUtil(noNewSegs); if (gId('selall')) gId('selall').checked = true; for (var i = 0; i <= lSeg; i++) { if (!gId(`seg${i}`)) continue; updateLen(i); updateTrail(gId(`seg${i}bri`)); gId(`segr${i}`).classList.add("hide"); if (!gId(`seg${i}sel`).checked && gId('selall')) gId('selall').checked = false; // uncheck if at least one is unselected. } if (segCount < 2) { gId(`segd${lSeg}`).classList.add("hide"); if (parseInt(gId("seg0bri").value)==255) gId(`segp0`).classList.add("hide"); // hide segment controls if there is only one segment in simplified UI if (simplifiedUI) gId("segcont").classList.add("hide"); } if (!isM && !noNewSegs && (cfg.comp.seglen?parseInt(gId(`seg${lSeg}s`).value):0)+parseInt(gId(`seg${lSeg}e`).value) 1) ? "block":"none"; // rsbtn parent if (Array.isArray(li.maps) && li.maps.length>1) { let cont = `Ledmap: "; gId("ledmap").innerHTML = cont; gId("ledmap").classList.remove('hide'); } else { gId("ledmap").classList.add('hide'); } } function populateEffects() { var effects = eJson; var html = ""; effects.shift(); // temporary 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" }); for (let ef of effects) { // add slider and color control to setFX (used by requestjson) let id = ef.id; let nm = ef.name+" "; let fd = ""; if (ef.name.indexOf("RSVD") < 0) { if (Array.isArray(fxdata) && fxdata.length>id) { if (fxdata[id].length==0) fd = ";;!;1" else fd = fxdata[id]; let eP = (fd == '')?[]:fd.split(";"); // effect parameters let p = (eP.length<3 || eP[2]==='')?[]:eP[2].split(","); // palette data if (p.length>0 && (p[0] !== "" && !isNumeric(p[0]))) nm += "🎨"; // effects using palette let m = (eP.length<4 || eP[3]==='')?'1':eP[3]; // flags if (id == 0) m = ''; // solid has no flags if (m.length>0) { if (m.includes('0')) nm += "•"; // 0D effects (PWM & On/Off) if (m.includes('1')) nm += "⋮"; // 1D effects if (m.includes('2')) nm += "▦"; // 2D effects if (m.includes('v')) nm += "♪"; // volume effects if (m.includes('f')) nm += "♫"; // frequency effects } } html += generateListItemHtml('fx',id,nm,'setFX','',fd); } } gId('fxlist').innerHTML=html; } function populatePalettes() { lJson.shift(); // temporary remove default lJson.sort((a,b) => (a[1]).localeCompare(b[1])); lJson.unshift([0,"Default"]); var html = ""; for (let pa of lJson) { html += generateListItemHtml( 'palette', pa[0], pa[1], 'setPalette', `
` ); } gId('pallist').innerHTML=html; // append custom palettes (when loading for the 1st time) let li = lastinfo; if (!isEmpty(li) && li.cpalcount) { for (let j = 0; j` ); } } if (li.cpalcount>0) gId("rmPal").classList.remove("hide"); else gId("rmPal").classList.add("hide"); } 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 e = paletteData[j]; let r, g, b; let index = false; if (Array.isArray(e)) { index = Math.round(e[0]/255*100); r = e[1]; g = e[2]; b = e[3]; } else if (e == 'r') { r = Math.random() * 255; g = Math.random() * 255; b = Math.random() * 255; } else { let i = e[1] - 1; var cd = gId('csl').children; r = parseInt(cd[i].dataset.r); g = parseInt(cd[i].dataset.g); b = parseInt(cd[i].dataset.b); } if (index === false) { index = Math.round(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 = '', effectPar = '') { return `
`+ ``+ extraHtml + `
`; } function btype(b) { switch (b) { case 2: case 32: return "ESP32"; case 3: case 33: return "ESP32-S2"; case 4: case 34: return "ESP32-S3"; case 5: case 35: return "ESP32-C3"; case 1: 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 o of n.nodes) { if (o.name) { let onoff = ``; var url = `${bname(o)}${o.vid<2307130?'':onoff}`; urows += inforow(url,`${btype(o.type&0x7F)}
${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() { fetch(getURL('/json/nodes'), { method: 'get' }) .then((res)=>{ if (!res.ok) showToast('Could not load Node list!', true); return res.json(); }) .then((json)=>{ clearErrorToast(100); 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; let sd = e.parentNode.getElementsByClassName('sliderdisplay')[0]; if (sd && getComputedStyle(sd).getPropertyValue("--bg").trim() !== "none") { // trim() for Safari var max = e.hasAttribute('max') ? e.attributes.max.value : 255; var perc = Math.round(e.value * 100 / max); if (perc < 50) perc += 2; var val = `linear-gradient(90deg, var(--bg) ${perc}%, var(--c-6) ${perc}%)`; sd.style.backgroundImage = 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) + (cfg.comp.seglen?start:0); var len = stop - start; let sY = gId(`seg${s}sY`); let eY = gId(`seg${s}eY`); let sX = gId(`seg${s}s`); let eX = gId(`seg${s}e`); let of = gId(`seg${s}of`); let mySH = gId("mkSYH"); let mySD = gId("mkSYD"); if (isM) { // do we have 1D segment *after* the matrix? if (start >= mw*mh) { if (sY) { sY.value = 0; sY.max = 0; sY.min = 0; } if (eY) { eY.value = 1; eY.max = 1; eY.min = 0; } sX.min = mw*mh; sX.max = ledCount-1; eX.min = mw*mh+1; eX.max = ledCount; if (mySH) mySH.classList.add("hide"); if (mySD) mySD.classList.add("hide"); if (of) of.classList.remove("hide"); } else { // matrix setup if (mySH) mySH.classList.remove("hide"); if (mySD) mySD.classList.remove("hide"); if (of) of.classList.add("hide"); let startY = parseInt(sY.value); let stopY = parseInt(eY.value) + (cfg.comp.seglen?startY:0); len *= (stopY-startY); let tPL = gId(`seg${s}lbtm`); if (stop-start>1 && stopY-startY>1) { // 2D segment if (tPL) tPL.classList.remove('hide'); // unhide transpose checkbox let sE = gId('fxlist').querySelector(`.lstI[data-id="${selectedFx}"]`); if (sE) { let sN = sE.querySelector(".lstIname").innerText; let seg = gId(`seg${s}map2D`); if (seg) { if(sN.indexOf("\u25A6")<0) seg.classList.remove('hide'); // unhide mapping for 1D effects (| in name) else seg.classList.add('hide'); // hide mapping otherwise } } } else { // 1D segment in 2D set-up if (tPL) { tPL.classList.add('hide'); // hide transpose checkbox gId(`seg${s}tp`).checked = false; // and uncheck it } } } } 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)`; } if (isM && start >= mw*mh) out += " [strip]"; gId(`seg${s}len`).innerHTML = out; } // updates background color of currently selected preset function updatePA() { let ps; ps = gEBCN("pres"); for (let p of ps) p.classList.remove('selected'); ps = gEBCN("psts"); for (let p of ps) p.classList.remove('selected'); if (currentPreset > 0) { var acv = gId(`p${currentPreset}o`); if (acv /*&& !acv.classList.contains('expanded')*/) { 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':''; gId('pxmb').style.display = (isM) ? "inline-block" : "none"; updateSelectedFx(); updateSelectedPalette(selectedPal); // must be after updateSelectedFx() to un-hide color slots for * palettes updateTrail(gId('sliderBri')); updateTrail(gId('sliderSpeed')); updateTrail(gId('sliderIntensity')); updateTrail(gId('sliderC1')); updateTrail(gId('sliderC2')); updateTrail(gId('sliderC3')); if (hasRGB) { updateTrail(gId('sliderR')); updateTrail(gId('sliderG')); updateTrail(gId('sliderB')); } if (hasWhite) updateTrail(gId('sliderW')); var ccfg = cfg.comp.colors; gId('wwrap').style.display = (hasWhite) ? "block":"none"; // white channel gId('wbal').style.display = (hasCCT) ? "block":"none"; // white balance gId('hexw').style.display = (ccfg.hex) ? "block":"none"; // HEX input gId('picker').style.display = (hasRGB && ccfg.picker) ? "block":"none"; // color picker wheel gId('hwrap').style.display = (hasRGB && !ccfg.picker) ? "block":"none"; // hue slider gId('swrap').style.display = (hasRGB && !ccfg.picker) ? "block":"none"; // saturation slider gId('vwrap').style.display = (hasRGB) ? "block":"none"; // brightness (value) slider gId('kwrap').style.display = (hasRGB && !hasCCT) ? "block":"none"; // Kelvin slider gId('rgbwrap').style.display = (hasRGB && ccfg.rgb) ? "block":"none"; // RGB sliders gId('qcs-w').style.display = (hasRGB && ccfg.quick) ? "block":"none"; // quick selection //gId('csl').style.display = (hasRGB || hasWhite) ? "block":"none"; // color selectors (hide for On/Off bus) //gId('palw').style.display = (hasRGB) ? "inline-block":"none"; // palettes are shown/hidden in setEffectParameters() updatePA(); updatePSliders(); } function updateSelectedPalette(s) { var parent = gId('pallist'); var selPaletteInput = parent.querySelector(`input[name="palette"][value="${s}"]`); if (selPaletteInput) selPaletteInput.checked = true; var selElement = parent.querySelector('.selected'); if (selElement) selElement.classList.remove('selected'); var selectedPalette = parent.querySelector(`.lstI[data-id="${s}"]`); if (selectedPalette) parent.querySelector(`.lstI[data-id="${s}"]`).classList.add('selected'); // Display selected palette name on button in simplified UI let selectedName = selectedPalette.querySelector(".lstIname").innerText; if (simplifiedUI) { gId("palwbtn").innerText = "Palette: " + selectedName; } // in case of special palettes (* Colors...), force show color selectors (if hidden by effect data) let cd = gId('csl').children; // color selectors if (s > 1 && s < 6) { cd[0].classList.remove('hide'); // * Color 1 if (s > 2) cd[1].classList.remove('hide'); // * Color 1 & 2 if (s > 3) cd[2].classList.remove('hide'); // all colors } else { for (let i of cd) if (i.dataset.hide == '1') i.classList.add('hide'); } } 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'); selElement.style.bottom = null; // remove element style added in slider handling } var selectedEffect = parent.querySelector(`.lstI[data-id="${selectedFx}"]`); if (selectedEffect) { selectedEffect.classList.add('selected'); setEffectParameters(selectedFx); // hide non-0D effects if segment only has 1 pixel (0D) parent.querySelectorAll('.lstI').forEach((fx)=>{ let ds = fx.dataset; if (ds.opt) { let opts = ds.opt.split(";"); if (ds.id>0) { if (segLmax==0) fx.classList.add('hide'); // none of the segments selected (hide all effects) else { if ((segLmax==1 && (!opts[3] || opts[3].indexOf("0")<0)) || (!isM && opts[3] && ((opts[3].indexOf("2")>=0 && opts[3].indexOf("1")<0)))) fx.classList.add('hide'); else fx.classList.remove('hide'); } } } }); var selectedName = selectedEffect.querySelector(".lstIname").innerText; // Display selected effect name on button in simplified UI let selectedNameOnlyAscii = selectedName.replace(/[^\x00-\x7F]/g, ""); if (simplifiedUI) { gId("fxbtn").innerText = "Effect: " + selectedNameOnlyAscii; } // hide 2D mapping and/or sound simulation options var segs = gId("segcont").querySelectorAll(`div[data-map="map2D"]`); for (const seg of segs) if (selectedName.indexOf("\u25A6")<0) seg.classList.remove('hide'); else seg.classList.add('hide'); var segs = gId("segcont").querySelectorAll(`div[data-snd="si"]`); for (const seg of segs) if (selectedName.indexOf("\u266A")<0 && selectedName.indexOf("\u266B")<0) seg.classList.add('hide'); else seg.classList.remove('hide'); // also "♫ "? } } function displayRover(i,s) { gId('rover').style.transform = (i.live && s.lor == 0 && i.liveseg<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 (cfg.comp.idsort || !a[1].n) return (parseInt(a[0]) > parseInt(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 || lastinfo.ws < 0) return; let url = loc ? getURL('/ws').replace("http","ws") : "ws://"+window.location.hostname+"/ws"; ws = new WebSocket(url); 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) { parseInfo(i); 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)"; if (wsRpt++ < 5) setTimeout(makeWS,1500); // retry WS connection ws = null; } ws.onopen = (e)=>{ //ws.send("{'v':true}"); // unnecessary (https://github.com/Aircoookie/WLED/blob/master/wled00/ws.cpp#L18) wsRpt = 0; reqsLegal = true; } } function readState(s,command=false) { if (!s) return false; if (s.success) return true; // no data to process 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; if (s.pl<0) currentPreset = s.ps; else currentPreset = s.pl; tr = s.transition; gId('tt').value = tr/10; populateSegments(s); var selc=0; var sellvl=0; // 0: selc is invalid, 1: selc is mainseg, 2: selc is first selected hasRGB = hasWhite = hasCCT = false; segLmax = 0; for (let i = 0; i < (s.seg||[]).length; i++) { if (sellvl == 0 && s.seg[i].id == s.mainseg) { selc = i; sellvl = 1; } if (s.seg[i].sel) { if (sellvl < 2) selc = i; // get first selected segment sellvl = 2; var lc = lastinfo.leds.seglc[s.seg[i].id]; hasRGB |= !!(lc & 0x01); hasWhite |= !!(lc & 0x02); hasCCT |= !!(lc & 0x04); let sLen = (s.seg[i].stop - s.seg[i].start)*(s.seg[i].stopY?(s.seg[i].stopY - s.seg[i].startY):1); segLmax = segLmax < sLen ? sLen : segLmax; } } var i=s.seg[selc]; if (sellvl == 1) { var lc = lastinfo.leds.seglc[i.id]; hasRGB = !!(lc & 0x01); hasWhite = !!(lc & 0x02); hasCCT = !!(lc & 0x04); } if (!i) { showToast('No Segments!', true); updateUI(); return true; } if (s.seg.length>2) d.querySelectorAll(".pop").forEach((e)=>{e.classList.remove("hide");}); var cd = gId('csl').querySelectorAll("button"); for (let e = cd.length-1; e >= 0; e--) { cd[e].dataset.r = i.col[e][0]; cd[e].dataset.g = i.col[e][1]; cd[e].dataset.b = i.col[e][2]; if (hasWhite || (!hasRGB && !hasWhite)) { cd[e].dataset.w = i.col[e][3]; } setCSL(cd[e]); } selectSlot(csel); if (i.cct != null && i.cct>=0) gId("sliderA").value = i.cct; gId('sliderSpeed').value = i.sx; gId('sliderIntensity').value = i.ix; gId('sliderC1').value = i.c1 ? i.c1 : 0; gId('sliderC2').value = i.c2 ? i.c2 : 0; gId('sliderC3').value = i.c3 ? i.c3 : 0; gId('checkO1').checked = !(!i.o1); gId('checkO2').checked = !(!i.o2); gId('checkO3').checked = !(!i.o3); if (s.error && s.error != 0) { var errstr = ""; switch (s.error) { case 8: errstr = "Effect RAM depleted!"; break; case 9: errstr = "JSON parsing error!"; break; 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); } selectedPal = i.pal; selectedFx = i.fx; redrawPalPrev(); // if any color changed (random palette did at least) updateUI(); return true; } // control HTML elements for Slider and Color Control (original ported form WLED-SR) // 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 setEffectParameters(idx) { if (!(Array.isArray(fxdata) && fxdata.length>idx)) return; var controlDefined = fxdata[idx].length; var effectPar = fxdata[idx]; var effectPars = (effectPar == '')?[]:effectPar.split(";"); var slOnOff = (effectPars.length==0 || effectPars[0]=='')?[]:effectPars[0].split(","); var coOnOff = (effectPars.length<2 || effectPars[1]=='')?[]:effectPars[1].split(","); var paOnOff = (effectPars.length<3 || effectPars[2]=='')?[]:effectPars[2].split(","); // set html slider items on/off let sliders = d.querySelectorAll("#sliders .sliderwrap"); sliders.forEach((slider, i)=>{ let text = slider.getAttribute("tooltip"); if ((!controlDefined && i<((idx<128)?2:nSliders)) || (slOnOff.length>i && slOnOff[i]!="")) { if (slOnOff.length>i && slOnOff[i]!="!") text = slOnOff[i]; // restore overwritten default tooltips if (i<2 && slOnOff[i]==="!") text = i==0 ? "Effect speed" : "Effect intensity"; slider.setAttribute("tooltip", text); slider.parentElement.classList.remove('hide'); } else slider.parentElement.classList.add('hide'); }); if (slOnOff.length > 5) { // up to 3 checkboxes gId('fxopt').classList.remove('fade'); let checks = d.querySelectorAll("#sliders .ochkl"); checks.forEach((check, i)=>{ let text = check.getAttribute("tooltip"); if (5+i5+i && slOnOff[5+i]!="!") text = slOnOff[5+i]; check.setAttribute("tooltip", text); check.classList.remove('hide'); } else check.classList.add('hide'); }); } else gId('fxopt').classList.add('fade'); // set the bottom position of selected effect (sticky) as the top of sliders div function setSelectedEffectPosition() { if (simplifiedUI) return; let top = parseInt(getComputedStyle(gId("sliders")).height); top += 5; let sel = d.querySelector('#fxlist .selected'); if (sel) sel.style.bottom = top + "px"; // we will need to remove this when unselected (in setFX()) } setSelectedEffectPosition(); setInterval(setSelectedEffectPosition,750); // set html color items on/off var cslLabel = ''; var sep = ''; var cslCnt = 0, oCsel = csel; d.querySelectorAll("#csl button").forEach((e,i)=>{ var btn = gId("csl" + i); // if no controlDefined or coOnOff has a value if (coOnOff.length>i && coOnOff[i] != "") { btn.classList.remove('hide'); btn.dataset.hide = 0; if (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"; if (!cslCnt || oCsel==i) selectSlot(i); // select 1st displayed slot or old one cslCnt++; } else if (!controlDefined) { // if no controls then all buttons should be shown for color 1..3 btn.classList.remove('hide'); btn.dataset.hide = 0; btn.innerHTML = `${i+1}`; if (!cslCnt || oCsel==i) selectSlot(i); // select 1st displayed slot or old one cslCnt++; } else { btn.classList.add('hide'); btn.dataset.hide = 1; btn.innerHTML = `${i+1}`; // name hidden buttons 1..3 for * palettes } }); gId("cslLabel").innerHTML = cslLabel; if (cslLabel!=="") gId("cslLabel").classList.remove("hide"); else gId("cslLabel").classList.add("hide"); // set palette on/off var palw = gId("palw"); // wrapper var pall = gId("pall"); // label var icon = ' '; var text = 'Color palette'; // if not controlDefined or palette has a value if (hasRGB && ((!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)))); paOnOff[0] = paOnOff[0].substring(0,dPos); } if (paOnOff.length>0 && paOnOff[0] != "!") text = paOnOff[0]; } else { // disable palette list text += ' not used'; palw.style.display = "none"; // Close palette dialog if not available if (gId("palw").lastElementChild.tagName == "DIALOG") { gId("palw").lastElementChild.close(); } } pall.innerHTML = icon + text; // not all color selectors shown, hide palettes created from color selectors // NOTE: this will disallow user to select "* Color ..." palettes which may be undesirable in some cases or for some users //for (let e of (gId('pallist').querySelectorAll('.lstI')||[])) { // let fltr = "* C"; // if (cslCnt==1 && csel==0) fltr = "* Colors"; // else if (cslCnt==2) fltr = "* Colors Only"; // if (cslCnt < 3 && e.querySelector('.lstIname').innerText.indexOf(fltr)>=0) e.classList.add('hide'); else e.classList.remove('hide'); //} } var jsonTimeout; var reqsLegal = false; function requestJson(command=null) { gId('connind').style.backgroundColor = "var(--c-y)"; if (command && !reqsLegal) return; // stop post requests from chrome onchange event on page restore if (!jsonTimeout) jsonTimeout = setTimeout(()=>{if (ws) ws.close(); ws=null; showErrorToast()}, 3000); var req = null; var useWs = (ws && ws.readyState === WebSocket.OPEN); var type = command ? 'post':'get'; if (command) { command.v = true; // force complete /json/si API response 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 > 1340) useWs = false; // do not send very long requests over websocket if (req.length > 500 && lastinfo && lastinfo.arch == "esp8266") useWs = false; // esp8266 can only handle 500 bytes }; if (useWs) { ws.send(req?req:'{"v":true}'); return; } fetch(getURL('/json/si'), { method: type, headers: { "Content-type": "application/json; charset=UTF-8" }, body: req }) .then(res => { clearTimeout(jsonTimeout); jsonTimeout = null; if (!res.ok) showErrorToast(); return res.json(); }) .then(json => { lastUpdate = new Date(); clearErrorToast(3000); gId('connind').style.backgroundColor = "var(--c-g)"; if (!json) { showToast('Empty response', true); return; } if (json.success) return; if (json.info) { let i = json.info; parseInfo(i); populatePalettes(i); if (isInfo) populateInfo(i); if (simplifiedUI) simplifyUI(); } var s = json.state ? json.state : json; readState(s); //load presets and open websocket sequentially if (!pJson || isEmpty(pJson)) setTimeout(()=>{ loadPresets(()=>{ wsRpt = 0; if (!(ws && ws.readyState === WebSocket.OPEN)) makeWS(); }); },25); reqsLegal = true; retry = false; }) .catch((e)=>{ if (!retry) { retry = true; setTimeout(requestJson,500); } showToast(e, true); }); } function togglePower() { isOn = !isOn; var obj = {"on": isOn}; if (isOn && lastinfo && lastinfo.live && lastinfo.liveseg>=0) { obj.live = false; obj.seg = []; obj.seg[0] = {"id": lastinfo.liveseg, "frz": false}; } if (cfg.comp.on >0 && isOn) obj = {"ps": cfg.comp.on }; // don't use setPreset() if (cfg.comp.off>0 && !isOn) obj = {"ps": cfg.comp.off}; // don't use setPreset() 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() { if (isInfo && isM) toggleInfo(); if (isNodes && isM) toggleNodes(); isLv = !isLv; let wsOn = ws && ws.readyState === WebSocket.OPEN; var lvID = "liveview"; if (isM && wsOn) { lvID += "2D"; if (isLv) gId('klv2D').innerHTML = ``; gId('mlv2D').style.transform = (isLv) ? "translateY(0px)":"translateY(100%)"; } gId(lvID).style.display = (isLv) ? "block":"none"; gId(lvID).src = (isLv) ? getURL("/" + lvID + ((wsOn) ? "?ws":"")):"about:blank"; gId('buttonSr').classList.toggle("active"); if (!isLv && wsOn) ws.send('{"lv":false}'); size(); } function toggleInfo() { if (isNodes) toggleNodes(); if (isLv && isM) toggleLiveview(); isInfo = !isInfo; if (isInfo) requestJson(); gId('info').style.transform = (isInfo) ? "translateY(0px)":"translateY(100%)"; gId('buttonI').className = (isInfo) ? "active":""; } function toggleNodes() { if (isInfo) toggleInfo(); if (isLv && isM) toggleLiveview(); isNodes = !isNodes; if (isNodes) loadNodes(); gId('nodes').style.transform = (isNodes) ? "translateY(0px)":"translateY(100%)"; gId('buttonNodes').className = (isNodes) ? "active":""; } function makeSeg() { var ns = 0, ct = 0; var lu = lowestUnused; let li = lastinfo; if (lu > 0) { let xend = parseInt(gId(`seg${lu -1}e`).value,10) + (cfg.comp.seglen?parseInt(gId(`seg${lu -1}s`).value,10):0); if (isM) { ns = 0; ct = mw; } else { if (xend < ledCount) ns = xend; ct = ledCount-(cfg.comp.seglen?ns:0) } } gId('segutil').scrollIntoView({ behavior: 'smooth', block: 'start', }); var cn = `
`+ `
`+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ ``+ `
${isM?'Start X':'Start LED'}${isM?(cfg.comp.seglen?"Width":"Stop X"):(cfg.comp.seglen?"LED count":"Stop LED")}
Start Y${cfg.comp.seglen?'Height':'Stop Y'}
`+ `
${ledCount - ns} LEDs
`+ `
`+ `
`+ `
`; gId('segutil').innerHTML = cn; } function resetUtil(off=false) { gId('segutil').innerHTML = `
` + '' + `
Add segment
` + '
' + `` + '
' + '
'; } function makePlSel(el, incPl=false) { var plSelContent = ""; delete pJson["0"]; // remove filler preset Object.entries(pJson).sort(cmpP).forEach((a)=>{ var n = a[1].n ? a[1].n : "Preset " + a[0]; if (cfg.comp.idsort) n = a[0] + ' ' + n; if (!(!incPl && a[1].playlist && a[1].playlist.ps)) // skip playlists, sub-playlists not yet supported plSelContent += ``; }); return plSelContent; } function refreshPlE(p) { var plEDiv = gId(`ple${p}`); if (!plEDiv) return; var content = "
Playlist entries
"; plJson[p].ps.forEach((e,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"; d.querySelectorAll(`#seg${p+100} .sel`).forEach((i)=>{ 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) { if (i===0) plJson[0] = { ps: [1], dur: [100], transition: [tr], repeat: 0, r: false, end: 0 }; 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>1) { 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() { let p = gId('putil'); p.classList.remove('staybot'); p.classList.add('pres'); p.innerHTML = `
${makeP(0)}
`; let pTx = gId('p0txt'); pTx.focus(); pTx.value = eJson.find((o)=>{return o.id==selectedFx}).name; pTx.select(); p.scrollIntoView({ behavior: 'smooth', block: 'center' }); gId('psFind').classList.remove('staytop'); } function makePlEntry(p,i) { return `
Duration Transition #${i+1}
s s
`; } function makePlUtil() { if (pNum < 2) { showToast("You need at least 2 presets to make a playlist!"); //return; } let p = gId('putil'); p.classList.remove('staybot'); p.classList.add('pres'); p.innerHTML = `
${makeP(0,true)}
`; refreshPlE(0); gId('p0txt').focus(); p.scrollIntoView({ behavior: 'smooth', block: 'center' }); gId('psFind').classList.remove('staytop'); } function resetPUtil() { gId('psFind').classList.add('staytop'); let p = gId('putil'); p.classList.add('staybot'); p.classList.remove('pres'); p.innerHTML = `` + ``; } 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) { let t = gId(s<100?`seg${s}t`:`p${s-100}txt`); if (t) { t.classList.toggle('show'); t.focus(); t.select(); } event.preventDefault(); event.stopPropagation(); } function selSegAll(o) { var obj = {"seg":[]}; for (let i=0; i<=lSeg; i++) if (gId(`seg${i}`)) obj.seg.push({"id":i,"sel":o.checked}); requestJson(obj); } function selSegEx(s) { var obj = {"seg":[]}; for (let i=0; i<=lSeg; i++) if (gId(`seg${i}`)) obj.seg.push({"id":i,"sel":(i==s)}); obj.mainseg = s; requestJson(obj); } function selSeg(s) { var sel = gId(`seg${s}sel`).checked; var obj = {"seg": {"id": s, "sel": sel}}; requestJson(obj); } function selGrp(g) { event.preventDefault(); event.stopPropagation(); var sel = gId(`segcont`).querySelectorAll(`div[data-set="${g}"]`); var obj = {"seg":[]}; for (let i=0; i<=lSeg; i++) if (gId(`seg${i}`)) obj.seg.push({"id":i,"sel":false}); for (let s of (sel||[])) { let i = parseInt(s.id.substring(3)); obj.seg[i] = {"id":i,"sel":true}; } if (obj.seg.length) requestJson(obj); } function rptSeg(s) { //TODO: 2D support 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 pwr = gId(`seg${s}pwr`).classList.contains('act'); var obj = {"seg": {"id": s, "n": name, "start": start, "stop": (cfg.comp.seglen?start:0)+stop, "rev": rev, "mi": mi, "on": pwr, "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; let sX = gId(`seg${s}s`); let eX = gId(`seg${s}e`); var start = parseInt(sX.value); var stop = parseInt(eX.value) + (cfg.comp.seglen?start:0); if (startsX.max) {sX.value=sX.min; return;} // prevent out of bounds if (stopeX.max) {eX.value=eX.max; return;} // prevent out of bounds if ((cfg.comp.seglen && stop == 0) || (!cfg.comp.seglen && stop <= start)) {delSeg(s); return;} var obj = {"seg": {"id": s, "n": name, "start": start, "stop": stop}}; if (isM && startsY.max) {sY.value=sY.min; return;} // prevent out of bounds if (stopYeY.max) {eY.value=eY.max; return;} // prevent out of bounds obj.seg.startY = startY; obj.seg.stopY = stopY; } let g = gId(`seg${s}grp`); if (g) { // advanced options, not present in new segment dialog (makeSeg()) let grp = parseInt(g.value); let spc = parseInt(gId(`seg${s}spc`).value); let ofs = parseInt(gId(`seg${s}of` ).value); obj.seg.grp = grp; obj.seg.spc = spc; obj.seg.of = ofs; if (isM && gId(`seg${s}tp`)) obj.seg.tp = gId(`seg${s}tp`).checked; } resetUtil(); // close add segment dialog just in case requestJson(obj); } function delSeg(s) { if (segCount < 2) { showToast("You need to have multiple segments to delete one!"); return; } 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 setRevY(s) { var rev = gId(`seg${s}rY`).checked; var obj = {"seg": {"id": s, "rY": rev}}; requestJson(obj); } function setMi(s) { var mi = gId(`seg${s}mi`).checked; var obj = {"seg": {"id": s, "mi": mi}}; requestJson(obj); } function setMiY(s) { var mi = gId(`seg${s}mY`).checked; var obj = {"seg": {"id": s, "mY": mi}}; requestJson(obj); } function setM12(s) { var value = gId(`seg${s}m12`).selectedIndex; var obj = {"seg": {"id": s, "m12": value}}; requestJson(obj); } function setSi(s) { var value = gId(`seg${s}si`).selectedIndex; var obj = {"seg": {"id": s, "si": value}}; requestJson(obj); } function setTp(s) { var tp = gId(`seg${s}tp`).checked; var obj = {"seg": {"id": s, "tp": tp}}; requestJson(obj); } function setGrp(s, g) { event.preventDefault(); event.stopPropagation(); var obj = {"seg": {"id": s, "set": g}}; requestJson(obj); } function setSegPwr(s) { var pwr = gId(`seg${s}pwr`).classList.contains('act'); var obj = {"seg": {"id": s, "on": !pwr}}; 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.seg.id = s; // if live segment, enter live override (which also unfreezes) if (lastinfo && s==lastinfo.liveseg && lastinfo.live) obj = {"lor":1}; } requestJson(obj); } function setFX(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; } // Close effect dialog in simplified UI if (simplifiedUI) { gId("fx").lastElementChild.close(); } var obj = {"seg": {"fx": parseInt(ind), "fxdef": cfg.comp.fxdef}}; // fxdef sets effect parameters to default values 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; } // Close palette dialog in simplified UI if (simplifiedUI) { gId("palw").lastElementChild.close(); } 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 setCustom(i=1) { if (i<1 || i>3) return; var obj = {"seg": {}}; var val = parseInt(gId(`sliderC${i}`).value); if (i===3) obj.seg.c3 = val; else if (i===2) obj.seg.c2 = val; else obj.seg.c1 = val; requestJson(obj); } function setOption(i=1, v=false) { if (i<1 || i>3) return; var obj = {"seg": {}}; if (i===3) obj.seg.o3 = !(!v); //make sure it is bool else if (i===2) obj.seg.o2 = !(!v); //make sure it is bool else obj.seg.o1 = !(!v); //make sure it is bool requestJson(obj); } function setLor(i) { var obj = {"lor": i}; requestJson(obj); } function setPreset(i) { var obj = {"ps":i}; if (!isPlaylist(i) && pJson && pJson[i] && (!pJson[i].win || pJson[i].win.indexOf("Please") <= 0)) { // we will send the complete preset content as to avoid delay introduced by // async nature of applyPreset() and having to read the preset from file system. obj = {"pd":i}; // use "pd" instead of "ps" to indicate that we are sending the preset content directly Object.assign(obj, pJson[i]); delete obj.ql; // no need for quick load delete obj.n; // no need for name } 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(); if (pI > 250) {alert("Preset ID must be 250 or less."); return;} 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`) && 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(); setTimeout(()=>{pmtLast=0; loadPresets();}, 750); // force reloading of presets } 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(); gId('putil').classList.add('staybot'); } 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 of cd) i.classList.remove('xxs-w'); cd[b].classList.add('xxs-w'); setPicker(rgbStr(cd[b].dataset)); // force slider update on initial load (picker "color:change" not fired if black) if (cpick.color.value == 0) updatePSliders(); gId('sliderW').value = parseInt(cd[b].dataset.w); updateTrail(gId('sliderW')); redrawPalPrev(); } // 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 = parseInt(gId("csl").children[csel].dataset.w); if (w > 0) str += w.toString(16); gId('hexc').value = str; gId('hexcnf').style.backgroundColor = "var(--c-3)"; // update HSV sliders var c; let h = cpick.color.hue; let s = cpick.color.saturation; let v = cpick.color.value; gId("sliderH").value = h; gId("sliderS").value = s; gId('sliderV').value = v; c = iro.Color.hsvToRgb({"h":h,"s":100,"v":100}); gId("sliderS").nextElementSibling.style.backgroundImage = 'linear-gradient(90deg, #aaa -15%, rgb('+c.r+','+c.g+','+c.b+'))'; c = iro.Color.hsvToRgb({"h":h,"s":s,"v":100}); gId('sliderV').nextElementSibling.style.backgroundImage = 'linear-gradient(90deg, #000 -15%, rgb('+c.r+','+c.g+','+c.b+'))'; // update Kelvin slider gId('sliderK').value = cpick.color.kelvin; } function hexEnter() { if(event.keyCode == 13) fromHex(); } function segEnter(s) { if(event.keyCode == 13) setSeg(s); } function fromHex() { var str = gId('hexc').value; let w = parseInt(str.substring(6), 16); try { setPicker("#" + str.substring(0,6)); } catch (e) { setPicker("#ffaa00"); } gId("csl").children[csel].dataset.w = isNaN(w) ? 0 : w; 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); updateTrail(gId('sliderR')); updateTrail(gId('sliderG')); updateTrail(gId('sliderB')); } function fromH() { cpick.color.setChannel('hsv', 'h', gId('sliderH').value); } function fromS() { cpick.color.setChannel('hsv', 's', gId('sliderS').value); } function fromV() { cpick.color.setChannel('hsv', 'v', gId('sliderV').value); } function fromK() { cpick.color.set({ kelvin: gId('sliderK').value }); } function fromRgb() { var r = gId('sliderR').value; var g = gId('sliderG').value; var b = gId('sliderB').value; setPicker(`rgb(${r},${g},${b})`); let cd = gId('csl').children; // color slots cd[csel].dataset.r = r; cd[csel].dataset.g = g; cd[csel].dataset.b = b; setCSL(cd[csel]); } function fromW() { let w = gId('sliderW'); let cd = gId('csl').children; // color slots cd[csel].dataset.w = w.value; setCSL(cd[csel]); updateTrail(w); } // sr 0: from RGB sliders, 1: from picker, 2: from hex function setColor(sr) { var cd = gId('csl').children; // color slots let cdd = cd[csel].dataset; let w = 0, r,g,b; if (sr == 1 && isRgbBlack(cdd)) cpick.color.setChannel('hsv', 'v', 100); if (sr != 2 && hasWhite) w = parseInt(gId('sliderW').value); var col = cpick.color.rgb; cdd.r = r = hasRGB ? col.r : w; cdd.g = g = hasRGB ? col.g : w; cdd.b = b = hasRGB ? col.b : w; cdd.w = w; setCSL(cd[csel]); var obj = {"seg": {"col": [[],[],[]]}}; obj.seg.col[csel] = [r, g, b, w]; requestJson(obj); } function setBalance(b) { var obj = {"seg": {"cct": parseInt(b)}}; requestJson(obj); } function rmtTgl(ip,i) { event.preventDefault(); event.stopPropagation(); fetch(`http://${ip}/win&T=2`, {method: 'get'}) .then((r)=>{ return r.text(); }) .then((t)=>{ let c = (new window.DOMParser()).parseFromString(t, "text/xml"); // perhaps just i.classList.toggle("off"); would be enough if (c.getElementsByTagName('ac')[0].textContent === "0") { i.classList.add("off"); } else { i.classList.remove("off"); } }); } 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 = getURL("/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}]}; if (isM) { obj.seg[0].stop = mw; obj.seg[0].startX = 0; obj.seg[0].stopY = mh; } 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(); return; } } catch (e) {} } palettesData = {}; getPalettesData(0, ()=>{ localStorage.setItem(lsKey, JSON.stringify({ p: palettesData, vid: lastinfo.vid })); redrawPalPrev(); if (callback) setTimeout(callback, 99); }); } function getPalettesData(page, callback) { fetch(getURL(`/json/palx?page=${page}`), { 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); }); } /* function hideModes(txt) { for (let e of (gId('fxlist').querySelectorAll('.lstI')||[])) { let iT = e.querySelector('.lstIname').innerText; let f = false; if (txt==="2D") f = iT.indexOf("\u25A6") >= 0 && iT.indexOf("\u22EE") < 0; // 2D && !1D else f = iT.indexOf(txt) >= 0; if (f) e.classList.add('hide'); //else e.classList.remove('hide'); } } */ function search(field, listId = null) { field.nextElementSibling.style.display = (field.value !== '') ? 'block' : 'none'; if (!listId) return; const search = field.value !== ''; // clear filter if searching in fxlist if (listId === 'fxlist' && search) { gId("filters").querySelectorAll("input[type=checkbox]").forEach((e) => { e.checked = false; }); } // do not search if filter is active if (gId("filters").querySelectorAll("input[type=checkbox]:checked").length) return; const listItems = gId(listId).querySelectorAll('.lstI'); // filter list items but leave (Default & Solid) always visible for (i = (listId === 'pcont' ? 0 : 1); i < listItems.length; i++) { const listItem = listItems[i]; const listItemName = listItem.querySelector('.lstIname').innerText.toUpperCase(); const searchIndex = listItemName.indexOf(field.value.toUpperCase()); listItem.style.display = (searchIndex < 0) ? 'none' : ''; listItem.dataset.searchIndex = searchIndex; } // sort list items by search index and name const sortedListItems = Array.from(listItems).sort((a, b) => { const aSearchIndex = parseInt(a.dataset.searchIndex); const bSearchIndex = parseInt(b.dataset.searchIndex); if (aSearchIndex !== bSearchIndex) { return aSearchIndex - bSearchIndex; } const aName = a.querySelector('.lstIname').innerText.toUpperCase(); const bName = b.querySelector('.lstIname').innerText.toUpperCase(); return aName.localeCompare(bName); }); sortedListItems.forEach(item => { gId(listId).append(item); }); // scroll to first search result const firstVisibleItem = sortedListItems.find(item => item.style.display !== 'none' && !item.classList.contains('sticky') && !item.classList.contains('selected')); if (firstVisibleItem && search) { firstVisibleItem.scrollIntoView({ behavior: "instant", block: "center" }); } } function clean(clearButton) { clearButton.style.display = 'none'; const inputField = clearButton.previousElementSibling; inputField.value = ''; search(inputField, clearButton.parentElement.nextElementSibling.id); } function initFilters() { gId("filters").querySelectorAll("input[type=checkbox]").forEach((e) => { e.checked = false; }); } function filterFocus(e) { const f = gId("filters"); if (e.type === "focus") f.classList.remove('fade'); // immediately show (still has transition) // compute sticky top (with delay for transition) setTimeout(() => { const sti = parseInt(getComputedStyle(d.documentElement).getPropertyValue('--sti')) + (e.type === "focus" ? 1 : -1) * f.offsetHeight; sCol('--sti', sti + "px"); }, 252); if (e.type === "blur") { setTimeout(() => { if (e.target === document.activeElement && document.hasFocus()) return; // do not hide if filter is active if (gId("filters").querySelectorAll("input[type=checkbox]:checked").length) return; f.classList.add('fade'); }, 255); // wait with hiding } } function filterFx() { const inputField = gId('fxFind').children[0]; inputField.value = ''; inputField.focus(); clean(inputField.nextElementSibling); const listItems = gId("fxlist").querySelectorAll('.lstI'); for (let i = 1; i < listItems.length; i++) { const listItem = listItems[i]; const listItemName = listItem.querySelector('.lstIname').innerText; let hide = false; gId("filters").querySelectorAll("input[type=checkbox]").forEach((e) => { if (e.checked && !listItemName.includes(e.dataset.flt)) hide = true; }); listItem.style.display = hide ? 'none' : ''; } } function preventBlur(e) { if (e.target === gId("fxFind").children[0] || e.target === gId("filters")) return; e.preventDefault(); } // 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) { var seg = i<100 ? gId('seg' +i) : gId(`p${i-100}o`); let ps = gId("pcont").children; // preset wrapper if (i>100) for (let p of ps) { p.classList.remove('selected'); if (p!==seg) p.classList.remove('expanded'); } // collapse all other presets & remove selected seg.classList.toggle('expanded'); // presets if (i >= 100) { var p = i-100; if (seg.classList.contains('expanded')) { 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; gId('seg' +i).innerHTML = makeP(p,true); refreshPlE(p); } else { gId('seg' +i).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); gId('putil').classList.remove('staybot'); } else { updatePA(); gId('seg' +i).innerHTML = ""; gId('putil').classList.add('staybot'); } } seg.scrollIntoView({ behavior: 'smooth', block: '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; 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; } //required by rangetouch.js function lock(e) { if (pcMode || simplifiedUI) 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)); } //required by rangetouch.js function move(e) { if(!locked || pcMode || simplifiedUI) return; var clientX = unify(e).clientX; var dx = clientX - x0; var s = Math.sign(dx); var f = +(s*dx/wW).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 size() { wW = window.innerWidth; var h = gId('top').clientHeight; sCol('--th', h + "px"); sCol('--bh', gId('bot').clientHeight + "px"); if (isLv) h -= 4; sCol('--tp', h + "px"); togglePcMode(); lastw = wW; } function togglePcMode(fromB = false) { if (fromB) { pcModeA = !pcModeA; localStorage.setItem('pcm', pcModeA); } pcMode = (wW >= 1024) && pcModeA; if (cpick) cpick.resize(pcMode && wW>1023 && wW<1250 ? 230 : 260); // for tablet in landscape if (!fromB && ((wW < 1024 && lastw < 1024) || (wW >= 1024 && lastw >= 1024))) return; // no change in size and called from size() openTab(0, true); 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 || simplifiedUI)?'100%':'400%'; } 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); } function tooltip() { const elements = d.querySelectorAll("[title]"); elements.forEach((element)=>{ element.addEventListener("mouseover", ()=>{ // save title element.setAttribute("data-title", element.getAttribute("title")); const tooltip = d.createElement("span"); tooltip.className = "tooltip"; tooltip.textContent = element.getAttribute("title"); // prevent default title popup element.removeAttribute("title"); let { top, left, width } = element.getBoundingClientRect(); d.body.appendChild(tooltip); const { offsetHeight, offsetWidth } = tooltip; const offset = element.classList.contains("sliderwrap") ? 4 : 10; top -= offsetHeight + offset; left += (width - offsetWidth) / 2; tooltip.style.top = top + "px"; tooltip.style.left = left + "px"; tooltip.classList.add("visible"); }); element.addEventListener("mouseout", ()=>{ const tooltips = d.querySelectorAll('.tooltip'); tooltips.forEach((tooltip)=>{ tooltip.classList.remove("visible"); d.body.removeChild(tooltip); }); // restore title element.setAttribute("title", element.getAttribute("data-title")); }); }); }; // Transforms the default UI into the simple UI function simplifyUI() { // Create dropdown dialog function createDropdown(id, buttonText, dialogElements = null) { // Create dropdown dialog const dialog = document.createElement("dialog"); // Move every dialogElement to the dropdown dialog or if none are given, move all children of the element with the given id if (dialogElements) { dialogElements.forEach((e) => { dialog.appendChild(e); }); } else { while (gId(id).firstChild) { dialog.appendChild(gId(id).firstChild); } } // Create button for the dropdown const btn = document.createElement("button"); btn.id = id + "btn"; btn.classList.add("btn"); btn.innerText = buttonText; function toggleDialog(e) { if (e.target != btn && e.target != dialog) return; if (dialog.open) { dialog.close(); return; } // Prevent autofocus on dialog open dialog.inert = true; dialog.showModal(); dialog.inert = false; clean(dialog.firstElementChild.children[1]); dialog.scrollTop = 0; }; btn.addEventListener("click", toggleDialog); dialog.addEventListener("click", toggleDialog); // Add the dialog and button to the element with the given id gId(id).append(btn); gId(id).append(dialog); } // Check if the UI was already simplified if (gId("Colors").classList.contains("simplified")) return; // Disable PC Mode as it does not exist in simple UI if (pcMode) togglePcMode(true); _C.style.width = '100%' _C.style.setProperty('--n', 1); gId("Colors").classList.add("simplified"); // Put effects below palett list gId("Colors").append(gId("fx")); gId("Colors").append(gId("sliders")); // Put segments before palette list gId("Colors").insertBefore(gId("segcont"), gId("pall")); // Put preset quick load before palette list and segemts gId("Colors").insertBefore(gId("pql"), gId("pall")); // Create dropdown for palette list createDropdown("palw", "Change palette"); createDropdown("fx", "Change effect", [gId("fxFind"), gId("fxlist")]); // Hide pallete label gId("pall").style.display = "none"; gId("Colors").insertBefore(document.createElement("br"), gId("pall")); // Hide effect label gId("modeLabel").style.display = "none"; // Hide buttons in top bar gId("buttonNl").style.display = "none"; gId("buttonSync").style.display = "none"; gId("buttonSr").style.display = "none"; gId("buttonPcm").style.display = "none"; // Hide bottom bar gId("bot").style.display = "none"; document.documentElement.style.setProperty('--bh', '0px'); // Hide other tabs gId("Effects").style.display = "none"; gId("Segments").style.display = "none"; gId("Presets").style.display = "none"; // Hide filter options gId("filters").style.display = "none"; // Hide buttons for pixel art and custom palettes (add / delete) gId("btns").style.display = "none"; } size(); _C.style.setProperty('--n', N); window.addEventListener('resize', size, true); _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);