-- -- Astra: Web Interface -- https://cesbo.com/astra/ -- -- Copyright (C) 2020, Andrey Dyldin -- log.info("Load UI bundle") astra_storage["/"] = [==[ Astra Control Panel ]==] astra_storage["/app.js"] = base64.decode([==[ /*
 * Astra: Web Interface
 * https://cesbo.com/astra/
 *
 * Copyright (C) 2020, Andrey Dyldin <and@cesbo.com>
 */

(function(){"use strict";window.$=function(x){if(typeof x==="string"){return document.querySelectorAll(x)}else if(typeof x==="function"){document.addEventListener("DOMContentLoaded",(function w(){document.removeEventListener("DOMContentLoaded",w,false);x()}))}};String.prototype.toUpperCaseFirst=function(){return this.charAt(0).toUpperCase()+this.slice(1)};if(!String.prototype.format)String.prototype.format=function(){var args=arguments;return this.replace(/{(\d+)}/g,(function(match,number){return typeof args[number]!="undefined"?args[number]:match}))};if(!String.prototype.trim)String.prototype.trim=function(){return this.replace(/^\s+|\s+$/gm,"")};Array.prototype.find=function(fn){for(var i=0,l=this.length;i<l;++i){var x=this[i];if(fn(x))return x}return undefined};Array.prototype.move=function(src,dst){var e=this.splice(src,1)[0];this.splice(dst,0,e)};Array.prototype.insertSorted=function(item,fn){var l=0,h=this.length,m;while(l<h){m=l+h>>>1;if(fn(this[m],item)<0)l=m+1;else h=m}this.splice(l,0,item)};Element.prototype.remove=function(){this.parentNode.removeChild(this)};Element.prototype.empty=function(){while(this.firstChild)this.removeChild(this.firstChild);return this};Element.prototype.addClass=function(){var clist=!!this.className?this.className.split(/ +/):[];for(var i=0;i<arguments.length;i++){var n=arguments[i];if(!n)continue;n=n.split(/ +/);while(n.length){var c=n.shift();if(clist.indexOf(c)==-1)clist.push(c)}}this.className=clist.join(" ");return this};Element.prototype.removeClass=function(){var clist=!!this.className?this.className.split(/ +/):[];for(var i=0;i<arguments.length;i++){var n=arguments[i];if(!n)continue;n=n.split(/ +/);while(n.length){var c=n.shift();var j=clist.indexOf(c);if(j!=-1)clist.splice(j,1)}}this.className=clist.join(" ");return this};Element.prototype.hasClass=function(c){var clist=this.className.split(/ +/);return clist.indexOf(c)!=-1};Element.prototype.addAttr=function(key,value){if(value==undefined)value="";this.setAttribute(key,value);return this};Element.prototype.removeAttr=function(key){this.removeAttribute(key);return this};Element.prototype.setValue=function(value){if(value)this.addAttr("value",value);else this.removeAttr("value");return this};Element.prototype.setText=function(text){this.textContent=text;return this};Element.prototype.setHtml=function(html){this.innerHTML=html;return this};Element.prototype.setReadonly=function(value){return this[!!value?"addAttr":"removeAttr"]("readonly")};Element.prototype.setDisabled=function(value){return this[!!value?"addAttr":"removeAttr"]("disabled")};Element.prototype.setRequired=function(value){return this[!!value?"addAttr":"removeAttr"]("required")};Element.prototype.setError=function(value){return this[!!value?"addClass":"removeClass"]("error")};Element.prototype.setStyle=function(key,value){var x=key.split("-");key=x.shift();while(x.length)key+=x.shift().toUpperCaseFirst();this.style[key]=value;return this};Element.prototype.addValidator=function(fn){if(this.$validate==undefined)this.$validate=[];this.$validate.push(fn);return this};Element.prototype.addChild=function(){for(var i=0;i<arguments.length;++i){var child=arguments[i];if(child){if(child.nodeName==undefined)child=document.createTextNode(child);this.appendChild(child)}}return this};Element.prototype.insertChild=function(child,before){if(child){if(child.nodeName==undefined)child=document.createTextNode(child);this.insertBefore(child,before||this.firstChild)}return this};Element.prototype.insertSorted=function(child,fn){var l=0,h=this.childNodes.length,m;while(l<h){m=l+h>>>1;if(fn(this.childNodes[m],child)<0)l=m+1;else h=m}if(l==this.childNodes.length)this.addChild(child);else this.insertChild(child,this.childNodes[l]);return this};Element.prototype.dataBind=function(key){if(key)this.addAttr("data-bind",key);return this};var eventOn=function(event,fn){if(!this.eventBind)this.eventBind={};var eb=this.eventBind[event];if(!eb){this.eventBind[event]=eb={list:[],fn:function(){for(var i=0;i<eb.list.length;i++)eb.list[i].apply(this,arguments)}};this.addEventListener(event,eb.fn)}eb.list.push(fn);return this};var eventOff=function(event,fn){var eb=!!this.eventBind?this.eventBind[event]:null;if(eb){if(!!fn){var x=eb.list.indexOf(fn);if(x!=-1)eb.list.splice(x,1);if(!eb.list.length)fn=undefined}if(!fn){this.removeEventListener(event,eb.fn);delete this.eventBind[event]}}return this};var eventEmit=function(event){var eb=!!this.eventBind?this.eventBind[event]:null;if(eb){eb.fn.apply(this,Array.prototype.slice.call(arguments,1))}return this};Element.prototype.on=eventOn;Element.prototype.off=eventOff;Element.prototype.emit=eventEmit;window.on=eventOn;window.off=eventOff;window.emit=eventEmit;document.on=eventOn;document.off=eventOff;document.emit=eventEmit;$.html=document.querySelector("html");$.head=document.querySelector("head");$.nop=function(){};$.element=function(t){return document.createElement(t||"div")};$.body=$.element();$.base=document.querySelector("body");$.base.addChild($.body);$.element.button=function(text){return $.element("button").addAttr("type","button").addClass("button").setText(text||"")};$.element.a=function(href,text){return $.element("a").addAttr("href",href).setText(text||"")};$.element.input=function(placeholder,type){return $.element("input").addAttr("type",type||"text").addClass("input").addAttr("placeholder",placeholder)};$.element.select=function(items){var x=$.element("select").addClass("input");var makeItems=function(p,x){for(var i=0;i<x.length;i++){var e,o=x[i];if(o.hide)continue;if(o.group){if(!o.items.length)continue;e=$.element("optgroup").addAttr("label",o.group);makeItems(e,o.items)}else{e=$.element("option").addAttr("value",o.value).setText(o.label||o.value)}if(o.disabled)e.setDisabled(true);p.addChild(e)}};makeItems(x,items);return x};$.element.checkbox=function(label){var x=$.element("input").addAttr("type","checkbox");var z=$.element("label").addClass("checkbox").addChild(x).addChild($.element("span").addClass("inner")).addChild($.element("span").addClass("text").setHtml(label||""));z.checkbox=x;z.dataBind=function(key){x.dataBind(key)};return z};$.element.script=function(src){return $.element("script").addAttr("type","text/javascript").addAttr("src",src)};$.isArray=function(v){return!!v&&v.length!=undefined&&(v.constructor==Array||v.constructor==NodeList)};$.isObject=function(v){return v===Object(v)&&$.isArray(v)==false};$.isObjectEmpty=function(v){for(var k in v){if(v.hasOwnProperty(k))return false}return true};$.clone=function(v){if($.isArray(v))return v.slice();if($.isObject(v))return JSON.parse(JSON.stringify(v));return v};$.forEach=function(v,fn){if(!v){}else if($.isArray(v)){for(var k=0,l=v.length;k<l;k++){var x=fn(v[k],k);if(x!=undefined)return x}}else if($.isObject(v)){for(var k in v){var x=fn(v[k],k);if(x!=undefined)return x}}};$.http=function(config,onLoad,onError){var u=config.url,m=config.method||"GET";if(u.match(/^\/\//))u=location.protocol+u;var x=new XMLHttpRequest;if(!u.match(/^[a-z]+:\/\//)){x.open(m,u)}else if("withCredentials"in x){x.open(m,u,true)}else if("XDomainRequest"in window){x=new XDomainRequest;x.open(m,u)}else{throw new Error("CORS not supported")}$.forEach(config.headers,(function(v,k){x.setRequestHeader(k,v)}));var response=function(){var fn=x.status==200?onLoad:onError;if(fn)fn({text:x.responseText,status:x.status,statusText:x.statusText})};x.addEventListener("load",response);x.addEventListener("error",response);x.send(config.data)};function Cookie(){var data={};$.forEach(document.cookie.split(/; +/),(function(kv){if(kv!=""){kv=kv.split("=");data[kv[0]]=decodeURIComponent(kv[1])}}));this.get=function(key){return data[key]||undefined};this.set=function(key,value,expires){var x=key+"=";if(value!=undefined){data[key]=value;x+=encodeURIComponent(value)}else{delete data[key];expires=-1}if(expires!=undefined)x+="; expires="+new Date(expires==-1?946684801e3:Date.now()+Math.round(expires*864e5)).toUTCString();document.cookie=x};this.getObject=function(key){var x=this.get(key);if(x)x=JSON.parse(x);return x};this.setObject=function(key,value){if(value)value=JSON.stringify(value);this.set(key,value)}}$.cookie=new Cookie;var msgFloat=null;$.msg=function(config){if(!msgFloat){msgFloat=$.element("div").addClass("alert-float");$.base.addChild(msgFloat)}var c,t,x=$.element("div").addClass("alert alert-"+(config.type||"info")),d=config.delay||3;if(config.title)x.addChild($.element("strong").addClass("alert-title").setText(config.title));if(config.text)x.addChild(c=$.element("p").addClass("alert-text").setText(config.text.format(d)));x.button=function(text,fn){x.addChild($.element().addClass("row").addChild($.element.button(text).addClass("col-4").on("click",(function(event){fn.call(x,event)}))));return x};x.remove=function(){if(t)clearInterval(t);Element.prototype.remove.call(this);x.emit("remove")};if(d>0){t=setInterval((function(){d--;if(!d)x.remove();else if(c)c.setText(config.text.format(d))}),1e3)}msgFloat.addChild(x);return x};$.err=function(config){config.type="error";return $.msg(config)};var modalStack=[];$.modal=function(){if(!modalStack.length){var backdrop=$.element("div").addClass("modal-backdrop");$.base.addChild(backdrop);$.base.addClass("modal-open");modalStack.push(backdrop)}var m=$.element("div").addClass("modal").on("click",(function(event){event.stopPropagation()}));var w=$.element("div").addClass("modal-wrap").addChild(m);if(modalStack.length>1)modalStack[modalStack.length-1].addClass("hide");modalStack.push(w);$.base.addChild(w);var remove=function(e){modalStack.splice(modalStack.length-1,1);var n=modalStack[modalStack.length-1];if(modalStack.length>1){n.removeClass("hide")}else{$.base.removeClass("modal-open");n.remove();modalStack.splice(0,1)}m.parentNode.remove();m.emit(e)};m.set=function(k,v){if(v==undefined)delete m[k];else m[k]=v;return m};m.remove=function(){remove("remove")};m.submit=function(){remove("submit")};return m};Element.prototype.bindScope=function(render,data){var self=this,args=Array.prototype.slice.call(arguments,2);if(!$.isObject(data)){console.warn("bindScope() argument #2 object expected");data={}}if(self.scope)self.scope.destroy();var dataErrors={};var dataValidate=function(){var e=this,r=false;$.forEach(e.$validate,(function(fn){if(!fn.call(e,e.value))r=true}));if(e.$error!=r){e.setError(r);e.$error=r;if(r)dataErrors[e.dataset.bind]=true;else delete dataErrors[e.dataset.bind]}};var setDataBind=function(e,t){var key=e.dataset.bind;if(!self.scope.map[key])self.scope.map[key]=[];self.scope.map[key].push(e);e.addAttr("name",e.dataset.bind);switch(t){case"text":case"password":case"textarea":case"number":e.bindEvent="input";e.bindSetValue=function(val){this.value=val!==undefined?val:""};e.bindGetValue=function(){return this.value!==""?this.value:undefined};break;case"hidden":e.bindSetValue=function(val){this.value=val!==undefined?val:""};break;case"select":e.bindEvent="change";e.bindSetValue=function(val){this.value=val!==undefined?val:""};e.bindGetValue=function(){return this.value!==""?this.value:undefined};break;case"checkbox":var x=function(v){var s=v.toString();if(!e.dataset.hasOwnProperty(s))return v;v=e.dataset[s];switch(v){case"true":return true;case"false":return false;case"undefined":return undefined;default:return v}};e.value_map={};e.value_map[true]=x(true);e.value_map[false]=x(false);e.bindEvent="change";e.bindSetValue=function(val){this.checked=this.value_map[true]===val};e.bindGetValue=function(){return this.value_map[this.checked]};break;case"radio":e.bindEvent="change";e.bindSetValue=function(val){this.checked=this.value===val};e.bindGetValue=function(){return this.value};break}e.bindSetValue(self.scope.get(key));if(e.bindEvent){e.on(e.bindEvent,(function(){var x=this.bindSetValue;this.bindSetValue=$.nop;self.scope.set(this.dataset.bind,this.bindGetValue());this.bindSetValue=x;if(this.$validate)dataValidate.call(this)}));var x=e.eventBind[e.bindEvent].list;if(x.length>1)x.move(x.length-1,0)}if(e.$validate)dataValidate.call(e)};var setDataValue=function(e){var key=e.dataset.bind;if(!self.scope.map[key])self.scope.map[key]=[];self.scope.map[key].push(e);e.scope=self;e.bindSetValue=function(val){this.textContent=val!==undefined?val:""};e.bindSetValue(self.scope.get(key))};var bind=function(){render.apply(self,args);var el=self.querySelectorAll("*[data-bind]");for(var i=0;i<el.length;++i){var e=el[i];switch(e.tagName){case"INPUT":setDataBind(e,e.type);break;case"TEXTAREA":setDataBind(e,"textarea");break;case"SELECT":setDataBind(e,"select");break;default:setDataValue(e);break}}};self.scope={};var scopeClear=function(){self.scope.map={};self.scope.events={};self.empty();dataErrors={}};self.scope.reset=function(scopeData){if(scopeData)data=scopeData;scopeClear();bind();self.emit("ready").off("ready")};self.scope.destroy=function(){self.emit("destroy").off("destroy");scopeClear();delete self.scope};self.scope.set=function(key,value){var dst=data;var k=key.match(/(\\\.|[^\.])+/g);while(k.length>1){var sk=k.shift().replace(/\\\./g,".");if(dst[sk]==undefined)dst[sk]={};dst=dst[sk]}k=k[0].replace(/\\\./g,".");if(value!=undefined){dst[k]=value}else{value="";delete dst[k]}$.forEach(self.scope.map[key],(function(e){e.bindSetValue(value);if(e.dataset.validate)validateAction.call(e)}))};self.scope.get=function(key){var src=data;var k=key.match(/(\\\.|[^\.])+/g);while(k.length>1){var sk=k.shift().replace(/\\\./g,".");src=src[sk];if(src==undefined)return undefined}k=k[0].replace(/\\\./g,".");return src[k]};self.scope.validate=function(){return!!$.form_check&&$.isObjectEmpty(dataErrors)};self.scope.serialize=function(){var x=$.clone(data);$.forEach(x,(function(v,k){if(k.charAt(0)=="$")delete x[k]}));return x};setTimeout((function(){self.scope.reset()}));return self};Element.prototype.dataOrder=function(fn,def){this.addAttr("data-order",def?1:0);this.$order=fn;return this};$.tableInit=function(table){var i,thead=table.querySelector("thead").firstChild,tbody=table.querySelector("tbody");table.$orderId=-1;var reorder=function(){var x=thead.cells[table.$orderId];var a=new Array;for(i=0;i<tbody.rows.length;++i)a[i]=tbody.rows[i];a.sort((function(a,b){return x.$order(a.cells[table.$orderId],b.cells[table.$orderId])*x.dataset.order}));while(a.length)tbody.appendChild(a.shift())};for(i=0;i<thead.cells.length;++i){var e=thead.cells[i];if(e.dataset.order!=undefined){if(table.$orderId==-1||e.dataset.order!=0)table.$orderId=i;e.$orderId=i;e.on("click",(function(){if(table.$orderId==this.$orderId){thead.cells[table.$orderId].dataset.order*=-1}else{thead.cells[table.$orderId].dataset.order=0;table.$orderId=this.$orderId;thead.cells[table.$orderId].dataset.order=1}reorder()}))}}reorder()};$.tableSortInsert=function(table,row){var thead=table.querySelector("thead").firstChild;var tbody=table.querySelector("tbody");var x=thead.cells[table.$orderId];var r=row.cells[table.$orderId];var l=0,m=tbody.rows.length;while(l<m){var mid=l+m>>>1;if(x.$order(tbody.rows[mid].cells[table.$orderId],r)*x.dataset.order<0)l=mid+1;else m=mid}if(l==tbody.rows.length)tbody.appendChild(row);else tbody.insertBefore(row,tbody.rows[l])};$.tableFilter=function(table,filter){for(var i=0;i<table.rows.length;++i){var row=table.rows[i];row.removeClass("hide");if(filter){var hide=true;for(var j=0;j<row.cells.length;++j){var cell=row.cells[j].innerHTML;if(cell.toLowerCase().indexOf(filter)!=-1){hide=false;break}}if(hide)row.addClass("hide")}}};var utf8={encode:function(string){string=string.replace(/\r\n/g,"\n");var utftext="";for(var i=0;i<string.length;i++){var c=string.charCodeAt(i);if(c<128){utftext+=String.fromCharCode(c)}else if(c>127&&c<2048){utftext+=String.fromCharCode(c>>6|192);utftext+=String.fromCharCode(c&63|128)}else{utftext+=String.fromCharCode(c>>12|224);utftext+=String.fromCharCode(c>>6&63|128);utftext+=String.fromCharCode(c&63|128)}}return utftext},decode:function(utftext){var string="";for(var i=0;i<utftext.length;){var c=utftext.charCodeAt(i);if(c<128){string+=String.fromCharCode(c);i++}else if(c>191&&c<224){string+=String.fromCharCode((c&31)<<6|utftext.charCodeAt(i+1)&63);i+=2}else{string+=String.fromCharCode((c&15)<<12|(utftext.charCodeAt(i+1)&63)<<6|utftext.charCodeAt(i+2)&63);i+=3}}return string}};$.base64Decode=function(text){return utf8.decode(atob(text))};$.base64Encode=function(text){return btoa(utf8.encode(text))}})();

app={scope:null,hosts:{},modules:[],menu:[],settings:[],themes:[{value:"",label:"Default: Light"},{value:"dark",label:"Dark"}]};monthMap=["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"];time2ddmmmyyyy=function(t){if(!t)return"";var d=new Date(t*1e3),dd=("0"+d.getDate()).slice(-2),dm=monthMap[d.getMonth()],dy=d.getFullYear();return dd+" "+dm+" "+dy};codepages=[{value:"",label:"Default: Latin (ISO 6937)"},{value:"1",label:"West European (ISO 8859-1)"},{value:"2",label:"East European  (ISO 8859-2)"},{value:"3",label:"South European (ISO 8859-3)"},{value:"4",label:"North European (ISO 8859-4)"},{value:"5",label:"Cyrillic (ISO 8859-5)"},{value:"6",label:"Arabic (ISO 8859-6)"},{value:"7",label:"Greek (ISO 8859-7)"},{value:"8",label:"Hebrew (ISO 8859-8)"},{value:"9",label:"Turkish (ISO 8859-9)"},{value:"10",label:"Nordic (ISO 8859-10)"},{value:"11",label:"Thai (ISO 8859-11)"},{value:"13",label:"Baltic (ISO 8859-13)"},{value:"15",label:"West European (ISO 8859-15)"},{value:"21",label:"UTF-8"}];dvbPolarization=[{value:""},{value:"V",label:"Vertical"},{value:"H",label:"Horizontal"},{value:"-",label:"----",disabled:true},{value:"R",label:"Right"},{value:"L",label:"Left"}];dvbFec=[{value:"",label:"Default: Auto"},{value:"NONE"},{value:"1/2"},{value:"2/3"},{value:"3/4"},{value:"4/5"},{value:"5/6"},{value:"6/7"},{value:"7/8"},{value:"8/9"},{value:"3/5"},{value:"9/10"}];dvbsModulation=[{value:"",label:"Default: not set"},{value:"QPSK"},{value:"PSK8",label:"8PSK"},{value:"QAM16",label:"16-QAM"}];dvbcModulation=[{value:"",label:"Default: not set"},{value:"QAM16",label:"16-QAM"},{value:"QAM32",label:"32-QAM"},{value:"QAM64",label:"64-QAM"},{value:"QAM128",label:"128-QAM"},{value:"QAM256",label:"256-QAM"}];validateId=function(value){if(value==undefined||value=="")return true;return/^[^\\\/&%\.+]*$/.test(value)};validatePort=function(value){if(value==undefined||value=="")return true;value=Number(value);return!isNaN(value)&&value>0&&value<=65535};validatePid=function(value){if(value==undefined||value=="")return true;value=Number(value);return!isNaN(value)&&value>20&&value<=8191};validatePnr=function(value){if(value==undefined||value=="")return true;value=Number(value);return!isNaN(value)&&value>0&&value<=65535};validateBiss=function(value){if(value==undefined||value=="")return true;return/^[0-9A-Fa-f]{16}$/.test(value)};validateUrl=function(value){if(value==undefined||value=="")return true;return!!parseUrl(value)};validateHex=function(value){if(value==undefined||value=="")return true;return value.length%2==0&&/^[0-9A-Fa-f]*$/.test(value)};Array.prototype.indexOfID=function(id){for(var i=0;i<this.length;++i){if(this[i].id==id)return i}return-1};Array.prototype.uniq=function(){var i=1;while(i<this.length){if(this[i-1]==this[i])this.splice(i,1);else++i}};Number.prototype.format=function(c,d,t){var n=this,c=isNaN(c=Math.abs(c))?2:c,d=d==undefined?" ":d,t=t==undefined?".":t,s=n<0?"-":"",i=parseInt(n=Math.abs(+n||0).toFixed(c))+"",j=(j=i.length)>3?j%3:0;return s+(j?i.substr(0,j)+d:"")+i.substr(j).replace(/(\d{3})(?=\d)/g,"$1"+d)+(c?t+Math.abs(n-i).toFixed(c).slice(2):"")};Number.prototype.toHex=function(p,u){var n=this,p=p==undefined?2:p,h=n.toString(16);if(u)h=h.toUpperCase();while(h.length<p)h="0"+h;return"0x"+h};function ip2num(x){if(!x)return 0;x=x.split(".");while(x.length!=4)x.splice(-1,0,0);return((+x[0]*256+ +x[1])*256+ +x[2])*256+ +x[3]}parseUrlFormat={};parseUrlFormat["udp"]=function(r){var b=r.addr.indexOf("/");if(b!=-1)r.addr=r.addr.substring(0,b);b=r.addr.indexOf("@");if(b!=-1){if(b>0)r.localaddr=r.addr.substring(0,b);r.addr=r.addr.substring(b+1)}b=r.addr.indexOf(":");if(b!=-1){r.port=Number(r.addr.substring(b+1));if(isNaN(r.port)||r.port<1||r.port>65535)return false;r.addr=r.addr.substring(0,b)}else{r.port=1234}if(!r.addr.length)return false;var a=r.addr.split(".");if(a.length>4)return false;while(a.length){var o=Number(a.shift());if(isNaN(o)||o<0||o>255)return false}return true};parseUrlFormat["rtp"]=parseUrlFormat["udp"];parseUrlFormat["http"]=function(r){var b=r.addr.indexOf("/");if(b!=-1){r.path=r.addr.substring(b);r.addr=r.addr.substring(0,b)}else{r.path="/"}b=r.addr.indexOf("@");if(b!=-1){if(b>0){var a=r.addr.substring(0,b).split(":");if(a.length==2){r.login=a[0];r.password=a[1]}}r.addr=r.addr.substring(b+1)}b=r.addr.indexOf(":");if(b!=-1){r.port=Number(r.addr.substring(b+1));if(isNaN(r.port)||r.port<1||r.port>65535)return false;r.host=r.addr.substring(0,b)}else{switch(r.format){case"http":r.port=80;break;case"https":r.port=443;break;case"rtsp":r.port=554;break;default:r.port=80;break}r.host=r.addr}delete r.addr;if(!r.host.length)return false;return true};parseUrlFormat["https"]=parseUrlFormat["http"];parseUrlFormat["rtsp"]=parseUrlFormat["http"];parseUrlFormat["np"]=parseUrlFormat["http"];parseUrlFormat["file"]=function(r){r.filename=r.addr;delete r.addr;return true};function parseUrl(url){if(!url)return null;var b=url.indexOf("://");if(b==-1||url.length<=b+3)return null;var r={};r.format=url.substring(0,b);url=url.substring(b+3);b=url.indexOf("#");if(b==-1){r.addr=url}else{r.addr=url.substring(0,b);url=url.substring(b+1).split("&");for(var i=0;i<url.length;++i){var o=url[i].split("="),k=o[0],v=o.length==2?o[1]:true;r[k]=v}}if(parseUrlFormat.hasOwnProperty(r.format)&&!parseUrlFormat[r.format](r))return null;return r}makeUrlFormat={};makeUrlFormat["udp"]=function(d){if(d.localaddr){d.addr=d.localaddr+"@"+d.addr;delete d.localaddr}if(d.port){d.addr+=":"+d.port;delete d.port}};makeUrlFormat["rtp"]=makeUrlFormat["udp"];makeUrlFormat["http"]=function(d){d.addr="";if(d.login){d.addr+=d.login;delete d.login;if(d.password){d.addr+=":"+d.password;delete d.password}d.addr+="@"}if(d.host){d.addr+=d.host!="0.0.0.0"?d.host:"0";delete d.host}if(d.port){d.addr+=":"+d.port;delete d.port}if(d.path){d.addr+=d.path;delete d.path}};makeUrlFormat["https"]=makeUrlFormat["http"];makeUrlFormat["rtsp"]=makeUrlFormat["http"];makeUrlFormat["np"]=makeUrlFormat["http"];makeUrlFormat["file"]=function(d){d.addr=d.filename;delete d.filename};function makeUrl(data){var d=$.clone(data);if(makeUrlFormat.hasOwnProperty(d.format))makeUrlFormat[d.format](d);var r=d.format+"://"+(d.addr||"");delete d.format;delete d.addr;var o=[];for(k in d){var v=d[k];if(!v){continue}else if(v===true){o.push(k)}else{o.push(k+"="+v)}}if(o.length)r+="#"+o.join("&");return r}app.renderInit=function(){var menu=$.element().addClass("main-menu fixed");var renderMenu=function(list){$.forEach(list,(function(item){menu.addChild($.element.a(item.link||"#",item.label).addClass(item.hide?"hide":null).on("click",(function(event){if(item.click){event.preventDefault();item.click(event)}})))}))};var renderSearch=function(){var s=!app.module.search?$.element():$.element.input("Search").addAttr("autocomplete","off").dataBind("search").on("input",(function(){app.search(this.value)})).on("keydown",(function(event){if(event.keyCode==27){this.value="";app.search("")}}));s.addClass("search");menu.addChild(s)};var renderHelp=function(){menu.addChild($.element.a("#","Help").on("click",(function(event){event.preventDefault();var modal=$.modal();var doc=$.element();modal.addChild(doc,$.element().addClass("text-center").addChild($.element.button("Ok").on("click",(function(event){event.preventDefault();modal.remove()}))))})))};renderMenu(app.menu);renderSearch();renderMenu(app.module.menu);$.body.addChild(menu);var content=$.element().addClass("main-content");$.body.addChild(content);return content};function Form(scope,parent){var form=$.element("form").addClass("form").addAttr("novalidate").addAttr("autocomplete","off").on("keydown",(function(event){if(event.which==13){switch(event.target.tagName){case"TEXTAREA":case"BUTTON":break;default:event.preventDefault();break}}}));this.node=form;this.scope=scope;if(parent)parent.addChild(form);this.loading=function(){var x=$.element().addClass("loading");for(var i=0;i<4;++i)x.addChild($.element().addClass("bullet"));form.addChild(x);return x};this.group=function(){var g=$.element().addClass("form-group");form.addChild(g);return g};this.header=function(title,action,click){var g=$.element().addClass("form-group-header").addChild(title||"");if(action&&click)g.addChild($.element.a("#",action).addClass("form-group-action").on("click",(function(event){event.preventDefault();click()})));form.addChild(g);return g};this.hr=function(){var x=$.element("hr");form.addChild(x);return x};this.tab=function(tabId,options){var g=this.group(),c=scope.get(tabId),l=$.element().addClass("tab");if(c==undefined)c=options[0].id;$.forEach(options,(function(v){l.addChild(c==v.id?$.element("span").setText(v.label):$.element.a(v.link||"#",v.label).on("click",(function(event){event.preventDefault();scope.set(tabId,v.id);scope.reset()})))}));g.addChild(l);return l};this.checkbox=function(title,bind){var g=this.group();var x=$.element.checkbox(title||"");x.checkbox.dataBind(bind).addAttr("data-false","undefined");x.checkbox.setDisabled=function(value){var f=!!value?"addAttr":"removeAttr";var a="disabled";x.checkbox[f](a);x[f](a);return x.checkbox};x.checkbox.setDanger=function(){x.addClass("danger");return x.checkbox};x.checkbox.awaiting=function(value){x.checkbox.setDisabled(value);var f=!!value?"addClass":"removeClass";x[f]("awaiting")};g.addChild(x);return x.checkbox};this.input=function(title,bind,placeholder){var x=$.element.input().dataBind(bind);if(placeholder)x.addAttr("placeholder",placeholder);var g=this.group(title).addAttr("data-label",title).addChild(x);x.on("mousewheel",(function(){this.blur()}));x.setRequired=function(){x.addAttr("required");x.addValidator((function(value){return value!=undefined&&value!=""}));return x};x.setError=function(value){if(value){g.addClass("error");x.addClass("error")}else{g.removeClass("error");x.removeClass("error")}};x.addButton=function(button){if(!x.buttonWrap){x.buttonWrap=$.element().addClass("button-wrap");g.addClass("io-wizard").addChild(x.buttonWrap)}x.buttonWrap.addChild(button);return x};return x};this.password=function(title,bind,placeholder){var x=this.input(title,"$"+bind,placeholder);var h=function(p){p=p||"";return new Array(p.length+1).join("*")};scope.set("$"+bind,h(scope.get(bind)));x.on("focus",(function(){scope.set("$"+bind,scope.get(bind)||"")})).on("blur",(function(){scope.set(bind,scope.get("$"+bind));scope.set("$"+bind,h(scope.get(bind)))}));return x};this.number=function(title,bind,placeholder){var x=this.input(title,bind,placeholder);x.addAttr("type","number");return x};this.choice=function(title,bind,options){var x=$.element.select(options).dataBind(bind);var g=this.group(title).addAttr("data-label",title).addChild(x);x.setRequired=function(){x.addAttr("required");x.addValidator((function(value){return value!=undefined&&value!=""}));return x};x.setError=function(value){if(value){g.addClass("error");x.addClass("error")}else{g.removeClass("error");x.removeClass("error")}};return x};this.hidden=function(bind){var x=$.element("input").addAttr("type","hidden").dataBind(bind);form.addChild(x);return x};this.submit=function(btn){var x=$.element().addClass("form-submit");form.addChild(x);return x}}function Host(hostConfig,callback){var self=this,host=hostConfig.host,token=null;if(hostConfig.port)host+=":"+hostConfig.port;if(hostConfig.user){token=hostConfig.user;if(hostConfig.pass)token+=":"+hostConfig.pass}self.name=hostConfig.name||host;self.sysinfo=null;self.config=null;self.makeUid=function(type){var arr=self.config[type];do{self.config.gid=self.config.gid+1;var r=self.config.gid.toString(36);if(!arr||arr.indexOfID(r)==-1)return r}while(true)};var ws,wsError;var connect=function(){ws=new WebSocket("ws://"+host+"/control/event/");ws.onopen=function(){ws.send(JSON.stringify({scope:"auth",auth:token}))};ws.onclose=function(){app.removeHost(hostConfig);if(host==location.host)$.body.scope.destroy();wsError=$.err({title:self.name,text:"Connection closed. Retrying in {0}...",delay:4}).on("remove",(function(){app.addHost(hostConfig,(function(){if(host==location.host)app.route()}))}))};ws.onmessage=function(event){var data;try{data=JSON.parse(event.data);if(!data.scope)throw"event scope not defined"}catch(e){console.error(e);return}app.emit(data.scope,{host:host,data:data})}};var headers={"Content-Type":"application/json"};self.request=function(data,onLoad,onError){$.http({method:"POST",url:"http://"+host+"/control/",data:JSON.stringify(data),headers:headers},(function(response){if(onLoad){response.data=JSON.parse(response.text);onLoad(response)}}),(function(response){if(onError){onError(response)}}))};self.restart=function(){self.request({cmd:"restart"})};self.upload=function(fn){self.request({cmd:"upload",config:self.config},(function(){if(fn)fn(true);self.restart()}),(function(){$.err({title:self.name,text:"Upload failed"});if(fn)fn(false)}))};var authForm=null;var authFormInit=function(){if(authForm){authForm.setDisabled(false);return}var u=$.element.input("Login");var p=$.element.input("Password","password");var r=$.element.checkbox("Remember me");var s=$.element.button("Sign In").addAttr("type","submit").addClass("submit");authForm=$.element("form").addClass("auth").addChild(u,p,r,s).on("submit",(function(event){event.preventDefault();authForm.setDisabled(true);app.login=u.value;token=u.value+":"+p.value;$.cookie.set("auth",r.checkbox.checked?token:undefined);p.value="";load()}));authForm.setDisabled=function(value){s.setDisabled(value)};$.body.addChild(authForm);u.focus()};var loadError=null;var load=function(){if(token)headers["Authorization"]="Basic "+$.base64Encode(token);if(loadError){loadError.remove();loadError=null}self.request({cmd:"status"},(function(response){var data=response.data;if(authForm){authForm.remove();authForm=null}if(data.if_list){data.if_list.sort();var ifList=[];var ipList=data.ip_list||{};$.forEach(data.if_list,(function(i){var n={value:i,label:i};if(data.ip_list&&data.ip_list[i]){n.ip=data.ip_list[i];n.label+=": "+n.ip}ifList.push(n)}));data.if_list=ifList}if(data.observer){SettingsModule.hide=true;app.observer=true}self.sysinfo=data;self.request({cmd:"load"},(function(response){var data=response.data;self.config=$.isObject(data)?data:{};if(!self.config.gid)self.config.gid=466560;connect();callback.call(self)}),(function(){loadError=$.err({title:"Configuration file has wrong format",delay:-1})}))}),(function(response){if(response.status==403){var text="Authentication failed.";if(location.host==host){authFormInit();if(!token)return}else{var idx=app.hosts[location.host].config.servers.indexOf(hostConfig);text+='<br/><a href="#/settings-servers/'+idx+'">Check streamer configuration</a>'}loadError=$.err({title:self.name,text:text,delay:-1})}else{loadError=$.err({title:self.name,text:"Connection failed. Retrying in {0}...",delay:10}).on("remove",(function(){loadError=null;if(load)load()}))}}))};load();self.destroy=function(){load=undefined;if(loadError){loadError.remove();loadError=null}if(wsError){wsError.remove();wsError=null}if(ws){ws.onclose=null;if(ws.readyState==WebSocket.OPEN)ws.close();ws=null}}}app.addHost=function(config,fn){var host=config.host;if(config.port)host+=":"+config.port;app.hosts[host]=new Host(config,(function(){if(fn)fn.call(this);$.forEach(app.modules,(function(m){if(m.addHost)m.addHost(host)}))}))};app.removeHost=function(config){var host=config.host;if(config.port)host+=":"+config.port;if(app.hosts[host]){$.forEach(app.modules,(function(m){if(m.removeHost)m.removeHost(host)}));app.hosts[host].destroy();delete app.hosts[host]}};app.selectHost=function(link,id){id=id?"/"+id:"";if(Object.keys(app.hosts).length==1){$.route(link+"/"+location.host+id);return}var modal=$.modal().addClass("main-menu");$.forEach(app.hosts,(function(appHost,hostId){if(appHost.config){modal.addChild($.element.a(link+"/"+hostId+id,appHost.name).on("click",(function(){modal.remove()})))}}));modal.addChild($.element("hr"));modal.addChild($.element.a("#","Cancel").addClass("text-center").on("click",(function(event){event.preventDefault();modal.remove()})))};function Scan(hostId,config,fn){var scanId=null,scanTimer=null,tsid=-1,scanData=[],psiCache=[],appHost=app.hosts[hostId],changed=false;var getScanItem=function(pnr){var i,x;for(i=0;i<scanData.length;++i){x=scanData[i];if(x.pnr==pnr)return x}x={pnr:pnr,name:"Unknown"};scanData.insertSorted(x,(function(a,b){return a.pnr-b.pnr}));return x};var scanCheckPsi=function(psi){if(tsid==-1&&psi.psi!="pat"){psiCache.push(psi)}else if(scanCheckPsi[psi.psi]!=undefined){scanCheckPsi[psi.psi](psi)}};scanCheckPsi.pat=function(pat){tsid=pat.tsid;$.forEach(psiCache,scanCheckPsi);psiCache=[];changed=true};scanCheckPsi.nit=function(nit){var data=getScanItem(0);$.forEach(nit.descriptors,(function(d){if(d.type_id==64)data.name=d.network_name}));$.forEach(nit.streams,(function(s){if(s.tsid!=tsid)return;$.forEach(s.descriptors,(function(d){switch(d.type_id){case 67:{data.system=d;data.name=(d.s2?"DVB-S2":"DVB-S")+" : "+data.name;break}case 68:{data.system=d;data.name="DVB-C : "+data.name;break}case 90:{data.system=d;data.name="DVB-T : "+data.name;break}}}))}));changed=true};scanCheckPsi.pmt=function(pmt){var data=getScanItem(pmt.pnr);data.streams=[];data.cas=[];$.forEach(pmt.descriptors,(function(d){if(d.type_id==9)data.cas.push(d.caid.toHex(4))}));$.forEach(pmt.streams,(function(x){if(["VIDEO","AUDIO"].indexOf(x.type_name)==-1)return;var r=x.type_name.charAt(0)+"PID:"+x.pid;var e=null;var l=null;if(x.type_id==27)e="MPEG-4";$.forEach(x.descriptors,(function(d){switch(d.type_id){case 9:data.cas.push(d.caid.toHex(4));break;case 10:l=d.lang;break;case 106:e="AC-3";break}}));if(e)r=r+" "+e;if(l)r=r+" "+l;data.streams.push(r)}));data.cas.sort();data.cas.uniq();changed=true};scanCheckPsi.sdt=function(sdt){$.forEach(sdt.services,(function(x){var data=getScanItem(x.sid);$.forEach(x.descriptors,(function(d){if(d.type_id==72){data.name=d.service_name||"Unknown";data.provider=d.service_provider}}))}));changed=true};var scanCheck=function(){appHost.request({cmd:"scan-check",id:scanId},(function(response){var data=response.data;if(scanData){$.forEach(data.scan,scanCheckPsi);if(changed){fn(scanData);changed=false}}}))};appHost.request({cmd:"scan-init",scan:config},(function(response){if(!scanData)return;var data=response.data;scanId=data.id;scanTimer=setInterval(scanCheck,2e3)}),(function(){$.err({title:appHost.name,text:"Failed to scan stream"})}));this.destroy=function(){if(scanTimer)clearInterval(scanTimer),scanTimer=null;if(scanData)scanData=null;if(scanId){appHost.request({cmd:"scan-kill",id:scanId},(function(){scanId=null}))}}}Element.prototype.dataRender=function(fn){console.warn("Deprecated method: dataRender()");this.dataset.render=fn;return this};Element.prototype._bindScope=Element.prototype.bindScope;Element.prototype.bindScope=function(){var args=arguments,self=this;if(args.length==1){console.warn("Deprecated method: bindScope()");var render,scope=args[0];if(self==$.body){if(!window.renderContent)throw"render target not found";render=function(){var object=app.renderInit();window.renderContent.call(self,object)}}else{var dst=window,el=self.querySelector("*[data-render]"),key=el.dataset.render.split(".");while(key.length>0&&!!dst)dst=dst[key.shift()];if(!dst)throw"render callback not found ["+el.dataset.render+"]";render=function(){dst.call(self,self)}}args=[render,scope].concat(Array.prototype.slice.call(args,1))}Element.prototype._bindScope.apply(self,args);return self};app.setTheme=function(value){var e=$.html;if(e.theme)e.removeClass(e.theme);if(value)e.addClass(value);e.theme=value;$.cookie.set("theme",value,7)};app.getTheme=function(){return $.cookie.get("theme")};app.menu.push=function(item){this.insertSorted(item,(function(a,b){var ao=a.order!=undefined?a.order:1e3;var bo=b.order!=undefined?b.order:1e3;return ao-bo}))};app.settings.push=function(item){this.insertSorted(item,(function(a,b){var ao=a.order!=undefined?a.order:1e3;var bo=b.order!=undefined?b.order:1e3;return ao>bo?1:-1}))};app.route=function(){var route=location.hash.match(/^(#\/[a-z-]*)\/*/);app.module=null;if(route)for(var i=0;i<app.modules.length;++i){app.module=app.modules[i];if(app.module.link==route[1])break;app.module=null}if(!app.module){$.route(app.modules[0].link)}else{if(app.search)app.search("");app.search=$.nop;app.module.init()}};app.eventBind={};app.on=function(event,fn){var eb=app.eventBind[event];if(!eb)app.eventBind[event]=eb=[];eb.push(fn)};app.off=function(event,fn){var eb=app.eventBind[event];if(eb){eb.splice(eb.indexOf(fn),1);if(!eb.length)delete app.eventBind[event]}};app.emit=function(event){var eb=app.eventBind[event];if(eb)for(var i=0;i<eb.length;i++)eb[i].apply(this,Array.prototype.slice.call(arguments,1))};app.scope={on:function(event,fn){console.log("Deprecated Web-API for event:"+event);app.on(event,fn)},off:function(event,fn){app.off(event,fn)}};$.route=function(l){location.href=l};$((function(){window.addEventListener("hashchange",app.route);var s=$.element("style").addAttr("type","text/css").setHtml(".card { width: 210px; }");$.head.addChild(s);var cardCss;for(var i=0;i<s.sheet.cssRules.length;++i){var r=s.sheet.cssRules[i];if(r.selectorText==".card"){cardCss=r;break}}var resize=function(){var w=$.base.clientWidth-20,b=w>390?210:160;if(w>b){var c=Math.round(w/b),m=$.base.clientWidth>768?6:2,m=m*c-m,l=Math.floor((w-m)/c);cardCss.style.width=""+l+"px"}else{cardCss.style.width="100%"}};window.addEventListener("resize",resize);resize();app.setTheme(app.getTheme());var token=$.cookie.get("auth");if(token)app.login=token.split(":")[0];app.addHost({host:location.hostname,port:location.port,user:token},(function(){$.forEach(app.modules,(function(m){if(m.run)m.run()}));app.route();$.forEach(this.config.servers,(function(s){if(s.type=="streamer"&&s.enable!=false)app.addHost(s)}))}))}));

(function(){"use strict";window.MainModule={label:"Dashboard",link:"#/",order:0,search:true,node:$.element(),dvbDevList:{},adapters:{},streams:{},menu:[{label:"New Adapter",hide:app.observer==true,click:function(){app.selectHost(AdaptersModule.link,"-")}},{label:"New Stream",hide:app.observer==true,click:function(){app.selectHost(StreamsModule.link,"-")}},{label:"View",click:function(){$.modal().bindScope(renderModalView,$.cookie.getObject("streams-view")||{})}}]};var view=$.cookie.getObject("streams-view")||{};var renderModalView=function(){var modal=this,form=new Form(modal.scope,modal),masterHost=app.hosts[location.host];var arrangeList=[{value:"",label:"Default: None"},{value:"-",label:"---",disabled:true},{value:"$type",label:"Type"},{value:"$host",label:"Servers"},{value:"$dvb",label:"Adapter"},{value:"-",label:"---",disabled:true}];$.forEach(masterHost.config.categories,(function(c){arrangeList.push({value:c.name})}));form.choice("Arrange By","arrange",arrangeList);form.checkbox("Hide disabled items","hide_disabled");form.checkbox("Hide inactive items","hide_inactive");form.checkbox("Hide items without error","hide_wo_error");var orderList=[{value:"",label:"Default: Type and Name"},{value:"lcn",label:"LCN"}];form.choice("Order By","order",orderList);form.hr();var btnOk=$.element.button("Ok").addClass("submit").on("click",(function(){modal.remove();view=modal.scope.serialize();$.cookie.setObject("streams-view",$.isObjectEmpty(view)?undefined:view);$.forEach(MainModule.adapters,(function(node){node.remove()}));$.forEach(MainModule.streams,(function(node){node.remove()}));$.forEach(app.hosts,(function(appHost,host){$.forEach(appHost.config.dvb_tune,(function(c){renderAdapter(host,c)}));$.forEach(appHost.config.make_stream,(function(c){renderStream(host,c)}))}))}));form.submit().addChild(btnOk)};MainModule.run=function(){app.on("set-adapter",(function(event){var host=event.host,data=event.data,appHost=app.hosts[host];if(data.gid)appHost.config.gid=data.gid;var al=appHost.config.dvb_tune;if(data.adapter.remove){var idx=!al?-1:al.indexOfID(data.adapter.id);if(!data.adapter.up)$.msg({title:'Adapter "{0}" removed'.format(al[idx].name)});MainModule.adapters[host+"/"+data.adapter.id].remove();al.splice(idx,1);if(!al.length)delete appHost.config.dvb_tune}else{if(!al)appHost.config.dvb_tune=al=[];al.push(data.adapter);renderAdapter(host,data.adapter);$.msg({title:'Adapter "{0}" saved'.format(data.adapter.name)})}}));app.on("adapter_event",(function(event){var data=event.data,node=MainModule.adapters[event.host+"/"+data.dvb_id];if(!node)return;data.ber_value=data.ber>99?"99+":data.ber;data.unc_value=data.unc>99?"99+":data.unc;if(node.$config.raw_signal){data.svalue=data.signal;data.qvalue=Number(data.snr/10).format(1)}else{data.svalue=Math.floor(data.signal*99/65535);data.qvalue=Math.floor(data.snr*99/65535)}node.setStatus(data)}));app.on("set-stream",(function(event){var host=event.host,appHost=app.hosts[host],data=event.data;if(data.gid)appHost.config.gid=data.gid;var sl=appHost.config.make_stream;if(data.stream.remove){var idx=!sl?-1:sl.indexOfID(data.stream.id);if(!data.stream.up)$.msg({title:'Stream "{0}" removed'.format(sl[idx].name)});MainModule.streams[host+"/"+data.stream.id].remove();sl.splice(idx,1);if(!sl.length)delete appHost.config.make_stream}else{if(!sl)appHost.config.make_stream=sl=[];sl.push(data.stream);renderStream(host,data.stream);$.msg({title:'Stream "{0}" saved'.format(data.stream.name)})}}));app.on("stream_event",(function(event){var data=event.data,node=MainModule.streams[event.host+"/"+data.channel_id];if(!node)return;node.lastEvent=new Date;node.setStatus(data)}));app.on("stream_image",(function(event){var host=event.host,data=event.data,node=MainModule.streams[host+"/"+data.channel_id];if(!node)return;var i=new Image;i.onload=function(){node.$image.setStyle("background-image","url('"+i.src+"')").setStyle("display","block")};i.src=data.src}));MainModule.refresh=setInterval((function(){var ct=new Date;$.forEach(MainModule.streams,(function(i){if(i.$config.type=="spts"&&i.lastEvent&&ct-i.lastEvent>2e3)i.reset()}))}),1e3)};MainModule.dvbLevel=function(t,r){var x=$.element("div").addClass("dvb-status row monospace");var x1=$.element("div").addClass("text").setText(t);if(r){var x2=$.element("div").addClass("text col-expand");x.addChild(x1,x2);if(t=="S")x.setLevel=function(v){x2.setText("-"+v+" dBm")};else x.setLevel=function(v){x2.setText(v+" dB")}}else{var x2=$.element("div").addClass("progress signal-level col-expand");var b=$.element("div").addClass("progress-level");var o=$.element("div").addClass("progress-overlay");x2.addChild(b,o);var x3=$.element("div").addClass("text");x.addChild(x1,x2,x3);x.setLevel=function(v){v=v+"%";b.style.width=v;x3.setText((" "+v).slice(-3))}}x.setLevel(0);return x};var cmpStack=function(a,b){return b.$name?a.$name.localeCompare(b.$name):1};var cmpItems={};cmpItems["default"]=function(a,b){if(a.$type!==b.$type)return a.$type==="adapter"?-1:1;if(a.$config.enable===b.$config.enable)return a.$config.name.localeCompare(b.$config.name);return a.$config.enable===true?-1:1};cmpItems["lcn"]=function(a,b){if(a.$type!==b.$type)return a.$type==="adapter"?-1:1;if(a.$config.lcn===undefined)return 1;if(b.$config.lcn===undefined)return-1;return a.$config.lcn-b.$config.lcn};var renderStack=function(name){var node=$.forEach(MainModule.node.childNodes,(function(i){if(i.$name==name)return i}));if(!node){node=$.element().addClass("card-stack");node.$name=name;if(name)node.addAttr("data-header",name);if(view.hide_disabled)node.addClass("card-hide-disabled");if(view.hide_inactive)node.addClass("card-hide-inactive");if(view.hide_wo_error)node.addClass("card-hide-wo-error");MainModule.node.insertSorted(node,cmpStack)}return node};var getItemGroup=function(item){switch(view.arrange){case undefined:return"";case"$type":return item.$type+"s";case"$host":return app.hosts[item.$host].name;case"$dvb":return item.$type=="adapter"?item.$config.name:$.forEach(item.$config.input,(function(i){i=i.match(/^dvb:\/\/([^#]*)/);if(i){i=item.$host+"/"+i[1];i=MainModule.adapters[i];if(i)return i.$config.name}}))||"";default:return(item.$config.groups||{})[view.arrange]||""}};var renderAdapter=function(host,config){var node=$.element("a").addClass("card").addAttr("href","#/adapter/"+host+"/"+encodeURIComponent(config.id));node.$type="adapter";node.$host=host;node.$config=config;node.$lock=0;node.addClass("card-"+config.enable.toString()+"-"+node.$lock);node.addChild($.element("div").addClass("card-name").setText(config.name).addAttr("title",config.name));node.addChild($.element("select").addClass("card-action button icon icon-more small").addChild($.element("option"),$.element("option").setValue("-3").setText("Restart"),$.element("option").setValue("-1").setText(config.enable==false?"Enable":"Disable"),$.element("option").setValue("-2").setText("Remove")).on("change",(function(){var appHost=app.hosts[host];var restartAdapter=function(){appHost.request({cmd:"restart-adapter",id:config.id},$.nop,(function(){$.err({title:"Failed to restart adapter"})}))};var setAdapter=function(a){appHost.request({cmd:"set-adapter",id:config.id,adapter:a},$.nop,(function(){$.err({title:"Failed to save adapter"})}))};switch(this.value){case"-3":{restartAdapter();break}case"-2":{if(confirm('Remove adapter "{0}" and all related streams?'.format(config.name)))setAdapter({remove:true});break}case"-1":{var a=$.clone(config);a.enable=!a.enable;setAdapter(a);break}}this.value=""})).on("click",(function(event){event.preventDefault()})));var sS=MainModule.dvbLevel("S",config.raw_signal).addClass("card-status");var sQ=MainModule.dvbLevel("Q",config.raw_signal).addClass("card-status");node.addChild(sS,sQ);var eT=$.element("span").addClass("text");node.addChild($.element("div").addClass("card-status monospace").addChild(eT));node.setStatus=function(s){sS.setLevel(s.svalue);sQ.setLevel(s.qvalue);eT.setText("ber:"+s.ber_value+" unc:"+s.unc_value);var l=s.status&16?2:0;if(l!=node.$lock){node.removeClass("card-true-"+node.$lock);node.$lock=l;node.addClass("card-true-"+node.$lock)}};node.setStatus({lock:false,svalue:0,qvalue:0,ber_value:0,unc_value:0});node.$stack=renderStack(getItemGroup(node));node.$stack.insertSorted(node,!view.order?cmpItems["default"]:cmpItems[view.order]);node.remove=function(){var stack=this.$stack;delete this.$stack;Element.prototype.remove.call(this);delete MainModule.adapters[host+"/"+config.id];if(!stack.childNodes.length)stack.remove()};MainModule.adapters[host+"/"+config.id]=node};var renderStream=function(host,config){var node=$.element("a").addClass("card","stream-"+(config.type||"unknown")).addAttr("href","#/stream/"+host+"/"+encodeURIComponent(config.id));var onairDef=0;if(config.enable){if(config.type=="mpts"){onairDef=2}else if(config.http_keep_active!=-1){node.$activeInputId=-1;onairDef=3;$.forEach(config.output,(function(o){if(!o.match(/^http:\/\//)){onairDef=0;return true}}))}}node.$onair=onairDef;node.addClass("card-"+config.enable.toString()+"-"+node.$onair);node.$type="stream";node.$host=host;node.$config=config;node.$input=[];node.$total=null;node.addChild($.element("div").addClass("card-name").setText(config.name).addAttr("title",config.name||"-"));node.addChild($.element.select([{value:""},{value:3,label:"Toggle Input",hide:!config.enable||config.backup_type!="passive"&&config.backup_type!="disable"},{value:1,label:!config.enable?"Enable":"Disable"},{value:2,label:"Remove"}]).removeClass("input").addClass("card-action button icon icon-more small").on("change",(function(){var appHost=app.hosts[host],req={id:config.id};switch(this.value){case"1":{req.cmd="set-stream";req.stream=$.clone(config);req.stream.enable=!config.enable;break}case"2":{if(!confirm('Remove stream "'+config.name+'"?')){this.value="";return}req.cmd="set-stream";req.stream={remove:true};break}case"3":{req.cmd="set-stream-input";break}}appHost.request(req,$.nop,(function(){$.err({title:"Failed to save stream"})}));this.value=""})).on("click",(function(event){event.preventDefault()})));node.$image=$.element("div").addClass("card-image");node.addChild(node.$image);$.forEach(config.input,(function(v){var inode=$.element("span").addClass("text").setText("Inactive");var iwrap=$.element("div").addClass("card-status").addAttr("title",v).addChild(inode);node.addChild(iwrap);node.$input.push(inode)}));if(config.type=="mpts"){var tc=$.element("span").addClass("text").setText("Inactive");var tr=$.element("div").addClass("card-status card-footer").addChild(tc);node.addChild(tr);node.$total=tc;var sm={};$.forEach(config.sdt,(function(v){sm[v.pnr]=v.name}));$.forEach(config.input,(function(v,i){var ac=parseUrl(v);var pnr=ac.set_pnr||ac.pnr||0;if(pnr!=0){var n=sm[pnr];if(n)node.$input[i].addAttr("data-name","["+n+"]")}}))}node.$stack=renderStack(getItemGroup(node));node.$stack.insertSorted(node,!view.order?cmpItems["default"]:cmpItems[view.order]);node.reset=function(){$.forEach(node.$input,(function(i){i.className="text";i.setText("Inactive");delete i.$onair}));node.removeClass("card-true-"+node.$onair);node.$onair=3;node.addClass("card-true-"+node.$onair)};node.setStatus=function(s){var c="text",t=""+s.bitrate+"Kbit/s",inode=!!s.input_id?node.$input[s.input_id-1]:node.$total;if(s.onair){c+=" onair"}else if(s.scrambled){t+=" Scrambled";c+=" scrambled"}else{if(s.pes_error>0)t+=" PES:"+(s.pes_error>99?"99+":s.pes_error);c+=" error"}if(s.cc_error>0){t+=" CC:"+(s.cc_error>99?"99+":s.cc_error);c+=" cc"}inode.setText(t);if(inode.className!=c)inode.className=c;if(onairDef==2)return;if(inode.$onair==s.onair)return;inode.$onair=s.onair;var onair=onairDef;var inodeReset=function(i){var x=node.$input[i];if(x.$onair!=undefined){x.className="text";x.setText("Inactive");delete x.$onair}};if(config.backup_type=="passive"||config.backup_type=="disable"){if(s.onair){onair=2}for(var i=0,l=s.input_id-1;i<l;i++)inodeReset(i);for(var i=s.input_id,l=node.$input.length;i<l;i++)inodeReset(i)}else{if(s.onair){if(s.input_id==1)onair=2;else if(onair==0)onair=1;for(var i=s.input_id,l=node.$input.length;i<l;i++)inodeReset(i)}}if(node.$onair!=onair){node.removeClass("card-true-"+node.$onair);node.$onair=onair;node.addClass("card-true-"+node.$onair)}};node.remove=function(){var stack=this.$stack;delete this.$stack;Element.prototype.remove.call(this);delete MainModule.streams[host+"/"+config.id];if(!stack.childNodes.length)stack.remove()};MainModule.streams[host+"/"+config.id]=node};var initDvbDevList=function(hostId){var appHost=app.hosts[hostId],gAvail={group:"Available",items:[]},gTaken={group:"Taken",items:[]},gError={group:"Error",items:[]},dvbDevList=[{value:"",label:""},gAvail,gTaken,gError],dl=appHost.sysinfo.dvb_list;if(dl)dl.sort((function(a,b){if(a.error===b.error){if(a.busy===b.busy){if(a.adapter===b.adapter){if(a.device===b.device){return 0}return a.device<b.device?-1:1}return a.adapter<b.adapter?-1:1}return a.busy!=true?-1:1}return a.error==undefined?-1:1}));$.forEach(dl,(function(a,i){a.value=String(i);var d=a.adapter+"."+a.device+" : ";if(a.error){a.label=d+a.error;gError.items.push(a)}else if(a.busy){a.label=d+a.frontend;gTaken.items.push(a)}else{a.label=d+a.frontend;gAvail.items.push(a)}}));return dvbDevList};MainModule.addHost=function(host){var appHost=app.hosts[host];if(!appHost.config)return;MainModule.dvbDevList[host]=initDvbDevList(host);$.forEach(appHost.config.dvb_tune,(function(c){renderAdapter(host,c)}));$.forEach(appHost.config.make_stream,(function(c){renderStream(host,c)}))};MainModule.removeHost=function(host){$.forEach(MainModule.adapters,(function(node,i){if(node.$host==host)node.remove()}));$.forEach(MainModule.streams,(function(node,i){if(node.$host==host)node.remove()}));delete MainModule.dvbDevList[host]};var renderCards=function(){var self=this,object=app.renderInit();object.addChild(MainModule.node);app.search=function(value){value=value.toLowerCase();$.forEach(MainModule.node.childNodes,(function(stack){stack.removeClass("hide");var sh=true;$.forEach(stack.childNodes,(function(item){if(!item.removeClass)console.log(item);item.removeClass("hide");if(!!value&&item.$config.name.toLowerCase().indexOf(value)==-1)item.addClass("hide");else sh=false}));if(sh)stack.addClass("hide")}))};var info=app.hosts[location.host].sysinfo;var text="Astra "+info.version+" ["+info.commit+"]";$.body.addChild($.element().addClass("version monospace").setText(text));if(app.hosts[location.host].config.users[app.login].firstrun){var x=$.msg({text:"For security reasons, please, change default password!",delay:-1});x.addChild($.element().addClass("row").addChild($.element.button("Change Password").addClass("col-4").on("click",(function(event){x.remove();SettingsUsersModule.openConfig(app.login)}))))}};MainModule.init=function(){$.body.bindScope(renderCards,{})};app.modules.push(MainModule);app.menu.push(MainModule)})();

(function(){"use strict";window.StreamsModule={link:"#/stream"};StreamsModule.makeUrl=function(x){var a=x.format+"://";if(x.login){a+=x.login;if(x.password)a+=":"+x.password;a+="@"}if(x.host)a+=x.host;if(x.port)a+=":"+x.port;if(x.path)a+=x.path;return a};var renderSoftcam=function(form,appHost){var softcamList=[{value:"",label:"None"},{value:"",label:"---",disabled:true}];var softcamSortedList=[];$.forEach(appHost.config.softcam,(function(i){softcamSortedList.push({value:i.id,label:i.name})}));softcamSortedList.sort((function(a,b){return a.label.localeCompare(b.label)}));softcamList=softcamList.concat(softcamSortedList);form.choice("SoftCAM","cam",softcamList);form.input("BISS Key","biss").addValidator(validateBiss)};StreamsModule.renderModal_input=function(hostId){var modal=this,appHost=app.hosts[hostId],form=new Form(modal.scope,modal),format=modal.scope.get("format");form.checkbox("Enable","$enable").addAttr("data-false","false");form.choice("Input Type","format",[{value:"",label:""},{value:"dvb",label:"DVB"},{value:"http",label:"HTTP/HLS"},{value:"udp",label:"UDP"},{value:"rtp",label:"RTP"},{value:"rtsp",label:"RTSP"},{value:"file",label:"MPEG-TS Files"}]).setRequired().on("change",(function(){modal.scope.reset({$enable:modal.scope.get("$enable"),format:this.value})}));if(!!format)form.hr();if(format=="dvb"){var aList=[{value:"",label:""}];$.forEach(MainModule.adapters,(function(a){if(a.$host==hostId)aList.push({value:a.$config.id,label:a.$config.name})}));form.choice("Adapter","addr",aList).setRequired();form.checkbox("DVB-CI CAM","cam");form.number("T2-MI","t2mi");form.number("PNR","pnr","Program Number").addValidator(validatePnr);form.number("DD-CI CAM","ddci");renderSoftcam(form,appHost)}else if(format=="http"){var a=modal.scope.serialize();modal.scope.set("$addr",makeUrl(a).split("#")[0]);form.input("HTTP Address","$addr","http://...").setRequired().addValidator(validateUrl).on("input",(function(){$.forEach(parseUrl(this.value),(function(v,k){if(a[k]!=v){a[k]=v;modal.scope.set(k,v)}}))}));form.input("User-Agent","ua","Custom HTTP User-Agent. Default: Astra");if(modal.scope.get("$advanced")){form.hr();form.number("Buffer Time","buffer_time","Receiving buffer in seconds. Default: 2");form.number("Timeout","timeout","Connection timeout in seconds. Default: 10");form.checkbox("Use SCTP instead of TCP","sctp")}form.hr();form.number("PNR","pnr","Program Number").addValidator(validatePnr);form.number("DD-CI CAM","ddci");renderSoftcam(form,appHost);form.checkbox("Advanced Options","$advanced").on("change",(function(){modal.scope.reset()}))}else if(format=="udp"||format=="rtp"){var ifList=[{value:"",label:"Default: Use system routes"}];$.forEach(appHost.sysinfo.if_list,(function(v){ifList.push(v)}));form.choice("Local Interface","localaddr",ifList);form.input("Address","addr","Source Address").setRequired();form.number("Port","port","Source Port. Default: 1234").addValidator(validatePort);if(modal.scope.get("$advanced")){form.hr();form.number("Renew","renew","Refresh multicast group membership");form.number("Socket Size","socket_size","Redefine system socket size")}form.hr();form.number("PNR","pnr","Program Number").addValidator(validatePnr);form.number("DD-CI CAM","ddci");renderSoftcam(form,appHost);form.checkbox("Advanced Options","$advanced").on("change",(function(){modal.scope.reset()}))}else if(format=="rtsp"){var a=modal.scope.serialize();modal.scope.set("$addr",makeUrl(a).split("#")[0]);form.input("RTSP Address","$addr","rtsp://...").setRequired().addValidator(validateUrl).on("input",(function(){$.forEach(parseUrl(this.value),(function(v,k){if(a[k]!=v){a[k]=v;modal.scope.set(k,v)}}))}));form.checkbox("Interleaved mode. Send data over TCP","tcp")}else if(format=="file"){form.input("File","filename","Full path to the MPEG-TS file").setRequired();form.checkbox("Repeat File","loop");form.number("PNR","pnr","Program Number").addValidator(validatePnr)}form.hr();var btnOk=$.element.button("Ok").addClass("submit").on("click",(function(){modal.submit()}));var btnCancel=$.element.button("Cancel").on("click",(function(){modal.remove()}));form.checkbox("Remove","$remove").setDanger().on("change",(function(){if(this.checked)btnOk.removeClass("submit").addClass("danger");else btnOk.removeClass("danger").addClass("submit")}));form.submit().addChild(btnOk,btnCancel)};StreamsModule.renderModal_output=function(hostId){var modal=this,appHost=app.hosts[hostId],form=new Form(modal.scope,modal),format=modal.scope.get("format");form.checkbox("Enable","$enable").addAttr("data-false","false");form.choice("Output Type","format",[{value:"",label:""},{value:"http",label:"HTTP/HLS"},{value:"udp",label:"UDP"},{value:"rtp",label:"RTP"},{value:"resi",label:"RESI Modulator"},{value:"np",label:"NetworkPush"},{value:"file",label:"MPEG-TS Files"}]).setRequired().on("change",(function(){modal.scope.reset({$enable:modal.scope.get("$enable"),format:this.value})}));if(!!format)form.hr();if(format=="http"){if(!modal.scope.get("host"))modal.scope.set("host","0");var ifList=[{value:"0",label:"Default: Any Interface"}];$.forEach(appHost.sysinfo.if_list,(function(v){if(v.ip)ifList.push({value:v.ip,label:v.label})}));form.choice("Local Interface","host",ifList);form.number("Port","port","HTTP Server port").setRequired().addValidator(validatePort);form.input("Path","path","Unique path for stream").setRequired().addValidator((function(value){return value[0]=="/"}))}else if(format=="udp"||format=="rtp"){var ifList=[{value:"",label:"Default: Use system routes"}];$.forEach(appHost.sysinfo.if_list,(function(v){ifList.push(v)}));form.choice("Local Interface","localaddr",ifList);form.input("Address","addr","Destination Address").setRequired();form.number("Port","port","Destination Port. Default: 1234").addValidator(validatePort)}else if(format=="resi"){form.number("Adapter","adapter").setRequired();form.number("Device","device").setRequired();form.number("Frequency","frequency","114..858 MHz").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&value>113&&value<858&&(value-114)%8==0}));form.choice("Modulation","modulation",[{value:"",label:"Default: 64-QAM"},{value:"QAM16",label:"16-QAM"},{value:"QAM32",label:"32-QAM"},{value:"QAM128",label:"128-QAM"},{value:"QAM256",label:"256-QAM"}]);form.number("Symbol Rate","symbolrate","1000...7100").addValidator((function(value){if(value==undefined||value=="")return true;value=Number(value);return!isNaN(value)&&value>=1e3&&value<=7100}));form.number("Attenuator","attenuator","0...31").addValidator((function(value){if(value==undefined||value=="")return true;value=Number(value);return!isNaN(value)&&value>=0&&value<=31}))}else if(format=="np"){var a=modal.scope.serialize();modal.scope.set("$addr",makeUrl(a).split("#")[0]);form.input("Address","$addr","np://...").setRequired().addValidator(validateUrl).on("input",(function(){$.forEach(parseUrl(this.value),(function(v,k){if(a[k]!=v){a[k]=v;modal.scope.set(k,v)}}))}))}else if(format=="file"){form.input("File","filename","Full path to the MPEG-TS file").setRequired()}form.hr();var btnOk=$.element.button("Ok").addClass("submit").on("click",(function(){if(modal.scope.validate())modal.submit()}));var btnCancel=$.element.button("Cancel").on("click",(function(){modal.remove()}));form.checkbox("Remove","$remove").setDanger().on("change",(function(){if(this.checked)btnOk.removeClass("submit").addClass("danger");else btnOk.removeClass("danger").addClass("submit")}));form.submit().addChild(btnOk,btnCancel)};var ioList=function(self,form,ioType,hostId){form.hr();var lE=self.scope.get(ioType),lD=self.scope.get("_"+ioType);if(!lE){lE=[""];self.scope.set(ioType,lE)}if(!lD){lD=[];self.scope.set("_"+ioType,lD)}var renderIoList=function(ioEnable){var l=ioEnable?lE:lD;$.forEach(l,(function(v,i){var x=ioEnable?form.input("#"+(i+1),ioType+"."+i):form.input("","_"+ioType+"."+i);x.addValidator(validateUrl);x.on("blur",(function(){self.scope.set(x.dataset.bind,this.value.trim())}));x.on("keydown",(function(){if(event.keyCode==13){event.preventDefault();x.blur();var n;if(event.shiftKey){if(i==0){n=0;l.splice(0,0,"");self.scope.reset()}else{n=i-1}}else{n=i+1;if(n==l.length){l.push("");self.scope.reset()}}document.querySelector('[name="'+(ioEnable?"":"_")+ioType+"."+n+'"]').focus()}}));var s=$.element("select").addClass("button icon icon-move").on("change",(function(){if(this.value=="-2"){l.splice(i,1);if(!lE.length)lE.push("")}else if(this.value=="-1"){lD.push(lE.splice(i,1)[0]);if(!lE.length)lE.push("")}else{var _i=i;if(!ioEnable){_i=lE.length;lE.push(lD.splice(i,1)[0])}lE.move(_i,Number(this.value))}self.scope.reset()}));for(var j=0;j<lE.length;++j){s.addChild($.element("option").setValue(j.toString()).setText(j+1))}if(!ioEnable){s.addChild($.element("option").setValue(lE.length.toString()).setText(lE.length+1))}s.addChild($.element("option").setDisabled(true).setText("---"),$.element("option").setValue("-1").setText("Disable"),$.element("option").setValue("-2").setText("Remove"));s.value=ioEnable?i.toString():"-1";x.addButton(s);x.addButton($.element.button().addClass("icon icon-settings").on("click",(function(){var modalData=parseUrl(x.value)||{};modalData.$enable=ioEnable;$.modal().bindScope(StreamsModule["renderModal_"+ioType],modalData,hostId).on("submit",(function(){l.splice(i,1);if(!this.scope.get("$remove")){var lM=this.scope.get("$enable")?lE:lD;lM.push(makeUrl(this.scope.serialize()));if(lM==l)l.move(l.length-1,i)}if(!lE.length)lE.push("");var x=window.scrollTop;self.scope.reset();window.scrollTop=x}))})))}))};form.header(ioType+" list","new "+ioType,(function(){lE.push("");self.scope.reset();document.querySelector('[name="'+ioType+"."+(lE.length-1)+'"]').focus()}));renderIoList(true);if(!lD.length)return;form.header("Disabled "+ioType+"'s");renderIoList(false)};var tabGeneral=function(self,form,streamId,hostId){form.checkbox("Enable","enable").addAttr("data-false","false");form.input("Name","name").setRequired();var x=form.input("ID","id").addValidator(validateId);if(streamId!="-")x.setRequired();form.checkbox("Multi Program Stream","type").addAttr("data-false","spts").addAttr("data-true","mpts").on("change",(function(){self.scope.reset({$streamId:streamId,enable:self.scope.get("enable"),id:self.scope.get("id"),type:this.checked?"mpts":"spts",name:self.scope.get("name"),groups:self.scope.get("groups"),input:self.scope.get("input"),output:self.scope.get("output")})}))};var tabGroups=function(self,form){var masterHost=app.hosts[location.host];if(!masterHost.config.categories){form.group().addClass("text-center").setHtml('Create new group in <a href="#/settings-groups">Settings &gt; Groups</a>')}else $.forEach(masterHost.config.categories,(function(c){var gl=[{value:"",test:""}];$.forEach(c.groups,(function(g){gl.push({value:g.name,label:g.name})}));form.choice(c.name,"groups."+c.name.replace(/\./g,"\\."),gl)}))};var tabSpts=function(self,form,streamId,hostId){tabGeneral(self,form,streamId,hostId);form.checkbox("Start stream on demand","http_keep_active").addAttr("data-true","undefined").addAttr("data-false","-1").on("change",(function(){self.scope.reset()}));if(self.scope.get("http_keep_active")!=-1)form.number("Keep Active","http_keep_active","Delay before stop stream if no active connections. Default: 0 (turn off immediately)");form.number("Channel Number","lcn","Logical number to arrange playlist");ioList(self,form,"input",hostId);ioList(self,form,"output",hostId)};var tabSptsSdt=function(self,form,streamId,hostId){form.choice("Service Type","service_type",[{value:"",label:"Default: original service type"},{value:"1",label:"Video"},{value:"2",label:"Radio"},{value:"3",label:"Teletext"}]);form.input("Service Provider","service_provider");form.input("Service Name","service_name");form.choice("Codepage","textcode",codepages);form.hr();form.input("HbbTV URL","hbbtv_url");form.hr();var casChannelList=self.scope.get("cas_list");form.header("Conditional Access","New CAS",(function(){if(!casChannelList){casChannelList=[];self.scope.set("cas_list",casChannelList)}casChannelList.push({});self.scope.reset()}));var casGlobalList=[{value:"",label:"None"},{value:"",label:"---",disabled:true}];var casSortedList=[];$.forEach(app.hosts[hostId].config.cas,(function(i){casSortedList.push({value:i.id,label:i.name+" ["+i.super_cas_id+"]"})}));casSortedList.sort((function(a,b){return a.label.localeCompare(b.label)}));casGlobalList=casGlobalList.concat(casSortedList);$.forEach(casChannelList,(function(cas,x){if(x!=0)form.hr();form.choice("CAS #"+(x+1),"cas_list."+x+".cas_id",casGlobalList).setRequired().on("change",(function(){self.scope.reset()}));form.number("ECM PID","cas_list."+x+".ecm_pid").setRequired().addValidator(validatePid);form.input("ECM Private Data (hex)","cas_list."+x+".ecm_data").addValidator(validateHex).addAttr("maxlength","512");form.input("Access Criteria (hex)","cas_list."+x+".ac").addValidator(validateHex).addAttr("maxlength","512");form.group().addClass("text-right").addChild($.element.button("Remove CAS").addClass("link danger").on("click",(function(){casChannelList.splice(x,1);if(!casChannelList.length)self.scope.set("cas_list");self.scope.reset()})))}))};var tabSptsRemap=function(self,form){form.input("Map PID's","map","Example: pmt=100, video=101, audio=102, 1003=103");form.input("Filter PID's","filter~","Keep only defined pids. Example: 101, 102, 103");form.number("Change PNR","set_pnr").addValidator(validatePnr);form.number("Change TSID","set_tsid")};var tabSptsBackup=function(self,form){form.choice("Backup Type","backup_type",[{value:"",label:"Default: Active Backup"},{value:"stop",label:"Active Backup and Stop streaming if all inputs are inactive"},{value:"passive",label:"Passive Backup"},{value:"disable",label:"Disable"}]).on("change",(function(){self.scope.reset()}));var backup_type=self.scope.get("backup_type");if(backup_type!="disable"){form.number("Start Delay","backup_start_delay","Delay before start next input. Default: 0");if(backup_type!="passive"){form.number("Return Delay","backup_return_delay","Delay before return to previous input. Default: 0")}}};var tabSptsEpg=function(self,form){form.header("EPG Import");form.input("XMLTV Channel ID","xmltv_id");form.header("EPG Export");form.choice("Format","epg_export_format",[{value:"",label:"Default: XMLTV"},{value:"json",label:"JSON"}]);form.input("Destination","epg_export","Destination address: file:// or http://");var x=form.choice("Codepage","epg_export_codepage",codepages);x.firstChild.setText("Default: Auto")};var tabMpts=function(self,form,streamId,hostId){tabGeneral(self,form,streamId,hostId);form.input("Country","country","Country Code ISO 3166-1 alpha-3").setRequired().addValidator((function(value){return!!value&&value.length==3}));form.input("UTC Offset","offset","Offset time from UTC in the range between -720 minutes and +780 minutes");form.number("Network ID","network_id","Default: 1");form.input("Network Name","network_name");form.input("Provider Name","provider");form.choice("Codepage","textcode",codepages);form.number("TSID","tsid","Transport Stream ID. Default: 1").addAttr("maxlength","5");form.number("ONID","onid","Original Network ID. Default: 1");ioList(self,form,"input",hostId);ioList(self,form,"output",hostId)};var tabMptsSdt=function(self,form){var sl=self.scope.get("sdt");$.forEach(sl,(function(v,k){var s=$.element("select").addClass("button icon icon-move").setStyle("float","right").on("change",(function(){if(this.value=="-1"){sl.splice(k,1)}else{sl.move(k,Number(this.value))}self.scope.reset()}));for(var i=0;i<sl.length;i++){s.addChild($.element("option").setValue(i.toString()).setText(i+1))}s.addChild($.element("option").setDisabled(true).setText("---"),$.element("option").setValue("-1").setText("Remove"));s.value=k;form.header("Service #"+(k+1)).addChild(s);form.choice("Service Type","sdt."+k+".type",[{value:"1",label:"Video"},{value:"2",label:"Radio"},{value:"3",label:"Teletext"}]);form.input("Service Name","sdt."+k+".name").setRequired();form.number("PNR","sdt."+k+".pnr","Program Number").setRequired().addValidator(validatePnr);form.checkbox("Scrambled channel","sdt."+k+".ca").on("change",(function(){self.scope.reset()}));form.number("LCN","sdt."+k+".lcn","Logical Channel Number").addValidator((function(value){if(value==undefined||value=="")return true;value=Number(value);return!isNaN(value)&&value>=0&&value<1e3}));form.hr()}));form.header("New Service").addChild($.element.button().addClass("icon  icon-add").setStyle("float","right").on("click",(function(event){event.preventDefault();if(sl==undefined){sl=[];self.scope.set("sdt",sl)}sl.push({type:"1"});self.scope.reset();window.scrollTo(0,$.base.scrollHeight);document.querySelector('[name="sdt.'+(sl.length-1)+'.name"]').focus()})))};var tabMptsNit=function(self,form,streamId,hostId){form.choice("LCN Version","nit_actual.lcn_version",[{value:"",label:"Default: EACEM"},{value:"nordig-v1",label:"Nordig v1"}]);form.hr();form.choice("Delivery Type","nit_actual.type",[{value:"",label:"Default: not defined"},{value:"S",label:"DVB-S"},{value:"C",label:"DVB-C"}]).on("change",(function(){self.scope.set("nit_actual",{type:this.value});self.scope.set("nit_other",undefined);self.scope.reset()}));var sType=self.scope.get("nit_actual.type");if(!sType){}else if(sType=="S"){form.number("Frequency","nit_actual.frequency","950..13250 MHz").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&value>=950&&value<13250}));form.choice("Polarization","nit_actual.polarization",dvbPolarization).setRequired();form.number("Symbolrate","nit_actual.symbolrate","1000..50000 Kbaud").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&value>1e3&&value<5e4}));form.input("Orbital Position","nit_actual.position","Orbital Position (Example: 0.8W, 36.0E)").setRequired().addValidator((function(value){return true}));form.choice("FEC","nit_actual.fec",dvbFec);form.choice("Modulation","nit_actual.modulation",dvbsModulation)}else if(sType=="C"){form.number("Frequency","nit_actual.frequency","80..1000 MHz").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&value>=80&&value<1e3}));form.number("Symbolrate","nit_actual.symbolrate","1000..10000 Kbaud").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&value>1e3&&value<1e4}));form.choice("FEC","nit_actual.fec",dvbFec);form.choice("Modulation","nit_actual.modulation",dvbcModulation)}form.header("Network Search");var nitOther=self.scope.get("nit_other");if(!nitOther)nitOther=[],self.scope.set("nit_other",nitOther);var mptsList=[];$.forEach(app.hosts[hostId].config.make_stream,(function(c){if(c.id&&c.id!=streamId&&c.type=="mpts")mptsList.push({id:c.id,name:c.name,tsid:c.tsid})}));$.forEach(mptsList,(function(c){var x=form.checkbox(c.name,"").on("change",(function(){if(this.checked){nitOther.push(c.id)}else{var i=nitOther.indexOf(c.id);if(i!=-1)nitOther.splice(i,1)}}));x.checked=nitOther.indexOf(c.id)!=-1}))};var tabMptsAdvanced=function(self,form){form.number("SI Packets Interval","si_interval","Interval in milliseconds to send stream information. Default: 500");form.hr();form.checkbox("Pass NIT","pass_nit");form.checkbox("Pass SDT","pass_sdt");form.checkbox("Pass EIT","pass_eit");form.checkbox("Pass TDT","pass_tdt");form.hr();form.number("PAT version","pat_version");form.number("NIT version","nit_version");form.number("CAT version","cat_version");form.number("SDT version","sdt_version")};StreamsModule.tabs={spts:[{label:"General",id:"spts",render:tabSpts},{label:"Groups",id:"groups",render:tabGroups},{label:"Service",id:"sptssdt",render:tabSptsSdt},{label:"Remap",id:"remap",render:tabSptsRemap},{label:"Backup",id:"backup",render:tabSptsBackup},{label:"EPG",id:"epg",render:tabSptsEpg}],mpts:[{label:"General",id:"mpts",render:tabMpts},{label:"Groups",id:"groups",render:tabGroups},{label:"SDT",id:"mptssdt",render:tabMptsSdt},{label:"NIT",id:"nit",render:tabMptsNit},{label:"Advanced",id:"advanced",render:tabMptsAdvanced}]};StreamsModule.render=function(hostId){var x,self=this,object=app.renderInit(),streamId=self.scope.get("$streamId"),appHost=app.hosts[hostId],type=self.scope.get("type"),tabId=self.scope.get("$tab")||type,form=new Form(self.scope,object),tabRender;x=[];$.forEach(StreamsModule.tabs[type],(function(v){x.push(v);if(tabId==v.id)tabRender=v.render}));form.tab("$tab",x);tabRender(self,form,streamId,hostId);var serialize=function(){if(self.scope.get("$remove"))return{remove:true};var data=self.scope.serialize();if(data.id==undefined)data.id=appHost.makeUid("make_stream");var ioClean=function(obj){var a=data[obj];if(a==undefined)return;for(var i=0;i<a.length;){if(!a[i])a.splice(i,1);else++i}if(!a.length)delete data[obj]};ioClean("input");ioClean("_input");ioClean("output");ioClean("_output");if(!data.input)data.enable=false;if(data.nit_actual){if(!data.nit_actual.type){delete data.nit_actual;delete data.nit_other}else{$.forEach(data.nit_actual,(function(v,k){if(!v)delete data.nit_actual[k]}));if(data.nit_other){for(var i=0;i<data.nit_other.length;){if(data.nit_other[i])++i;else data.nit_other.splice(i,1)}if(!data.nit_other.length)delete data.nit_other}}}$.forEach(data.groups,(function(v,k){if(!v)delete data.groups[k]}));if($.isObjectEmpty(data.groups))delete data.groups;return data};form.hr();var btnApply=$.element.button("Apply").addClass(self.scope.get("$remove")?"danger":"submit").on("click",(function(){if(!self.scope.get("$remove")&&!self.scope.validate()){$.err({title:"Form has errors"});return}appHost.request({cmd:"set-stream",gid:appHost.config.gid,id:streamId,stream:serialize()},(function(){$.route(MainModule.link)}),(function(){$.err({title:"Failed to save stream"})}))}));var btnClone=$.element.button("Clone").on("click",(function(){var data=serialize();data.name=(data.name||"")+" (clone)";if(data.id!=undefined)delete data.id;StreamsModule.streamClone=data;app.selectHost(StreamsModule.link,"-")}));var btnHelp=$.element.button("Help").on("click",(function(){var m=$.modal().addClass("help").addChild($.element("iframe").addAttr("src","https://cesbo.com/wiki/embed/stream/"+tabId+"?embed")).addChild($.element("div").addClass("text-center").addChild($.element.button("ok").on("click",(function(){m.remove()}))))}));if(streamId!="-"){if(tabId==type){form.checkbox("Remove","$remove").setDanger().on("change",(function(){if(this.checked)btnApply.removeClass("submit").addClass("danger");else btnApply.removeClass("danger").addClass("submit")}))}}else{btnClone.setDisabled(true)}form.submit().addChild(btnApply,btnClone,btnHelp)};StreamsModule.init=function(){var scope;var x=location.hash.slice(StreamsModule.link.length+1).split("/"),sid=decodeURIComponent(x.pop()),host=x.pop();if(!host||!app.hosts[host]){$.route(MainModule.link);return}if(sid=="-"){if(StreamsModule.streamClone){scope=StreamsModule.streamClone;delete StreamsModule.streamClone}else{scope={enable:true,type:"spts"}}}else{var s=MainModule.streams[host+"/"+sid];if(!s){$.route(MainModule.link);return}scope=$.clone(s.$config)}scope.$streamId=sid;$.body.bindScope(StreamsModule.render,scope,host)};app.modules.push(StreamsModule)})();

(function(){"use strict";window.AdaptersModule={link:"#/adapter"};AdaptersModule.renderScan=function(host){var modal=this,appHost=app.hosts[host];var dvbStatus=function(){var i=function(t){return $.element("span").addAttr("title",t)};var x=$.element("div").addClass("info-status").addChild(i("SIGNAL"),i("CARRIER"),i("FEC"),i("SYNC"),i("LOCK"),i("BER"),i("UNC"),i("BITRATE"));var xl=x.childNodes;x.setStatus=function(s){xl[0].className=s.signal?"ok":"er";xl[1].className=s.carrier?"ok":"er";xl[2].className=s.viterbi?"ok":"er";xl[3].className=s.sync?"ok":"er";xl[4].className=s.lock?"ok":"er";xl[5].setText("BER:"+s.ber_value);xl[6].setText("UNC:"+s.unc_value);xl[7].setText(s.bitrate+" Kbit/s")};return x};var sI=dvbStatus();var sS=MainModule.dvbLevel("S",modal.scope.get("raw_signal"));var sQ=MainModule.dvbLevel("Q",modal.scope.get("raw_signal"));var form=new Form(modal.scope,modal);form.submit().addChild(sI,sS,sQ);modal.setStatus=function(s){sS.setLevel(s.svalue);sQ.setLevel(s.qvalue);sI.setStatus(s);modal.status=s};modal.setStatus(modal.status||{lock:false,svalue:0,qvalue:0,ber_value:0,unc_value:0,bitrate:0});form.hr();var renderNit=function(v){form.input("Provider").addAttr("readonly").setValue(v.name);var s=v.system||{};switch(s.type_id){case 67:{form.input("Position").addAttr("readonly").setValue(s.orbital_position);form.input("Frequency").addAttr("readonly").setValue(s.frequency);form.input("Polarization").addAttr("readonly").setValue(s.polarization);form.input("Symbolrate").addAttr("readonly").setValue(s.symbolrate);form.input("Modulation").addAttr("readonly").setValue(s.modulation);form.input("FEC").addAttr("readonly").setValue(s.fec);break}case 68:{form.input("Frequency").addAttr("readonly").setValue(s.frequency);form.input("Symbolrate").addAttr("readonly").setValue(s.symbolrate);form.input("Modulation").addAttr("readonly").setValue(s.modulation);form.input("FEC").addAttr("readonly").setValue(s.fec);break}case 90:{form.input("Frequency").addAttr("readonly").setValue(s.frequency);form.input("Bandwidth").addAttr("readonly").setValue(s.bandwidth);form.input("Guard Interval").addAttr("readonly").setValue(s.guard_interval);form.input("Transmit Mode").addAttr("readonly").setValue(s.transmitmode);form.input("Hierarchy").addAttr("readonly").setValue(s.hierarchy);form.input("Modulation").addAttr("readonly").setValue(s.modulation);break}}form.hr()};var renderItem=function(v,i){if(v.pnr==0){renderNit(v)}else{var text="PNR:"+v.pnr+"&nbsp;<strong>"+v.name+"</strong>";if(v.cas&&v.cas.length)text+="<br/>CAS:"+v.cas.join(",&nbsp;");var s=!!v.streams&&v.streams.length;if(s)text+="<br/>"+v.streams.join("<br/>");form.checkbox(text,"streams."+i+".select")}};var streams=modal.scope.get("streams");if(!streams||!streams.length){form.loading()}else{$.forEach(streams,renderItem);form.hr();form.checkbox("Set DVB-CI CAM for selected channels","cam");var softcamList=[{value:"",label:"None"},{value:"",label:"---",disabled:true}];var softcamSortedList=[];$.forEach(appHost.config.softcam,(function(i){softcamSortedList.push({value:i.id,label:i.name})}));softcamSortedList.sort((function(a,b){return a.label.localeCompare(b.label)}));softcamList=softcamList.concat(softcamSortedList);form.choice("Set SoftCAM for selected channels","cam",softcamList)}form.hr();var btnApply=$.element.button("Apply").addClass("submit").on("click",(function(){modal.submit()}));var btnCancel=$.element.button("Cancel").on("click",(function(){modal.remove()}));modal.addChild($.element("div").addClass("text-center").addChild(btnApply,btnCancel))};AdaptersModule.startScan=function(host,fn){var self=this,scan,modal,config=self.scope.serialize();config.format="dvb";if(config.id==undefined)config.id="-";var onScan=function(scanData){modal.scope.set("streams",$.clone(scanData));modal.scope.reset()};var onEvent=function(event){var data=event.data;if(event.host!=host||data.dvb_id!=config.id)return;var status={bitrate:data.bitrate,ber_value:data.ber>99?"99+":data.ber,unc_value:data.unc>99?"99+":data.unc,signal:(data.status&1)!==0,carrier:(data.status&2)!==0,viterbi:(data.status&4)!==0,sync:(data.status&8)!==0,lock:(data.status&16)!==0};if(config.raw_signal){status.svalue=data.signal;status.qvalue=Number(data.snr/10).format(1)}else{status.svalue=Math.floor(data.signal*99/65535);status.qvalue=Math.floor(data.snr*99/65535)}modal.setStatus(status)};var onModalRemove=function(){app.off("adapter_event",onEvent);scan.destroy();modal.scope.destroy()};modal=$.modal().addClass("dvb-scan").bindScope(AdaptersModule.renderScan,{raw_signal:config.raw_signal},host).on("remove",(function(){onModalRemove()})).on("submit",(function(){var streams=[],cam=modal.scope.get("cam");$.forEach(modal.scope.get("streams"),(function(s){if(s.select)streams.push({name:s.name,pnr:s.pnr})}));onModalRemove();if(streams.length)fn(streams,cam)}));app.on("adapter_event",onEvent);scan=new Scan(host,config,onScan)};AdaptersModule.renderPlsCals=function(){var modal=this,form=new Form(modal.scope,modal);form.choice("PLS Mode","mode",[{label:"Root",value:0},{label:"Gold",value:1}]).setRequired();form.number("PLS Code (0 - 262143)","code").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&value>=0&&value<=262143}));form.number("Stream (0 - 255)","id").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&value>=0&&value<=255}));form.hr();var btnOk=$.element.button("Ok").addClass("submit").on("click",(function(){if(modal.scope.validate())modal.submit()}));var btnCancel=$.element.button("Cancel").on("click",(function(){modal.remove()}));form.submit().addChild(btnOk,btnCancel)};AdaptersModule.render=function(host){var x,self=this,object=app.renderInit(),adapterId=self.scope.get("$adapterId"),appHost=app.hosts[host],tabId=self.scope.get("$tab")||"",dvbList=MainModule.dvbDevList[host],form=new Form(self.scope,object),adapterType=self.scope.get("type"),adapterTypeMap={S:"S",S2:"S",T:"T",T2:"T",C:"C","C/A":"C","C/B":"C","C/C":"C",ATSC:"ATSC",ISDBT:"ISDBT"},adapterTypeBase=adapterTypeMap[adapterType];switch(adapterTypeBase){case"S":form.tab("$tab",[{label:"General",id:""},{label:"LNB",id:"lnb"},{label:"DiSEqC",id:"diseqc"},{label:"Unicable",id:"unicable"},{label:"Advanced",id:"advanced"}]);break;case"T":case"C":case"ATSC":case"ISDBT":form.tab("$tab",[{label:"General",id:""},{label:"Advanced",id:"advanced"}]);break;default:form.tab("$tab",[{label:"General",id:""}]);break}if(tabId==""){form.checkbox("Enable","enable").addAttr("data-false","false");form.input("Name","name","Adapter Name").setRequired();x=form.input("ID","id").addValidator(validateId);if(adapterId!="-")x.setRequired();if(self.scope.get("$adapter")==undefined){var a=Number(self.scope.get("adapter"));var d=Number(self.scope.get("device")||0);var m=self.scope.get("mac");if(!isNaN(a)&&!isNaN(d)){self.scope.set("$adapter",$.forEach(appHost.sysinfo.dvb_list,(function(v,k){if(v.adapter==a&&v.device==d)return k})))}else if(m){self.scope.set("$adapter",$.forEach(appHost.sysinfo.dvb_list,(function(v,k){if(v.mac==m)return k})))}}form.choice("Adapter","$adapter",dvbList).setRequired().on("change",(function(){var adapter=appHost.sysinfo.dvb_list[this.value];self.scope.set("adapter",adapter.adapter);self.scope.set("device",adapter.device)}));form.choice("Type","type",[{value:""},{group:"Satellite",items:[{label:"DVB-S",value:"S"},{label:"DVB-S2",value:"S2"}]},{group:"Terrestrial",items:[{label:"DVB-T",value:"T"},{label:"DVB-T2",value:"T2"},{label:"ATSC",value:"ATSC"},{label:"ISDB-T",value:"ISDBT"}]},{group:"Cable",items:[{label:"DVB-C",value:"C"},{label:"DVB-C (Annex A)",value:"C/A"},{label:"DVB-C (Annex B)",value:"C/B"},{label:"DVB-C (Annex C)",value:"C/C"}]}]).setRequired().on("change",(function(){if(adapterTypeMap[this.value]!=adapterTypeBase){self.scope.reset({$tab:tabId,$adapterId:adapterId,adapter:self.scope.get("adapter"),device:self.scope.get("device"),enable:self.scope.get("enable"),id:self.scope.get("id"),type:this.value,name:self.scope.get("name")})}else{self.scope.reset()}}))}if(adapterType&&tabId=="advanced"){form.choice("Modulation","modulation",[{value:"",label:"Default: Auto"},{value:"QPSK"},{value:"QAM16",label:"16-QAM"},{value:"QAM32",label:"32-QAM"},{value:"QAM64",label:"64-QAM"},{value:"QAM128",label:"128-QAM"},{value:"QAM256",label:"256-QAM"},{value:"VSB8"},{value:"VSB16"},{value:"PSK8",label:"8PSK"},{value:"APSK16"},{value:"APSK32"},{value:"DQPSK"}])}if(adapterTypeBase=="S"){if(tabId==""){form.header("Transponder");form.number("Frequency","frequency","950..13250 MHz").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&value>=950&&value<13250}));form.choice("Polarization","polarization",dvbPolarization).setRequired();form.number("Symbolrate","symbolrate","1000..50000 Kbaud").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&value>1e3&&value<5e4}))}if(tabId=="advanced"){form.choice("FEC","fec",dvbFec);if(adapterType=="S2"){form.choice("Roll-off","rolloff",[{value:"",label:"Default: 35"},{value:"25"},{value:"20"},{value:"AUTO"}])}form.number("Stream ID","stream_id","Multistream filtering").addButton($.element.button().addClass("icon icon-settings").on("click",(function(){var x=self.scope.get("stream_id")||0;$.modal().bindScope(AdaptersModule.renderPlsCals,{mode:x>>26,code:x>>8&262143,id:x&255}).on("submit",(function(){var data=this.scope.serialize();var streamId=(Number(data.mode)<<26)+(Number(data.code)<<8)+Number(data.id);self.scope.set("stream_id",streamId>0?streamId:undefined)}))})))}if(tabId=="lnb"){form.number("LOF1","lof1","Low sub-band");form.number("LOF2","lof2","High sub-band");form.number("SLOF","slof","Sub-band range");form.checkbox("LNB Sharing. Disable LNB voltage supply and tone signal","lnb_sharing");form.checkbox("Force Tone","tone")}if(tabId=="unicable"){var uniChId=[{value:"",label:"Default: unicable disabled"}];var uniPos,uniFreq;for(var i=1;i<10;++i){uniChId.push({value:i})}form.choice("Unicable Slot","uni_scr",uniChId).on("change",(function(){if(!this.value){self.scope.set("uni_pos","");self.scope.set("uni_frequency","");uniPos.setDisabled(true);uniFreq.setDisabled(true)}else{uniPos.setDisabled(false);uniFreq.setDisabled(false)}}));var isUnicableSlot=self.scope.get("uni_scr")!=undefined;uniPos=form.choice("Slot Position","uni_pos",[{value:""},{value:"A"},{value:"B"}]).setDisabled(!isUnicableSlot);uniFreq=form.number("Slot Frequency","uni_frequency","950..2150 MHz").setDisabled(!isUnicableSlot).addValidator((function(value){if(value==undefined||value=="")return true;value=Number(value);return!isNaN(value)&&value>=950&&value<2150}))}if(tabId=="diseqc"){form.choice("DiSEqC Mode","diseqc_mode",[{value:"",label:"Default: DiSEqC 1.0"},{value:"1.1",label:"DiSEqC: 1.1"},{value:"toneburst",label:"Tone Burst"},{value:"cmd",label:"DiSEqC Command"}]).on("change",(function(){self.scope.reset()}));switch(self.scope.get("diseqc_mode")){case"1.1":var x=[{value:""}];for(var i=1;i<=16;i++)x.push({value:i});form.choice("DiSEqC 1.1","diseqc",x);break;case"toneburst":form.choice("Tone Burst","diseqc",[{value:""},{value:"A"},{value:"B"}]);break;case"cmd":form.input("DiSEqC Command","diseqc");break;case"1.0":default:var x=[{value:""}];for(var i=1;i<=4;i++)x.push({value:i});form.choice("DiSEqC 1.0","diseqc",x);break}}}else if(adapterTypeBase=="T"){if(tabId==""){form.number("Frequency","frequency","1..1000 MHz").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&(value>0&&value<1e3||value>1e6&&value<1e9)}))}if(tabId=="advanced"){form.number("PLP ID","stream_id","Multistream filtering");form.choice("Bandwidth","bandwidth",[{value:"",label:"Default: Auto"},{value:"6MHz"},{value:"7MHz"},{value:"8MHz"}]);form.choice("Guard","guardinterval",[{value:"",label:"Default: Auto"},{value:"1/32"},{value:"1/16"},{value:"1/8"},{value:"1/4"},{value:"1/128"},{value:"19/128"},{value:"19/256"}]);form.choice("Transmit","transmitmode",[{value:"",label:"Default: Auto"},{value:"1K"},{value:"2K"},{value:"4K"},{value:"8K"},{value:"16K"},{value:"32K"}]);form.choice("Hierarchy","hierarchy",[{value:"",label:"Default: Auto"},{value:"NONE"},{value:"1"},{value:"2"},{value:"4"}])}}else if(adapterTypeBase=="C"){if(tabId==""){form.number("Frequency","frequency","80..1000 MHz").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&(value>=80&&value<1e3||value>=8e7&&value<1e9)}));form.number("Symbolrate","symbolrate","1000..10000 Kbaud").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&value>1e3&&value<1e4}))}if(tabId=="advanced"){form.choice("FEC","fec",dvbFec)}}else if(adapterTypeBase=="ATSC"){if(tabId==""){form.number("Frequency","frequency","0..1000 MHz").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&(value>0&&value<1e3||value>1e6&&value<1e9)}))}}else if(adapterTypeBase=="ISDBT"){if(tabId==""){form.number("Frequency","frequency","0..1000 MHz").setRequired().addValidator((function(value){value=Number(value);return!isNaN(value)&&(value>0&&value<1e3||value>1e6&&value<1e9)}))}if(tabId=="advanced"){form.choice("Bandwidth","bandwidth",[{value:"",label:"Default: Auto"},{value:"6MHz"},{value:"7MHz"},{value:"8MHz"}])}}if(adapterType&&tabId=="advanced"){form.input("Timeout","timeout","Delay in seconds before check DVR errors. Default: 2 sec");form.input("DDCI","ddci","Bind adapter to the DigitalDevices CI");form.checkbox("Budget Mode. Disable hardware PID filtering","budget");form.checkbox("Signal in dBm","raw_signal");form.checkbox("Scale DD MaxS8 SNR","mxl5xx_snr");form.input("CA Delay","ca_pmt_delay","Delay before initialize CA module. Default: 5 sec")}var serialize=function(){if(self.scope.get("$remove"))return{remove:true};var data=self.scope.serialize();if(!data.id)data.id=appHost.makeUid("dvb_tune");if(!data.diseqc)delete data["diseqc_mode"];return data};var apply=function(streams,cam){var data=serialize();$.forEach(streams,(function(s){s.enable=true;s.type="spts";s.id=appHost.makeUid("make_stream");var x="dvb://"+data.id+"#pnr="+s.pnr;if(cam)x+=cam==true?"&cam":"&cam="+cam;s.input=[x];delete s.pnr}));appHost.request({cmd:"set-adapter",gid:appHost.config.gid,id:adapterId,adapter:data,scan:streams},(function(){$.route(MainModule.link)}),(function(){$.err({title:"Failed to save adapter"})}))};form.hr();var btnApply=$.element.button("Apply").addClass(self.scope.get("$remove")?"danger":"submit").on("click",(function(){if(!self.scope.get("$remove")&&!self.scope.validate())$.err({title:"Form has errors"});else apply()}));var btnScan=$.element.button("Scan").on("click",(function(){if(!self.scope.validate())$.err({title:"Form has errors"});else AdaptersModule.startScan.call(self,host,apply)}));if(adapterId!="-"&&tabId==""){form.checkbox("Remove","$remove").setDanger().on("change",(function(){if(this.checked)btnApply.removeClass("submit").addClass("danger");else btnApply.removeClass("danger").addClass("submit")}))}form.submit().addChild(btnApply,btnScan)};AdaptersModule.init=function(){var scope;var x=location.hash.slice(AdaptersModule.link.length+1).split("/"),adapterId=decodeURIComponent(x.pop()),host=x.pop();if(!host||!app.hosts[host]){$.route(MainModule.link);return}if(adapterId=="-"){scope={enable:true}}else{var a=MainModule.adapters[host+"/"+adapterId];if(!a){$.route(MainModule.link);return}scope=$.clone(a.$config)}scope.$adapterId=adapterId;$.body.bindScope(AdaptersModule.render,scope,host)};app.modules.push(AdaptersModule)})();

(function(){"use strict";window.SessionsModule={label:"Sessions",link:"#/sessions",order:5};SessionsModule.run=function(){SessionsModule.sessions={};SessionsModule.hide=(app.hosts[location.host].config.settings||{}).http_disable_sessions;app.on("session_event",(function(event){var hostId=event.host,data=event.data,cdate=Date.now();$.forEach(data.sessions,(function(e){var sessionsId=hostId+"/"+e.client_id;if(e.hasOwnProperty("channel_id")){e.host=hostId;e.sdate=new Date(cdate-e.uptime*1e3);if(SessionsModule.addSession)SessionsModule.addSession(e);SessionsModule.sessions[sessionsId]=e}else{if(SessionsModule.removeSession)SessionsModule.removeSession(sessionsId);delete SessionsModule.sessions[sessionsId]}}))}))};SessionsModule.addHost=function(hostId){var appHost=app.hosts[hostId];appHost.request({cmd:"sessions"},(function(response){var cdate=Date.now();$.forEach(response.data.sessions,(function(e){var sessionsId=hostId+"/"+e.client_id;e.host=hostId;e.sdate=new Date(cdate-e.uptime*1e3);SessionsModule.sessions[sessionsId]=e;if(SessionsModule.addSession)SessionsModule.addSession(e)}))}))};SessionsModule.removeHost=function(hostId){$.forEach(SessionsModule.sessions,(function(s,i){if(s.host==hostId){if(SessionsModule.removeSession)SessionsModule.removeSession(i);delete SessionsModule.sessions[i]}}))};SessionsModule.render=function(){var object=app.renderInit();var orderString=function(a,b){var ca=a.textContent;var cb=b.textContent;return ca.localeCompare(cb)};var orderNumber=function(a,b){var ca=Number(a.dataset.value);var cb=Number(b.dataset.value);return ca==cb?0:ca>cb?1:-1};var sessionsTotalValue=0;var sessionsTotal=$.element("span").setText(0);var total=$.element("p").addAttr("style","font-weight: bold; padding-left: 5px;").addChild("Total Sessions: ",sessionsTotal);var header=$.element("thead").addChild($.element("tr").addChild($.element("th").setText("Server").setStyle("width","200px").dataOrder(orderString),$.element("th").setText("Stream").dataOrder(orderString,true),$.element("th").setText("IP").setStyle("width","150px").dataOrder(orderNumber),$.element("th").setText("Login").setStyle("width","150px").dataOrder(orderString),$.element("th").setText("Uptime (min)").setStyle("width","150px").dataOrder(orderNumber),$.element("th").setStyle("width","32px")));var content=$.element("tbody");content.nodes={};var table=$.element("table").addClass("table hover").addChild(header,content);object.addChild(total,table);$.tableInit(table);SessionsModule.addSession=function(s){sessionsTotal.setText(++sessionsTotalValue);var uptime=$.element("td").addAttr("data-value",s.uptime);var hostName=app.hosts[s.host].name;var row=$.element("tr").addChild($.element("td").addAttr("data-value",hostName).setText(hostName),$.element("td").setText(s.channel_name),$.element("td").addAttr("data-value",ip2num(s.addr)).setText(s.addr),$.element("td").addChild(s.login!=undefined?$.element("a").setText(s.login).on("click",(function(event){event.preventDefault();SettingsUsersModule.openConfig(s.login)})):""),uptime,$.element("td").addClass("action").addChild($.element.button().addClass("icon icon-close").on("click",(function(){app.hosts[s.host].request({cmd:"close-session",id:s.client_id})}))));row.refreshUptime=function(){var c=Date.now(),u=c-s.sdate,d=Math.round(u/6e4),m=d%60,h=Math.floor(d/60);if(m<10)m="0"+m;uptime.addAttr("data-value",u).setText(h+":"+m)};row.refreshUptime();content.nodes[s.host+"/"+s.client_id]=row;$.tableSortInsert(table,row)};SessionsModule.removeSession=function(id){sessionsTotal.setText(--sessionsTotalValue);if(content.nodes[id]){content.nodes[id].remove();delete content.nodes[id]}};$.forEach(SessionsModule.sessions,(function(e){SessionsModule.addSession(e)}));var refresh=setInterval((function(){$.forEach(content.nodes,(function(r){r.refreshUptime()}))}),1e4);self.on("destroy",(function(){clearInterval(refresh);content.empty();content.nodes={};delete SessionsModule.addSession;delete SessionsModule.removeSession}))};SessionsModule.init=function(){$.body.bindScope(SessionsModule.render,{})};app.modules.push(SessionsModule);app.menu.push(SessionsModule)})();

(function(){"use strict";window.SettingsModule={label:"Settings",order:8};SettingsModule.click=function(){var modal=$.modal();modal.addClass("main-menu");var makeItem=function(item){modal.addChild($.element("a").setText(item.label).addAttr("href",item.link||"#").on("click",(function(event){modal.remove();if(item.click){event.preventDefault();item.click(event)}})))};var makeItems=function(items){if(items){$.forEach(items,makeItem);modal.addChild($.element("hr"))}};makeItems(app.settings);modal.addChild($.element("a").setText("Cancel").addClass("text-center").addAttr("href","#/").on("click",(function(event){event.preventDefault();modal.remove()})))};app.modules.push(SettingsModule);app.menu.push(SettingsModule)})();

(function(){"use strict";var SettingsGeneralModule={label:"General",link:"#/settings-general",order:1};SettingsGeneralModule.run=function(){app.on("set-settings",(function(event){var hostId=event.host,data=event.data,appHost=app.hosts[hostId];if(data.gid)appHost.config.gid=data.gid;if(data.settings)appHost.config.settings=data.settings;else delete appHost.config.settings;$.msg({title:"Settings saved"})}));app.on("set-config",(function(event){var hostId=event.host,data=event.data,appHost=app.hosts[hostId];if(data.data)appHost.config[data.key]=data.data;else delete appHost.config[data.key];$.msg({title:"Settings saved"})}))};SettingsGeneralModule.render=function(){var self=this,object=app.renderInit(),masterHost=app.hosts[location.host],form=new Form(self.scope,object);form.input("Monitoring","event_request","Export telemetry and events to the external server");form.checkbox("HTTP Sessions","http_disable_sessions").addAttr("data-true","undefined").addAttr("data-false","true");form.header("Default Stream Options");form.checkbox("Start stream on demand","http_keep_active").addAttr("data-true","undefined").addAttr("data-false","-1");form.number("HTTP Keep Active","http_keep_active","Delay before stop stream if no active connections. Default: 0 (turn off immediately)");form.number("Backup Start Delay","backup_start_delay","Delay before start next input. Default: 0");form.number("Backup Return Delay","backup_return_delay","Delay before return to previous input. Default: 0");form.number("HTTP Output Buffer","http_buffer","Size in Kb of the buffer for each client. Default: 1024");form.input("TCP Congestion Control","http_congestion");form.number("CC Limit","cc_limit","Default: 0 (not limited)");form.checkbox("Use multithreading for UDP receiving and transmitting","udp_threads");form.header("EPG Import");form.input("XMLTV Address","xmltv_address","file://");var btnApply=$.element.button("Apply & Restart").addClass("submit").on("click",(function(){masterHost.request({cmd:"set-settings",settings:self.scope.serialize()},(function(data){}),(function(){$.err({title:"Failed to save settings"})}))}));form.submit().addChild(btnApply)};SettingsGeneralModule.init=function(){var h=app.hosts[location.host];var s=$.clone(h.config.settings)||{};$.body.bindScope(SettingsGeneralModule.render,s)};app.modules.push(SettingsGeneralModule);app.settings.push(SettingsGeneralModule)})();

(function(){"use strict";window.SettingsUsersModule={label:"Users",link:"#/settings-users",order:5,search:true,menu:[{label:"New User",click:function(){SettingsUsersModule.openConfig()}}]};SettingsUsersModule.typeMap={3:"User",2:"Observer",1:"Administrator"};SettingsUsersModule.modules=[];SettingsUsersModule.run=function(){app.on("set-user",(function(event){var hostId=event.host,data=event.data,appHost=app.hosts[hostId];if(data.id==app.login){location.reload(false);return}if(data.user.remove){$.msg({title:'User "{0}" removed'.format(data.id)});delete appHost.config.users[data.id];if(SettingsUsersModule.removeUser){SettingsUsersModule.removeUser(data.id)}}else{appHost.config.users[data.id]=data.user;$.msg({title:'User "{0}" saved'.format(data.id)});if(SettingsUsersModule.addUser){SettingsUsersModule.removeUser(data.id);SettingsUsersModule.addUser(data.id,data.user)}}}))};var value2date=function(v){if(!v)return"";var d=new Date(v*1e3);var dd=("0"+d.getDate()).slice(-2);var dm=("0"+(d.getMonth()+1)).slice(-2);var dy=d.getFullYear();return dy+"-"+dm+"-"+dd};var daysInMonth=function(date){return new Date(date.getYear(),date.getMonth()+1,0).getDate()};var daysInYear=function(date){date=date.getYear()+1900;return date%4==0&&date%100!=0||date%4==0&&date%100==0&&date%400==0?366:365};var addDays=function(date,v){var x=new Date(date);switch(v){case"1d":v=1;break;case"1w":v=7;break;case"1m":{v=daysInMonth(date);var next=new Date(date.getYear(),date.getMonth()+1,1);next=daysInMonth(next);if(date.getDate()>next)v=v-date.getDate()+next;break}case"1y":v=daysInYear(date);break;default:v=0}x.setDate(date.getDate()+v);return x};SettingsUsersModule.renderConfig=function(){var modal=this,form=new Form(modal.scope,modal),login=modal.scope.get("login"),appHost=app.hosts[location.host];if(app.login!=login){form.checkbox("Enable","user.enable").addAttr("data-false","false")}var inputLogin=form.input("Login","login");var inputPass=form.password("Password","user.password");if(login){inputLogin.setDisabled(true);if(modal.scope.get("user.firstrun")){inputPass.setRequired();modal.scope.set("user.firstrun")}}else{inputLogin.setRequired().addValidator((function(value){if(value==login)return true;return appHost.config.users[value]==undefined}))}form.input("Comment","user.comment");if(app.login!=login){var typeMap=[];$.forEach(SettingsUsersModule.typeMap,(function(label,value){typeMap.push({value:value,label:label})}));typeMap.sort((function(a,b){return a.label.localeCompare(b.label)}));form.choice("Type","user.type",typeMap)}var auth_enable=appHost.config.auth&&!!appHost.config.auth.enable;if(auth_enable){form.input("Token","user.token");form.input("IP","user.ip");form.input("STB","user.stb");var d=form.scope.get("user.expire");if(d)form.scope.set("$user-expire",value2date(d));form.input("Expiration","$user-expire","Format: YYYY-MM-DD").on("input",(function(){var x=this.value||"0";x=x.split("-");if(x.length!=3){x=undefined}else{x=new Date(Number(x[0]),Number(x[1])-1,Number(x[2]));x=x.getTime()/1e3}form.scope.set("user.expire",x)})).addButton($.element("select").addClass("button icon icon-more").on("change",(function(){var x=form.scope.get("user.expire");x=!!x?new Date(x*1e3):new Date;x=addDays(x,this.value);x=x.getTime()/1e3;form.scope.set("user.expire",x);form.scope.set("$user-expire",value2date(x));this.value=""})).addChild($.element("option")).addChild($.element("option").setValue("1d").setText("+1 Day")).addChild($.element("option").setValue("1w").setText("+1 Week")).addChild($.element("option").setValue("1m").setText("+1 Month")).addChild($.element("option").setValue("1y").setText("+1 Year")));form.input("Limit connections","user.conlimit","Default: 0 - unlimited");var categoriesMap=[{value:""}];$.forEach(app.hosts[location.host].config.categories,(function(x){categoriesMap.push({value:x.name})}));categoriesMap.sort((function(a,b){return a.value.localeCompare(b.value)}));var packages=form.scope.get("user.packages");form.header("Packages","New Package",(function(){if(!packages){packages=[];form.scope.set("user.packages",packages)}packages.push("");form.scope.reset()}));$.forEach(packages,(function(v,k){form.choice("","user.packages."+k,categoriesMap).on("change",(function(){if(!this.value){packages.splice(k,1);if(!packages.length)form.scope.set("user.packages");form.scope.reset()}}))}))}$.forEach(SettingsUsersModule.modules,(function(module){form.hr();module.renderSettings.call(modal,form)}));form.hr();var btnApply=$.element.button("Apply").addClass("submit").on("click",(function(){if(modal.scope.validate())modal.submit()}));var btnCancel=$.element.button("Cancel").on("click",(function(){modal.remove()}));if(login&&app.login!=login)form.checkbox("Remove user","user.remove").setDanger().on("change",(function(){if(this.checked)btnApply.removeClass("submit").addClass("danger");else btnApply.removeClass("danger").addClass("submit")}));form.submit().addChild(btnApply,btnCancel)};SettingsUsersModule.openConfig=function(login){var modalData={};if(login==undefined){modalData.login="";modalData.user={enable:true,type:3}}else{modalData.login=login;modalData.user=$.clone(app.hosts[location.host].config.users[login])}$.modal().bindScope(SettingsUsersModule.renderConfig,modalData).on("submit",(function(){var data=this.scope.serialize();if(data.user.remove)data.user={remove:true};var appHost=app.hosts[location.host];appHost.request({cmd:"set-user",id:data.login,user:data.user},(function(){}),(function(){$.err({title:"Failed to save user"})}))}))};SettingsUsersModule.render=function(){var self=this,object=app.renderInit(),appHost=app.hosts[location.host];var orderString=function(a,b){var ca=a.innerHTML;var cb=b.innerHTML;return ca.localeCompare(cb)};var orderNumber=function(a,b){var ca=Number(a.dataset.value);var cb=Number(b.dataset.value);return ca==cb?0:ca>cb?1:-1};var header=$.element("thead").addChild($.element("tr").addChild($.element("th").setText("Login").setStyle("width","200px").dataOrder(orderString,true),$.element("th").setText("Comment").addClass("expand").dataOrder(orderString),$.element("th").setText("Type").setStyle("width","150px").dataOrder(orderNumber),$.element("th").setText("Created").setStyle("width","150px").dataOrder(orderNumber)));var auth_enable=appHost.config.auth&&!!appHost.config.auth.enable;if(auth_enable){header.firstChild.addChild($.element("th").setText("IP").setStyle("width","150px").addAttr("data-order","on").dataOrder(orderNumber),$.element("th").setText("Expiration").setStyle("width","150px").addAttr("data-order","on").dataOrder(orderNumber))}$.forEach(SettingsUsersModule.modules,(function(module){if(module.renderHeader)module.renderHeader(header.firstChild)}));var content=$.element("tbody");content.nodes={};var table=$.element("table").addClass("table hover").addChild(header,content);object.addChild(table);$.tableInit(table);SettingsUsersModule.addUser=function(login,u){var row=$.element("tr").addChild($.element("td").setText(login),$.element("td").setText(u.comment||""),$.element("td").addAttr("data-value",u.type).setText(SettingsUsersModule.typeMap[u.type]),$.element("td").addAttr("data-value",u.created).setText(time2ddmmmyyyy(u.created))).on("click",(function(){SettingsUsersModule.openConfig(login)}));if(auth_enable){row.addChild($.element("td").addAttr("data-value",ip2num(u.ip)).setText(u.ip),$.element("td").addAttr("data-value",u.expire).setText(time2ddmmmyyyy(u.expire)))}$.forEach(SettingsUsersModule.modules,(function(module){if(module.renderItem)module.renderItem(row,u)}));if(u.enable==undefined)u.enable=true;if(!u.enable)row.addClass("text-delete text-gray");content.nodes[login]=row;$.tableSortInsert(table,row)};SettingsUsersModule.removeUser=function(id){if(content.nodes[id]){content.nodes[id].remove();delete content.nodes[id]}};$.forEach(appHost.config.users,(function(data,login){SettingsUsersModule.addUser(login,data)}));app.search=function(value){$.tableFilter(content,value.toLowerCase())};self.on("destroy",(function(){content.empty();content.nodes={};delete SettingsUsersModule.addUser;delete SettingsUsersModule.removeUser}));self.on("ready",(function(){document.querySelector(".search").focus()}))};SettingsUsersModule.init=function(){$.body.bindScope(SettingsUsersModule.render,{})};app.modules.push(SettingsUsersModule);app.settings.push(SettingsUsersModule)})();

(function(){"use strict";var SettingsHlsModule={label:"HLS",link:"#/settings-hls",order:6};SettingsHlsModule.click=function(){app.selectHost(SettingsHlsModule.link)};SettingsHlsModule.render=function(hostId){var self=this,object=app.renderInit(),appHost=app.hosts[hostId],form=new Form(self.scope,object);form.input("Hostname","hostname","Append server address into HLS URL's");form.input("Duration","duration","Segments duration in seconds. Default: 5");form.input("Quantity","quantity","Number of segments. Default: 6");form.choice("Segments naming","naming",[{value:"",label:"Default: PCR hash"},{value:"seq",label:"Sequence"}]);form.checkbox("Round duration value","round_duration");form.checkbox("Use Expires header","expires_header");var defaultHeaders={m3u8:["Access-Control-Allow-Origin: *","Access-Control-Allow-Methods: GET","Access-Control-Allow-Credentials: true","Content-Type: application/vnd.apple.mpegURL"],ts:["Access-Control-Allow-Origin: *","Access-Control-Allow-Methods: GET","Access-Control-Allow-Credentials: true","Content-Type: video/MP2T"]};var makeHeader=function(v){var l=self.scope.get(v+"_headers");self.scope.set("$default_"+v+"_headers",!l);form.checkbox("Use default headers for ."+v,"$default_"+v+"_headers").on("change",(function(){var x=this.checked?undefined:defaultHeaders[v];self.scope.set(v+"_headers",x);self.scope.reset()}));if(!!l){form.header(v+" Headers","New Header",(function(){l.push("");self.scope.reset();document.querySelector('[name="'+v+"_headers."+(l.length-1)+'"]').focus()}));$.forEach(l,(function(lv,k){form.input("",v+"_headers."+k)}))}};form.hr();makeHeader("m3u8");form.hr();makeHeader("ts");form.hr();var serialize=function(){var data=self.scope.serialize();$.forEach(["m3u8_headers","ts_headers"],(function(v){var a=data[v];if(a){for(var i=0;i<a.length;){if(!a[i])a.splice(i,1);else i++}if(!a.length)delete data[v]}}));return $.isObjectEmpty(data)?undefined:data};var btnApply=$.element.button("Apply & Restart").addClass("submit").on("click",(function(){appHost.request({cmd:"set-config",key:"hls",restart:true,data:serialize()},(function(data){}),(function(){$.err({title:"Failed to save settings"})}))}));form.submit().addChild(btnApply)};SettingsHlsModule.init=function(){var hostId=location.hash.slice(SettingsHlsModule.link.length+1).split("/");if(!hostId||!app.hosts[hostId]){$.route(StreamsModule.link);return}var s=$.clone(app.hosts[hostId].config.hls)||{};$.body.bindScope(SettingsHlsModule.render,s,hostId,1234,5678)};app.modules.push(SettingsHlsModule);app.settings.push(SettingsHlsModule)})();

(function(){"use strict";var SettingsHttpPlayModule={label:"HTTP Play",link:"#/settings-http-play",order:7};SettingsHttpPlayModule.render=function(){var self=this,object=app.renderInit(),masterHost=app.hosts[location.host],form=new Form(self.scope,object);var categoriesMap=[{value:""}];$.forEach(masterHost.config.categories,(function(x){categoriesMap.push({value:x.name})}));categoriesMap.sort((function(a,b){return a.value.localeCompare(b.value)}));form.checkbox("Allow HTTP access to all streams","http_play_stream");form.checkbox("Allow HLS access to all streams","http_play_hls");form.choice("Playlist Arrange","playlist_arrange",categoriesMap);form.group().setHtml('<a href="/playlist.m3u8" target="_blank">playlist.m3u8</a>');form.hr();var btnApply=$.element.button("Apply & Restart").addClass("submit").on("click",(function(){masterHost.request({cmd:"set-settings",settings:self.scope.serialize()},(function(data){}),(function(){$.err({title:"Failed to save settings"})}))}));form.submit().addChild(btnApply)};SettingsHttpPlayModule.init=function(){var h=app.hosts[location.host];var s=$.clone(h.config.settings)||{};$.body.bindScope(SettingsHttpPlayModule.render,s)};app.modules.push(SettingsHttpPlayModule);app.settings.push(SettingsHttpPlayModule)})();

(function(){"use strict";var SettingsAuthOptionsModule={label:"HTTP Authentication",link:"#/settings-auth",order:9};SettingsAuthOptionsModule.run=function(){app.on("set-auth",(function(event){var hostId=event.host,data=event.data,appHost=app.hosts[hostId];if(data.auth)appHost.config.auth=data.auth;else delete appHost.config.auth;$.msg({title:"HTTP Authentication saved"});if(location.hash==SettingsAuthOptionsModule.link)SettingsAuthOptionsModule.init()}))};SettingsAuthOptionsModule.render=function(){var self=this,object=app.renderInit(),appHost=app.hosts[location.host],form=new Form(self.scope,object);form.checkbox("Enable built-in HTTP/HLS authentication (restart required)","enable").on("change",(function(){self.scope.reset()}));var renderIPList=function(key,title){var l=self.scope.get(key);if(!l){l=[];self.scope.set(key,l)}form.header(title,"new ip",(function(){l.push("");self.scope.reset();document.querySelector('[name="'+key+"."+(l.length-1)+'"]').focus()}));$.forEach(l,(function(v,i){var x=form.input("#"+(i+1),key+"."+i);x.on("keydown",(function(event){if(event.keyCode==13){event.preventDefault();x.blur();var n;if(event.shiftKey){if(i==0){n=0;l.splice(0,0,"");self.scope.reset()}else{n=i-1}}else{n=i+1;if(n==l.length){l.push("");self.scope.reset()}}document.querySelector('[name="'+key+"."+n+'"]').focus()}}));x.addButton($.element.button().addClass("icon icon-close").on("click",(function(){l.splice(i,1);if(!l.length)self.scope.set(key);self.scope.reset()})))}))};if(self.scope.get("enable")==true){form.input("Ministra/Stalker Authorization","ministra","Ministra/Stalker portal address");renderIPList("http_ip_allow","Allow access without authorization");renderIPList("http_ip_deny","Deny access")}var serialize=function(){var data=self.scope.serialize();var cleanIPList=function(key){var a=data[key];if(a==undefined)return;for(var i=0;i<a.length;){if(!a[i])a.splice(i,1);else++i}if(!a.length)delete data[key]};cleanIPList("http_ip_allow");cleanIPList("http_ip_deny");$.forEach(data,(function(v,k){if(!v)delete data[k]}));return data};var btnApply=$.element.button("Apply").addClass("submit").on("click",(function(){appHost.request({cmd:"set-auth",auth:serialize()},(function(data){}),(function(){$.err({title:"Failed to save settings"})}))}));form.submit().addChild(btnApply)};SettingsAuthOptionsModule.init=function(){var appHost=app.hosts[location.host],scope=appHost.config.auth!=undefined?$.clone(appHost.config.auth):{};$.body.bindScope(SettingsAuthOptionsModule.render,scope)};app.modules.push(SettingsAuthOptionsModule);app.settings.push(SettingsAuthOptionsModule)})();

(function(){"use strict";window.SettingsSoftcamModule={label:"Softcam",link:"#/settings-softcam",order:10};SettingsSoftcamModule.click=function(){app.selectHost(SettingsSoftcamModule.link,"-")};SettingsSoftcamModule.run=function(){app.on("set-softcam",(function(event){var hostId=event.host,appHost=app.hosts[hostId],data=event.data;if(data.gid)appHost.config.gid=data.gid;var sl=appHost.config.softcam;var idx=sl==undefined?-1:sl.indexOfID(data.softcam.id);if(data.softcam.remove&&!data.softcam.up){$.msg({title:'Softcam "{0}" removed'.format(sl[idx].name)})}if(idx!=-1){sl.splice(idx,1);if(!sl.length)delete appHost.config.softcam}if(!data.softcam.remove){if(!sl)appHost.config.softcam=sl=[];sl.push(data.softcam);$.msg({title:'Softcam "{0}" saved'.format(data.softcam.name)})}}))};SettingsSoftcamModule.renderTest=function(){var modal=this,form=new Form(modal.scope,modal);form.input("CaID","caid","Testing...").addAttr("readonly");if(modal.scope.get("caid")!=undefined){form.input("AU","au").addAttr("readonly");form.input("UA","ua").addAttr("readonly");var idents=modal.scope.get("idents");if(idents){form.header("Idents");$.forEach(idents,(function(v,k){form.input(v.id,"idents."+k+".sa").addAttr("readonly")}))}}var btnOk=$.element.button("Ok").on("click",(function(){modal.remove()}));form.hr();form.submit().addChild(btnOk)};SettingsSoftcamModule.renderStreams=function(hostId,softcamId){var modal=this,appHost=app.hosts[hostId],form=new Form(modal.scope,modal);var streams=[],allowedInputs=["dvb","udp","rtp","http"];$.forEach(appHost.config.make_stream,(function(s){$.forEach(s.input,(function(i,k){i=parseUrl(i);if(allowedInputs.indexOf(i.format)!=-1)streams.push({id:hostId+"/"+s.id,inputId:k,name:s.name+" #"+(k+1),cam:softcamId===i.cam})}))}));streams.sort((function(a,b){return a.name.localeCompare(b.name)}));$.forEach(streams,(function(v){var x=form.checkbox(v.name,"").on("change",(function(){var s=$.clone(MainModule.streams[v.id].$config),i=parseUrl(s.input[v.inputId]);if(this.checked)i.cam=softcamId;else delete i.cam;s.input[v.inputId]=makeUrl(i);var id=v.id.split("/"),streamId=id.pop(),hostId=id.pop();app.hosts[hostId].request({cmd:"set-stream",id:streamId,stream:s},(function(){}),(function(){$.err({title:"Failed to save stream"})}))}));if(v.cam)x.checked=true}));var btnOk=$.element.button("Ok").addClass("submit").on("click",(function(){modal.remove()}));form.hr();form.submit().addChild(btnOk)};SettingsSoftcamModule.render=function(hostId){var self=this,object=app.renderInit(),softcamId=self.scope.get("$softcamId"),appHost=app.hosts[hostId],tabId=self.scope.get("$tab")||"",form=new Form(self.scope,object),type=self.scope.get("type");var softcamList=[{value:"-",label:"New Softcam"},{value:"",label:"---",disabled:true}];var softcamSortedList=[];$.forEach(appHost.config.softcam,(function(i){softcamSortedList.push({value:i.id,label:i.name})}));softcamSortedList.sort((function(a,b){return a.label.localeCompare(b.label)}));softcamList=softcamList.concat(softcamSortedList);form.choice("Softcam","$softcamId",softcamList).on("change",(function(){if(this.value=="-")app.selectHost(SettingsSoftcamModule.link,"-");else $.route(SettingsSoftcamModule.link+"/"+hostId+"/"+this.value)}));form.hr();form.tab("$tab",[{label:"General",id:""},{label:"Advanced",id:"advanced"}]);if(tabId==""){form.input("Name","name","SoftCAM Name").setRequired();var x=form.input("ID","id","Unique softcam identifier").addValidator(validateId);if(softcamId!="-")x.setRequired();form.choice("Protocol","type",[{value:"newcamd",label:"NewCAMd"}]);if(type=="newcamd"){form.input("Address","host","Domain or IP Address").setRequired();form.number("Port","port","Port").setRequired().addValidator(validatePort);form.input("Login","user").setRequired();form.password("Password","pass").setRequired()}}if(tabId=="advanced"){if(type=="newcamd"){form.input("DES Key","key","Default: 0102030405060708091011121314");form.input("CaID","caid","Change server CaID. Example: 06B0");form.number("Timeout","timeout","Default: 8 sec.")}form.checkbox("Make new connection for each input","split_cam");form.checkbox("Allow EMM if allowed by the server","disable_emm").addAttr("data-true","undefined").addAttr("data-false","true");form.checkbox("Use ECM response with valid checksum only","ignore_cw").addAttr("data-true","undefined").addAttr("data-false","true");form.number("Skip ECM","skip_ecm","Ignore first ECM packets")}var serialize=function(){if(self.scope.get("$remove"))return{remove:true};var data=self.scope.serialize();if(!data.id)data.id=appHost.makeUid("softcam");return data};form.hr();var btnApply=$.element.button("Apply").addClass("submit").on("click",(function(){if(!self.scope.get("$remove")&&!self.scope.validate()){$.err({title:"Form has errors"});return}var softcam=serialize();appHost.request({cmd:"set-softcam",gid:appHost.config.gid,id:softcamId,softcam:softcam},(function(){if(softcam.remove){app.selectHost(SettingsSoftcamModule.link,"-")}else if(softcamId!=softcam.id){$.route(SettingsSoftcamModule.link+"/"+hostId+"/"+softcam.id)}else{self.scope.reset()}}),(function(){$.err({title:"Failed to save softcam"})}))}));var btnTest=$.element.button("Test").on("click",(function(){if(!self.scope.validate()){$.err({title:"Form has errors"});return}var modal=$.modal().bindScope(SettingsSoftcamModule.renderTest,{});appHost.request({cmd:"test-softcam",config:serialize()},(function(response){var data=response.data;if(data.error){modal.remove();$.err({title:"Softcam test failed",text:data.error})}else{data.info.caid=data.info.caid.toHex(4);data.info.au=data.info.au?"YES":"NO";modal.scope.reset(data.info)}}),(function(){modal.remove();$.err({title:"Softcam test failed"})}))}));var btnClone=$.element.button("Clone").on("click",(function(){var data=self.scope.serialize();data.name=(data.name||"")+" (clone)";if(data.id!=undefined)delete data.id;SettingsSoftcamModule.softcamClone=data;app.selectHost(SettingsSoftcamModule.link,"-")}));var btnStreams=$.element.button("Streams").on("click",(function(){$.modal().bindScope(SettingsSoftcamModule.renderStreams,{},hostId,softcamId)}));if(softcamId!="-"){if(tabId==""){form.checkbox("Remove Softcam","$remove").setDanger().on("change",(function(){if(this.checked)btnApply.removeClass("submit").addClass("danger");else btnApply.removeClass("danger").addClass("submit")}))}}else{btnClone.setDisabled(true);btnStreams.setDisabled(true)}form.submit().addChild(btnApply,btnTest,btnClone,btnStreams)};SettingsSoftcamModule.init=function(){var scope,x=location.hash.slice(SettingsSoftcamModule.link.length+1).split("/"),softcamId=decodeURIComponent(x.pop()),hostId=x.pop(),appHost=app.hosts[hostId];if(!appHost){$.route(StreamsModule.link);return}if(softcamId=="-"){if(SettingsSoftcamModule.softcamClone){scope=SettingsSoftcamModule.softcamClone;delete SettingsSoftcamModule.softcamClone}else{scope={name:"",type:"newcamd"}}}else{var idx=appHost.config.softcam?appHost.config.softcam.indexOfID(softcamId):-1;if(idx==-1){$.route(SettingsSoftcamModule.link);return}scope=$.clone(appHost.config.softcam[idx])}scope.$softcamId=softcamId;$.body.bindScope(SettingsSoftcamModule.render,scope,hostId)};app.modules.push(SettingsSoftcamModule);app.settings.push(SettingsSoftcamModule)})();

(function(){"use strict";window.SettingsCasModule={label:"CAS",link:"#/settings-cas",order:11};SettingsCasModule.click=function(){app.selectHost(SettingsCasModule.link,"-")};SettingsCasModule.run=function(){app.on("set-cas",(function(event){var hostId=event.host,appHost=app.hosts[hostId],data=event.data;if(data.gid)appHost.config.gid=data.gid;var sl=appHost.config.cas;var idx=sl==undefined?-1:sl.indexOfID(data.cas.id);if(data.cas.remove&&!data.cas.up){$.msg({title:'CAS "{0}" removed'.format(sl[idx].name)})}if(idx!=-1){sl.splice(idx,1);if(!sl.length)delete appHost.config.cas}if(!data.cas.remove){if(!sl)appHost.config.cas=sl=[];sl.push(data.cas);$.msg({title:'CAS "{0}" saved'.format(data.cas.name)})}}))};SettingsCasModule.render=function(hostId){var self=this,object=app.renderInit(),casId=self.scope.get("$casId"),appHost=app.hosts[hostId],form=new Form(self.scope,object),advanced=self.scope.get("$advanced")==true;var casList=[{value:"-",label:"New CAS"},{value:"",label:"---",disabled:true}];var casSortedList=[];$.forEach(appHost.config.cas,(function(i){casSortedList.push({value:i.id,label:i.name})}));casSortedList.sort((function(a,b){return a.label.localeCompare(b.label)}));casList=casList.concat(casSortedList);form.choice("CAS","$casId",casList).on("change",(function(){if(this.value=="-")app.selectHost(SettingsCasModule.link,"-");else $.route(SettingsCasModule.link+"/"+hostId+"/"+this.value)}));form.hr();form.input("Name","name","CAS Name").setRequired();form.input("Super CAS ID (hex)","super_cas_id","00000000").setRequired().addValidator(validateHex).addValidator((function(value){return!value||value.length==8}));form.hr();form.input("ECMG Channel ID","ecmg_channel_id").setRequired().addValidator((function(value){if(value==undefined||value=="")return true;value=Number(value);return!isNaN(value)&&value>=0&&value<=65535}));form.input("ECMG Address","ecmg_host").setRequired();form.input("ECMG Port","ecmg_port").setRequired().addValidator(validatePort);form.input("Crypto Period","ecmg_cp").setRequired();form.hr();form.choice("EMMG Protocol","emmg_protocol",[{value:"",label:"Default: TCP"},{value:"udp",label:"UDP",disabled:true}]);form.number("EMMG Port","emmg_port").addValidator(validatePort);form.number("EMM PID","emm_pid").setRequired().addValidator(validatePid);form.input("EMM Private Data (hex)","emm_data").addValidator(validateHex);form.checkbox("EMM Clone (duplicate EMM packets to all streams)","emm_clone");form.hr();var serialize=function(){if(self.scope.get("$remove"))return{remove:true};var data=self.scope.serialize();if(!data.id)data.id=appHost.makeUid("cas");return data};var btnApply=$.element.button("Apply").addClass("submit").on("click",(function(){if(!self.scope.get("$remove")&&!self.scope.validate()){$.err({title:"Form has errors"});return}var cas=serialize();appHost.request({cmd:"set-cas",gid:appHost.config.gid,id:casId,cas:cas},(function(){if(cas.remove){app.selectHost(SettingsCasModule.link,"-")}else if(casId!=cas.id){$.route(SettingsCasModule.link+"/"+hostId+"/"+cas.id)}else{self.scope.reset()}}),(function(){$.err({title:"Failed to save CAS"})}))}));var btnClone=$.element.button("Clone").on("click",(function(){var data=self.scope.serialize();data.name=(data.name||"")+" (clone)";if(data.id!=undefined)delete data.id;SettingsCasModule.casClone=data;app.selectHost(SettingsCasModule.link,"-")}));if(casId!="-"){form.checkbox("Remove","$remove").setDanger().on("change",(function(){if(this.checked)btnApply.removeClass("submit").addClass("danger");else btnApply.removeClass("danger").addClass("submit")}))}else{btnClone.setDisabled(true)}form.submit().addChild(btnApply,btnClone)};SettingsCasModule.init=function(){var scope,x=location.hash.slice(SettingsCasModule.link.length+1).split("/"),casId=decodeURIComponent(x.pop()),hostId=x.pop(),appHost=app.hosts[hostId];if(!appHost){$.route(StreamsModule.link);return}if(casId=="-"){if(SettingsCasModule.casClone){scope=SettingsCasModule.casClone;delete SettingsCasModule.casClone}else{scope={name:""}}}else{var idx=appHost.config.cas?appHost.config.cas.indexOfID(casId):-1;if(idx==-1){$.route(SettingsCasModule.link);return}scope=$.clone(appHost.config.cas[idx])}scope.$casId=casId;$.body.bindScope(SettingsCasModule.render,scope,hostId)};app.modules.push(SettingsCasModule);app.settings.push(SettingsCasModule)})();

(function(){"use strict";window.SettingsGroupsModule={label:"Groups",link:"#/settings-groups",order:20};SettingsGroupsModule.run=function(){app.on("set-category",(function(event){var masterHost=app.hosts[location.host],data=event.data;if(data.category.remove){var n=masterHost.config.categories[data.id].name;$.msg({title:'Category "{0}" removed'.format(n)});masterHost.config.categories.splice(data.id,1);if(!masterHost.config.categories.length)delete masterHost.config.categories}else{if(data.id!=undefined){masterHost.config.categories[data.id]=data.category}else{if(!masterHost.config.categories)masterHost.config.categories=[];masterHost.config.categories.push(data.category)}$.msg({title:'Category "{0}" saved'.format(data.category.name)})}}))};SettingsGroupsModule.renderStreams=function(){var modal=this,form=new Form(modal.scope,modal),category=modal.scope.get("category"),group=modal.scope.get("group");if(group=="")group=undefined;form.header(category);form.choice("","group",modal.scope.get("groups")).on("change",(function(){modal.scope.reset()}));form.hr();var streams=[];$.forEach(MainModule.streams,(function(v,k){var g;if(v.$config.groups)g=v.$config.groups[category];streams.push({id:k,name:v.$config.name,group:g==group})}));streams.sort((function(a,b){return a.name.localeCompare(b.name)}));$.forEach(streams,(function(v){var x=form.checkbox(v.name,"");if(v.group)x.checked=true;x.on("change",(function(){var s=$.clone(MainModule.streams[v.id].$config);if(this.checked==false){if(!group){this.checked=true;return}else{delete s.groups[category];if($.isObjectEmpty(s.groups))delete s.groups}}else{if(!group){delete s.groups[category];if($.isObjectEmpty(s.groups))delete s.groups}else{if(!s.groups)s.groups={};s.groups[category]=group}}var id=v.id.split("/"),streamId=id.pop(),hostId=id.pop(),appHost=app.hosts[hostId];appHost.request({cmd:"set-stream",id:streamId,stream:s},(function(){}),(function(){$.err({title:"Failed to save stream"})}))}))}));var btnOk=$.element.button("Ok").addClass("submit").on("click",(function(){modal.remove()}));form.hr();form.submit().addChild(btnOk)};SettingsGroupsModule.render=function(){var self=this,object=app.renderInit(),categoryId=self.scope.get("$categoryId"),masterHost=app.hosts[location.host],form=new Form(self.scope,object);var categoriesList=[{value:"-",label:"New Category"},{value:"",label:"---",disabled:true}];var categoriesSortedList=[];$.forEach(masterHost.config.categories,(function(i,k){categoriesSortedList.push({value:k,label:i.name})}));categoriesSortedList.sort((function(a,b){return a.label.localeCompare(b.label)}));categoriesList=categoriesList.concat(categoriesSortedList);form.choice("Category","$categoryId",categoriesList).on("change",(function(){$.route(SettingsGroupsModule.link+"/"+this.value)}));form.hr();form.input("Name","name","Category Name").setRequired();var groups=self.scope.get("groups");if(!groups){groups=[];self.scope.set("groups",groups)}form.header("Groups","Add Group",(function(){groups.push({name:""});self.scope.reset();document.querySelector('[name="groups.'+(groups.length-1)+'.name"]').focus()}));$.forEach(groups,(function(v,k){if(v.name==undefined)return;var s=$.element("select").addClass("button icon icon-move").on("change",(function(){if(this.value=="-1")delete groups[k].name;else groups.move(k,Number(this.value));self.scope.reset()}));for(var j=0;j<groups.length;++j)s.addChild($.element("option").setValue(j.toString()).setText(j+1));s.addChild($.element("option").setDisabled(true).setText("---"),$.element("option").setValue("-1").setText("Remove"));s.value=k.toString();form.input(k+1,"groups."+k+".name","Group Name").addButton(s).on("keydown",(function(){if(event.keyCode==13){event.preventDefault();var n=k+1;if(n==groups.length){groups.push({name:""});self.scope.reset()}document.querySelector('[name="groups.'+n+'.name"]').focus()}}))}));form.hr();var serialize=function(){if(self.scope.get("$remove"))return{remove:true};var data=self.scope.serialize();$.forEach(data.groups,(function(g){if(!g.name){g.remove=true;delete g.name}}));return data};var btnApply=$.element.button("Apply").addClass("submit").on("click",(function(){var category=serialize();masterHost.request({cmd:"set-category",id:categoryId!="-"?categoryId:undefined,category:category},(function(data){if(category.remove){$.route(SettingsGroupsModule.link)}else if(categoryId=="-"){var i=masterHost.config.categories.length-1;$.route(SettingsGroupsModule.link+"/"+i)}else{self.scope.reset()}}),(function(){$.err({title:"Failed to save category"})}))}));var btnStreams=$.element.button("Streams").setDisabled(true).on("click",(function(){var category=masterHost.config.categories[categoryId];var modalData={category:category.name,groups:[{value:"",label:"No Group"},{value:"",label:"---",disabled:true}]};$.forEach(category.groups,(function(v){modalData.groups.push({value:v.name})}));$.modal().bindScope(SettingsGroupsModule.renderStreams,modalData)}));if(categoryId!="-"){btnStreams.setDisabled(false);form.checkbox("Remove Category","$remove").setDanger().on("change",(function(){if(this.checked)btnApply.removeClass("submit").addClass("danger");else btnApply.removeClass("danger").addClass("submit")}))}form.submit().addChild(btnApply,btnStreams)};SettingsGroupsModule.init=function(){var scope,categoryId=location.hash.slice(SettingsGroupsModule.link.length+1);if(!categoryId){$.route(SettingsGroupsModule.link+"/-");return}if(categoryId=="-"){scope={name:"",groups:[]}}else{var masterHost=app.hosts[location.host];categoryId=Number(categoryId);scope=$.clone(masterHost.config.categories[categoryId]);if(!scope){$.route(SettingsGroupsModule.link);return}if(!scope.groups||!scope.groups.length)scope.groups=[]}scope.$categoryId=categoryId;$.body.bindScope(SettingsGroupsModule.render,scope)};app.modules.push(SettingsGroupsModule);app.settings.push(SettingsGroupsModule)})();

(function(){"use strict";window.SettingsServersModule={label:"Servers",link:"#/settings-servers",order:15};SettingsServersModule.run=function(){app.on("set-server",(function(event){var masterHost=app.hosts[location.host],data=event.data;if(data.server.remove){var s=masterHost.config.servers[data.id];app.removeHost(s);$.msg({title:'Server "{0}" removed'.format(s.name)});masterHost.config.servers.splice(data.id,1);if(!masterHost.config.servers.length)delete masterHost.config.servers}else{if(data.id!=undefined){app.removeHost(masterHost.config.servers[data.id]);masterHost.config.servers[data.id]=data.server}else{if(!masterHost.config.servers)masterHost.config.servers=[];masterHost.config.servers.push(data.server)}if(data.server.enable!=false)app.addHost(data.server);$.msg({title:'Server "{0}" saved'.format(data.server.name)})}}))};SettingsServersModule.renderTest=function(){var modal=this,form=new Form(modal.scope,modal);form.input("Version","version","Testing...").addAttr("readonly");var btnOk=$.element.button("Ok").on("click",(function(){modal.remove()}));form.hr();form.submit().addChild(btnOk)};SettingsServersModule.render=function(){var self=this,object=app.renderInit(),serverId=self.scope.get("$serverId"),masterHost=app.hosts[location.host],form=new Form(self.scope,object);var serverHost;if(serverId!="-")serverHost=self.scope.get("host")+":"+self.scope.get("port");var serversList=[{value:"-",label:"New Server"},{value:"",label:"---",disabled:true}];var sortedList={streamer:[],transcoder:[],relay:[]};$.forEach(masterHost.config.servers,(function(s,i){sortedList[s.type].push({value:i,label:s.name})}));$.forEach(sortedList,(function(sl){sl.sort((function(a,b){return a.label.localeCompare(b.label)}))}));if(sortedList.streamer.length)serversList.push({group:"Streamers",items:sortedList.streamer});if(sortedList.transcoder.length)serversList.push({group:"Transcoders",items:sortedList.transcoder});if(sortedList.relay.length)serversList.push({group:"Relays",items:sortedList.relay});form.choice("Server","$serverId",serversList).on("change",(function(){$.route(SettingsServersModule.link+"/"+this.value)}));form.hr();if(self.scope.get("enable")==undefined)self.scope.set("enable",true);form.checkbox("Enable","enable").addAttr("data-false","false");form.input("Name","name","Server Name").setRequired();form.choice("Type","type",[{value:"streamer",label:"Streamer"},{value:"transcoder",label:"Transcoder",disabled:true},{value:"relay",label:"Relay",disabled:true}]);form.input("Address","host","Domain or IP Address").setRequired();form.number("Port","port","Port").setRequired().addValidator(validatePort);var maskPass=function(p){var pm=!p?"":new Array(p.length+1).join("*");self.scope.set("$maskPass",pm)};maskPass(self.scope.get("pass"));form.input("Login","user");form.input("Password","$maskPass").on("focus",(function(){self.scope.set("$maskPass",self.scope.get("pass")||"")})).on("blur",(function(){self.scope.set("pass",this.value);maskPass(this.value)}));form.hr();var serialize=function(){if(self.scope.get("$remove"))return{remove:true};else return self.scope.serialize()};var btnApply=$.element.button("Apply").addClass("submit").on("click",(function(){if(!self.scope.get("$remove")&&!self.scope.validate()){$.err({title:"Form has errors"});return}if(serverHost&&app.hosts[serverHost]){app.hosts[serverHost].destroy();delete app.hosts[serverHost]}var server=serialize();masterHost.request({cmd:"set-server",id:serverId!="-"?serverId:undefined,server:server},(function(data){if(server.remove){$.route(SettingsServersModule.link)}else if(serverId=="-"){var i=masterHost.config.servers.length-1;$.route(SettingsServersModule.link+"/"+i)}else{self.scope.reset()}}),(function(){$.err({title:"Failed to save server"})}))}));var btnTest=$.element.button("Test").on("click",(function(){if(!self.scope.validate()){$.err({title:"Form has errors"});return}var modal=$.modal().bindScope(SettingsServersModule.renderTest,{});var data=self.scope.serialize();var headers={};if(data.user){var token=data.user;if(data.pass)token+=":"+data.pass;headers["Authorization"]="Basic "+$.base64Encode(token)}$.http({method:"POST",url:"http://"+data.host+":"+data.port+"/control/",data:JSON.stringify({cmd:"version"}),headers:headers},(function(response){var data=JSON.parse(response.text);data.version="Astra "+data.version;modal.scope.reset(data)}),(function(response){modal.remove();$.err({title:"Failed to test server"})}))}));if(serverId!="-"){form.checkbox("Remove Server","$remove").setDanger().on("change",(function(){if(this.checked)btnApply.removeClass("submit").addClass("danger");else btnApply.removeClass("danger").addClass("submit")}))}form.submit().addChild(btnApply,btnTest)};SettingsServersModule.init=function(){var scope,serverId=location.hash.slice(SettingsServersModule.link.length+1);if(!serverId){$.route(SettingsServersModule.link+"/-");return}if(serverId=="-"){scope={enable:true,name:"",type:"streamer"}}else{var masterHost=app.hosts[location.host];serverId=Number(serverId);scope=$.clone(masterHost.config.servers[serverId]);if(!scope){$.route(SettingsServersModule.link);return}}scope.$serverId=serverId;$.body.bindScope(SettingsServersModule.render,scope)};app.modules.push(SettingsServersModule);app.settings.push(SettingsServersModule)})();

(function(){"use strict";var SettingsThemeModule={label:"Theme",order:80};SettingsThemeModule.click=function(){var modal=$.modal();var theme=app.getTheme()||"";modal.addClass("main-menu");$.forEach(app.themes,(function(item){var x=$.element("a").setText(item.label).addAttr("href","#");if(theme==item.value)x.addClass("active");x.on("click",(function(event){event.preventDefault();modal.remove();app.setTheme(item.value)}));modal.addChild(x)}));modal.addChild($.element("hr"));modal.addChild($.element("a").setText("Cancel").addClass("text-center").addAttr("href","#/").on("click",(function(event){event.preventDefault();modal.remove()})))};app.modules.push(SettingsThemeModule);app.settings.push(SettingsThemeModule)})();

(function(){"use strict";var SettingsImportModule={label:"Import",link:"#/settings-import",order:90};SettingsImportModule.click=function(){app.selectHost(SettingsImportModule.link)};SettingsImportModule.render=function(hostId){var self=this,object=app.renderInit(),appHost=app.hosts[hostId];var wrap=$.element("div").addClass("settings-edit");object.addChild(wrap);var text=$.element("textarea").addClass("input monospace").addAttr("placeholder","Paste here Astra Script (any version) or JSON configuration").addAttr("tabindex","0").on("keydown",(function(event){switch(event.keyCode){case 9:event.preventDefault();document.execCommand("insertText",false,"    ");break}}));var btnImport=$.element.button("Import").addClass("submit").on("click",(function(){appHost.request({cmd:"import",gid:appHost.config.gid,data:text.value},(function(response){var data=response.data;if(data.error){$.err({title:"Import failed",text:data.error,delay:5});return}var c=appHost.config;c.gid=data.gid;if(data.dvb_tune){if(!c.dvb_tune)c.dvb_tune=[];c.dvb_tune=c.dvb_tune.concat(data.dvb_tune)}if(data.softcam){if(!c.softcam)c.softcam=[];c.softcam=c.softcam.concat(data.softcam)}if(data.make_stream){if(!c.make_stream)c.make_stream=[];c.make_stream=c.make_stream.concat(data.make_stream)}MainModule.removeHost(hostId);MainModule.addHost(hostId);var x=$.msg({title:appHost.name,text:'Press "Apply & Restart" button to complete import or Reload page to discard changes',delay:-1});x.addChild($.element().addClass("row").addChild($.element.button("Apply & Restart").addClass("col-4").on("click",(function(event){event.target.setDisabled(true);appHost.upload((function(success){if(success)x.remove();else event.target.setDisabled(false)}))}))))}),(function(){$.err({title:"Failed to import settings"})}))}));var submit=$.element("div").addClass("form-submit").addChild(btnImport);wrap.addChild(text,submit);self.on("ready",(function(){text.focus()}))};SettingsImportModule.init=function(){var hostId=location.hash.slice(SettingsImportModule.link.length+1);if(!app.hosts[hostId]){$.route(SettingsModule.link)}else{$.body.bindScope(SettingsImportModule.render,{},hostId)}};app.modules.push(SettingsImportModule);app.settings.push(SettingsImportModule)})();

(function(){"use strict";var SettingsEditModule={label:"Edit Config",link:"#/settings-edit",order:91};SettingsEditModule.click=function(){app.selectHost(SettingsEditModule.link)};SettingsEditModule.render=function(hostId){var self=this,object=app.renderInit(),appHost=app.hosts[hostId];var wrap=$.element("div").addClass("settings-edit");object.addChild(wrap);var text=$.element("textarea").addClass("input monospace").addAttr("tabindex","0").on("keydown",(function(event){switch(event.keyCode){case 9:event.preventDefault();document.execCommand("insertText",false,"    ");break}}));var btnSave=$.element.button("Save").addClass("danger").on("click",(function(){appHost.request({cmd:"import",gid:appHost.config.gid,data:text.value},(function(){try{var r=JSON.parse(text.value||"{}");appHost.config=r}catch(e){$.err({title:"Error",text:e.toString()});return}MainModule.removeHost(hostId);MainModule.addHost(hostId);var x=$.msg({title:appHost.name,text:'Press "Apply & Restart" button to complete export or Reload page to discard changes',delay:-1});x.addChild($.element().addClass("row").addChild($.element.button("Apply & Restart").addClass("col-4").on("click",(function(event){event.target.setDisabled(true);appHost.upload((function(success){if(success)x.remove();else event.target.setDisabled(false)}))}))))}),(function(){$.err({title:"Failed to save settings"})}))}));var submit=$.element("div").addClass("form-submit").addChild(btnSave);wrap.addChild(text,submit);self.on("ready",(function(){text.textContent=JSON.stringify($.clone(appHost.config),null,4);text.setSelectionRange(0,0);text.focus()}))};SettingsEditModule.init=function(){var hostId=location.hash.slice(SettingsEditModule.link.length+1);if(!app.hosts[hostId]){$.route(SettingsModule.link)}else{$.body.bindScope(SettingsEditModule.render,{},hostId)}};app.modules.push(SettingsEditModule);app.settings.push(SettingsEditModule)})();

(function(){"use strict";var SettingsLicenseModule={label:"License",order:98};SettingsLicenseModule.run=function(){var appHost=app.hosts[location.host],license=appHost.sysinfo.license;if(!license)return;if(license.type==4){$.form_check=true;return}if(license.type==99){$.err({title:"License Terminated",delay:-1}).button("Buy Now!",(function(){this.remove();window.open("https://cesbo.com/astra-license","_blank").focus()}));return}var r=Math.floor((license.expire*1e3-Date.now())/(24*60*60*1e3));var d=r>1?"days":"day";if(license.id=="000000"){$.msg({title:"License Notification",text:"Unregistered evalution version! Expires in "+r+" "+d+".",delay:7}).button("Buy Now!",(function(){this.remove();window.open("https://cesbo.com/astra-license","_blank").focus()}));return}$.form_check=true;if(r>5)return;var x;if(r>0)x=$.msg({title:"License Notification",text:"Subscription expires in "+r+" "+d+"!",delay:7});else if(license.type==3)return;else x=$.err({title:"License expired!",delay:-1});x.button("Renew License",(function(){this.remove();window.open("https://cesbo.com/astra-renew?sn="+license.id,"_blank").focus()}))};SettingsLicenseModule.render=function(appHost){var modal=this,license=appHost.sysinfo.license,form=new Form(modal.scope,modal);var license_type;var expire_title="Expiration Date";switch(license.type){case 1:license_type=license.id=="000000"?"Unregistered evalution version!":"Trial";break;case 2:license_type="Subscription";break;case 3:license_type="Corporate";break;case 4:expire_title="Purchase Date";license_type="Lifetime";break;case 99:expire_title=null;license_type="Terminated";break}if(license.id!="000000"){modal.scope.set("$sn",license.id);form.input("SN","$sn").setReadonly(true)}modal.scope.set("$type",license_type);form.input("Type","$type").setReadonly(true);if(license.email!="DEMO"){modal.scope.set("$to",license.email+" ("+license.company+")");form.input("Licensed To","$to").setReadonly(true)}if(!!expire_title){modal.scope.set("$expire",time2ddmmmyyyy(license.expire));form.input(expire_title,"$expire").setReadonly(true)}form.hr();form.input("Activate License","license");form.hr();var btnOk=$.element.button("Activate & Restart").addClass("submit").on("click",(function(){modal.submit()}));var btnCancel=$.element.button("Cancel").on("click",(function(){modal.remove()}));var btnPurchase;if(license.type==1||license.type==99){btnPurchase=$.element.button("Buy Now!")}else if(license.type==3&&license.expire*1e3>Date.now()){}else if(license.type==4){}else{btnPurchase=$.element.button("Renew")}if(btnPurchase){btnPurchase.on("click",(function(){if(license.type==99||license.id=="000000")window.open("https://cesbo.com/astra-license","_blank").focus();else window.open("https://cesbo.com/astra-renew?sn="+license.id,"_blank").focus()}))}form.submit().addChild(btnOk,btnCancel,btnPurchase)};SettingsLicenseModule.click=function(){var appHost=app.hosts[location.host],scope={};$.modal().bindScope(SettingsLicenseModule.render,scope,appHost).on("submit",(function(){var license=this.scope.get("license");if(license)appHost.request({cmd:"set-license",license:license},(function(data){console.log(data);if(data.data["set-license"]!="ok")$.err({title:"Failed to save license"});else $.msg({title:"License saved",text:"Restart required"})}),(function(){$.err({title:"set-license: failed to call API"})}))}))};app.modules.push(SettingsLicenseModule);app.settings.push(SettingsLicenseModule)})();

(function(){"use strict";var SettingsRestartModule={label:"Restart",order:99};SettingsRestartModule.click=function(){var modal=$.modal();modal.addClass("main-menu");$.forEach(app.hosts,(function(appHost,hostId){var x=$.element("a").setText(appHost.name).addAttr("href","#");x.on("click",(function(event){event.preventDefault();modal.remove();appHost.restart()}));modal.addChild(x)}));modal.addChild($.element("hr"));modal.addChild($.element("a").setText("Cancel").addClass("text-center").addAttr("href","#/").on("click",(function(event){event.preventDefault();modal.remove()})))};app.modules.push(SettingsRestartModule);app.settings.push(SettingsRestartModule)})();

(function(){"use strict";window.LogModule={label:"Log",link:"#/log",order:9,search:true};LogModule.run=function(){LogModule.inner=$.element("div").addClass("log monospace");LogModule.log=[];LogModule.logSkip=0;LogModule.isReady=false;var appHost=app.hosts[location.host];appHost.request({cmd:"log"},(function(response){var data=response.data;if(data.log){var logData=[];Array.prototype.push.apply(logData,data.log);Array.prototype.push.apply(logData,LogModule.log);if(logData.length>2e3)logData.splice(0,logData.length-2e3);LogModule.log=logData;LogModule.logSkip=0}LogModule.isReady=true}));app.on("log_event",(function(event){if(event.data.log){Array.prototype.push.apply(LogModule.log,event.data.log);if(LogModule.log.length>2e3){var drop=LogModule.log.length-2e3;LogModule.log.splice(0,drop);LogModule.logSkip=Math.max(0,LogModule.logSkip-drop)}}if(event.data.set){appHost.sysinfo.log=event.data.set}}))};LogModule.render=function(){var self=this,object=app.renderInit(),appHost=app.hosts[location.host];var levelCheckbox=function(title,bind){return $.element("label").addClass("checkbox").addChild($.element("input").addAttr("type","checkbox").dataBind(bind).on("change",(function(event){var v=event.target.checked,s={};s[bind]=v;appHost.request({cmd:"set-log",set:s},(function(response){appHost.sysinfo.log[bind]=response.data[bind]}),(function(){self.set(bind,!v)}))}))).addChild($.element("span").addClass("inner")).addChild(title)};var logClear=function(){LogModule.inner.empty();LogModule.log=[];LogModule.logSkip=0};object.addChild(LogModule.inner).addChild($.element("div").addClass("log-footer row").addChild(levelCheckbox("Debug","debug")).addChild(levelCheckbox("Info","info")).addChild($.element("div").addClass("col-expand")).addChild($.element.button("Clear").on("click",logClear)));var search="";var autoScroll=true;var logGrep=function(tail){var il=LogModule.inner.childNodes,l=il.length,i=!tail?0:l-tail;for(;i<l;++i){var e=il[i].removeClass("hide");if(search&&e.lastChild.textContent.toLowerCase().indexOf(search)==-1)e.addClass("hide")}if(autoScroll)window.scrollTo(0,$.base.scrollHeight)};var logLimit=2e3;var stepLimit=100;var dateFormat=function(d){d=new Date(d);var dd=("0"+d.getDate()).slice(-2);var dm=monthMap[d.getMonth()];var th=("0"+d.getHours()).slice(-2);var tm=("0"+d.getMinutes()).slice(-2);var ts=("0"+d.getSeconds()).slice(-2);return dm+" "+dd+" "+th+":"+tm+":"+ts};var makeItem=function(e){var li=$.element("div").addClass("log-item log-"+e.type);var ld=$.element("div").addClass("log-time").setText(dateFormat(e.time*1e3));var lt=$.element("div").addClass("log-text").setText(e.text);li.addChild(ld).addChild(lt);LogModule.inner.addChild(li);if(LogModule.inner.childNodes.length>logLimit)LogModule.inner.firstChild.remove()};var cacheTimeout=null;var cacheUpdate=function(){if(!LogModule.isReady){cacheTimeout=setTimeout(cacheUpdate,200);return}if(LogModule.logSkip==LogModule.log.length){cacheTimeout=setTimeout(cacheUpdate,1e3);return}var ds=Math.min(LogModule.log.length-LogModule.logSkip,stepLimit);var dc=LogModule.logSkip+ds;for(var i=LogModule.logSkip;i<dc;++i)makeItem(LogModule.log[i]);LogModule.logSkip+=ds;logGrep(ds);cacheTimeout=setTimeout(cacheUpdate,50)};cacheUpdate();logGrep();app.search=function(value){search=value||"";search=search.toLowerCase();if(!search)autoScroll=true;logGrep()};var autoScrollCheck=function(){autoScroll=window.innerHeight+window.scrollY>=$.base.offsetHeight};window.addEventListener("scroll",autoScrollCheck);self.on("destroy",(function(){window.removeEventListener("scroll",autoScrollCheck);clearTimeout(cacheTimeout)}));self.on("ready",(function(){document.querySelector(".search").focus()}))};LogModule.init=function(){var h=app.hosts[location.host];var s={debug:h.sysinfo.log.debug,info:h.sysinfo.log.info};$.body.bindScope(LogModule.render,s)};app.modules.push(LogModule);app.menu.push(LogModule)})();

]==]) astra_storage["/app.css"] = base64.decode([==[ /*
 * Astra: Web Interface
 * https://cesbo.com/astra/
 *
 * Copyright (C) 2020, Andrey Dyldin <and@cesbo.com>
 */

/* pony.css */

:root {
	--main-bg-color: #ffffff;
	--text-color: #212121;
	--info-color: #31708f;
	--info-bg-color: #d9edf7;
	--error-color: #b71c1c;
	--error-bg-color: #f2dede;
	--border-color: #cccccc;
	--backdrop-color: #000000;
	--link-color: #519ed4;
	--hover-bg-color: #f5f5f5;
}

.dark {
	--main-bg-color: #252830;
	--text-color: #cfd2da;
	--info-color: #31708f;
	--info-bg-color: #d9edf7;
	--error-color: #b71c1c;
	--error-bg-color: #f2dede;
	--border-color: #434857;
	--backdrop-color: #ffffff;
	--link-color: #519ed4;
	--hover-bg-color: #434857;
}

html {
	margin: 0;
	padding: 0;
	display: block;
	height: 0;
	-ms-text-size-adjust: 100%;
	-webkit-text-size-adjust: 100%;
}

body {
	margin: 0;
	padding: 0;
	display: block;
	font: normal 14px "Helvetica Neue", Helvetica, Arial, sans-serif;
	color: var(--text-color);
}

body,
div {
	background-color: var(--main-bg-color);
}

*,
*:before,
*:after {
	-moz-box-sizing: border-box;
	-webkit-box-sizing: border-box;
	box-sizing: border-box;
}

*[disabled] {
	cursor: not-allowed !important;
}

.hide {
	display: none !important;
}

/* MENU */

.main-menu {
	display: -webkit-flex;
	display: flex;
	border-bottom: 1px solid var(--border-color);
	z-index: 1000;
	overflow-x: auto;
}

.main-menu>* {
	display: inline-block;
	-webkit-flex: 0 0 auto;
	flex: 0 0 auto;
	font-size: 16px;
	padding: 0 10px;
	font-weight: 200;
	line-height: 30px;
}

.main-menu>a {
	text-decoration: none;
	outline: none;
}

.main-menu>a:hover,
.main-menu>a:focus,
.main-menu>a.active {
	background-color: var(--hover-bg-color);
	text-decoration: none;
}

.main-menu>a.active {
	cursor: default;
}

.main-menu>.search {
	-webkit-flex: 2 0 auto;
	flex: 2 0 auto;
}

.main-menu>.search.input {
	font-size: 16px;
	font-weight: 200;
	outline: none;
	border: none;
	height: 30px;
	width: auto;
}

.main-menu>.search.input:focus {
	box-shadow: none;
}

.main-menu>hr {
	padding: 0;
}

.main-menu.fixed {
	position: fixed;
	top: 0;
	left: 0;
	width: 100%;
}

.main-menu.fixed+.main-content {
	padding-top: 30px;
}

.main-menu.modal {
	-webkit-flex-direction: column;
	flex-direction: column;
}

.main-menu.modal>* {
	padding: 0 24px;
}

.main-content {
	padding-left: 10px;
	padding-right: 10px;
}

/* MODAL */

.modal-backdrop {
	position: fixed;
	top: 0;
	right: 0;
	bottom: 0;
	left: 0;
	z-index: 1040;
	background-color: var(--backdrop-color);
	opacity: 0.1;
}

.modal-wrap {
	position: fixed;
	top: 0;
	right: 0;
	bottom: 0;
	left: 0;
	z-index: 1041;
	background-color: transparent !important;
	overflow: auto;
}

.modal-open {
	overflow: hidden;
}

.modal {
	border: 1px solid var(--border-color);
	width: 50%;
	padding: 20px 10px;
	margin: 40px auto;
}

@media (max-width: 768px) {
	.modal {
		margin: 0;
		width: 100%;
		min-height: 100%;
	}
}

/* GRID */

.row {
	display: -webkit-flex;
	display: flex;
	-webkit-align-items: center;
	align-items: center;
	width: 100%;
	background-color: transparent;
}

.row.top {
	-webkit-align-items: flex-start;
	align-items: flex-start;
}

.row.center {
	-webkit-justify-content: center;
	justify-content: center;
}

.row:before,
.row:after {
	display: table;
	content: " ";
}

.row:after {
	clear: both;
}

.row>* {
	display: inline-block;
	-webkit-flex: 0 0 auto;
	flex: 0 0 auto;
	position: relative;
}

.row>.col-expand {
	-webkit-flex: 2 0 auto;
	flex: 2 0 auto;
}

.row>.col-1 {
	width: 25%;
}

.row>.col-2 {
	width: 50%;
}

.row>.col-3 {
	width: 75%;
}

.row>.col-4 {
	width: 100%;
}

.float-left {
	float: left;
}

.float-right {
	float: right;
}

/* FORMS */

hr {
	border: none;
	border-bottom: 1px solid var(--border-color);
	margin: 20px 0;
}

.label {
	font-weight: bold;
}

.label.required:after {
	content: "*";
	color: var(--error-color);
	font-weight: 200;
	margin-left: 3px;
	margin-top: -3px;
	font-size: 18px;
	position: absolute;
	height: 1px;
}

.label.danger,
.label.error {
	color: var(--error-color);
}

.input {
	display: block;
	width: 100%;
	padding: 0 12px;
	border: 1px solid var(--border-color);
	background-color: var(--main-bg-color);
	outline: none;
	height: 36px;
	border-radius: 0;
	-webkit-appearance: none;
	appearance: none;
	font: inherit;
	color: inherit;
}

.input.error {
	border-color: var(--error-color);
}

.input::-webkit-outer-spin-button,
.input::-webkit-inner-spin-button {
	-webkit-appearance: none; margin: 0;
}

select.input {
	border-radius: 0;
	background-image: url('data:image/svg+xml;utf8,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 140 110"><path stroke="%23555555" fill="none" stroke-width="15" d="m10,10 60,80 60,-80"/></svg>');
	background-size: 18px;
	background-position: right 10px center;
	background-repeat: no-repeat;
	padding-right: 40px;
	-webkit-appearance: none;
	appearance: none;
}

@-moz-document url-prefix() {
	select.input {
		background: none;
	}
}

textarea.input {
	-webkit-overflow-scrolling: touch;
}

.radio,
.checkbox {
	display: block;
	width: 100%;
	cursor: pointer;
	position: relative;
	line-height: 16px;
}

.radio>*,
.checkbox>* {
	display: inline-block;
	line-height: 16px;
}

.radio>input,
.checkbox>input {
	margin: 0 8px 0 0;
	cursor: pointer;
	opacity: 0;
}

.checkbox>input+.inner,
.radio>input+.inner {
	width: 14px;
	height: 14px;
	border: 1px solid var(--border-color);
	position: absolute;
	left: 0px;
	top: 1px;
}

.checkbox>input:checked+.inner:before,
.radio>input:checked+.inner:before {
	content: " ";
	display: block;
	border: 2px solid var(--main-bg-color);
	width: 100%;
	height: 100%;
	background-color: var(--link-color);
}

.checkbox.danger>.text {
	color: var(--error-color);
}

.checkbox.danger>input:checked+.inner:before {
	background-color: var(--error-color);
}

.radio>input+.inner {
	border-radius: 100%;
}

.radio>input:checked+.inner:before {
	border-radius: 100%;
}

.input:focus,
.checkbox>input:focus+.inner,
.radio>input:focus+.inner {
	box-shadow: 0 0 3px var(--border-color);
	z-index: 2;
}

.button {
	cursor: pointer;
	padding: 0 16px;
	margin: 0 6px;
	border: none;
	outline: none;
	height: 36px;
	background-color: transparent;
	font: inherit;
	color: inherit;
	font-weight: 600;
	text-transform: uppercase;
	-moz-appearance: none;
	-webkit-appearance: none;
	appearance: none;
}

.button:hover,
.button:focus {
	background-color: var(--hover-bg-color);
}

.button.submit {
	color: var(--link-color);
}

.button.danger {
	color: var(--error-color);
}

.button[disabled] {
	color: var(--border-color) !important;
	cursor: not-allowed;
	background-color: transparent !important;
}

.input[disabled] {
	cursor: not-allowed;
	border-style: dotted;
}

/* PROGRESS */

.progress {
	display: -webkit-flex;
	display: flex;
	width: 100%;
	position: relative;
}

.progress>* {
	display: inline-block;
	height: 100%;
}

.progress>.progress-level {
	-webkit-flex: 0 0 auto;
	flex: 0 0 auto;
	background-color: transparent;
}

.progress>.progress-overlay {
	-webkit-flex: 2 0 auto;
	flex: 2 0 auto;
	background-color: var(--hover-bg-color);
}

/* TEXT */

a {
	text-decoration: none;
	color: var(--link-color);
	outline: none;
}

a:hover,
a:focus,
a.active {
	text-decoration: underline;
}

code,
kbd,
pre,
samp,
.monospace {
	font-family: "Lucida Console", Monaco, monospace;
}

.text-left {
	text-align: left;
}

.text-center {
	text-align: center;
}

.text-right {
	text-align: right;
}

.text-gray {
	opacity: 0.5;
}

.text-delete {
	text-decoration: line-through;
}

.text-italic {
	font-style: italic;
}

/* ALERT */

.alert-float {
	position: fixed;
	z-index: 2001;
	right: 10px;
	bottom: 10px;
	background: transparent !important;
}

.alert-float>.alert {
	margin-bottom: 5px;
	width: 300px;
	padding: 10px;
	border: 1px solid transparent;
}

.alert-float>.alert.alert-info {
	color: var(--info-color);
	background-color: var(--info-bg-color);
	border-color: var(--info-color);
}

.alert-float>.alert.alert-error {
	color: var(--error-color);
	background-color: var(--error-bg-color);
	border-color: var(--error-color);
}

.alert-float>.alert>.alert-text {
	margin: 0;
}

.alert-float>.alert>.alert-title+.alert-text {
	margin-top: 5px;
}

.alert-float>.alert .button {
	margin: 5px 0 0 0;
}

@media (max-width: 768px) {
	.alert-float {
		left: 10px;
	}

	.alert-float>.alert {
		width: 100%;
	}
}

/* TABLE */

.table {
	width: 100%;
	border-spacing: 0;
}

.table tr>* {
	height: 32px;
	padding: 0 5px;
}

.table tr>th {
	text-align: left;
	position: relative;
}

.table>thead>tr>th[data-order] {
	cursor: pointer;
}

.table>thead>tr>th[data-order]:before,
.table>thead>tr>th[data-order]:after {
	content: '';
	display: block;
	position: absolute;
	right: 10px;
	border-color: transparent;
	border-style: solid;
	border-width: 5px;
	width: 0;
	height: 0;
}

.table>thead>tr>th[data-order]:before {
	border-bottom-color: var(--hover-color);
	top: 5px;
}

.table>thead>tr>th[data-order]:after {
	border-top-color: var(--hover-color);
	top: 17px;
}

.table>thead>tr>th[data-order="-1"]:before {
	border-bottom-color: var(--text-color);
}

.table>thead>tr>th[data-order="1"]:after {
	border-top-color: var(--text-color);
}

.table.hover>tbody>tr {
	cursor: pointer;
}

.table.hover>tbody>tr:hover {
	background-color: var(--hover-color);
}

/* app.css */

html, body { min-height: 100%; }

.version {
	position: fixed;
	bottom: 2px;
	left: 10px;
	text-transform: uppercase;
	color: #bdbdbd;
	font-size: 10px;
	background: none;
}

/* Loading */

.loading { position: relative; left: 50%; height: 5px; text-align: center; margin: 10px 0; width: 0; }
.loading:after { content: ""; display: table; clear: both; }
.loading .bullet { position: absolute; padding: 5px; border-radius: 50%; background: var(--link-color); -webkit-animation: animIn 1s ease-in-out 0s infinite; animation: animIn 1s ease-in-out 0s infinite; }
.loading .bullet:nth-child(1) { -webkit-animation-delay: 0s; animation-delay: 0s; }
.loading .bullet:nth-child(2) { -webkit-animation-delay: 0.15s; animation-delay: 0.15s; }
.loading .bullet:nth-child(3) { -webkit-animation-delay: 0.3s; animation-delay: 0.3s; }
.loading .bullet:nth-child(4) { -webkit-animation-delay: 0.45s; animation-delay: 0.45s; }
@-webkit-keyframes animIn {
	0% { -webkit-transform: translateX(-50px); transform: translateX(-50px); opacity: 0; }
	50% { opacity: 1; }
	100% { -webkit-transform: translateX(50px); transform: translateX(50px); opacity: 0; }
}
@keyframes animIn {
	0% { -webkit-transform: translateX(-50px); transform: translateX(-50px); opacity: 0; }
	50% { opacity: 1; }
	100% { -webkit-transform: translateX(50px); transform: translateX(50px); opacity: 0; }
}

/* Awaiting Checkbox */

.checkbox.awaiting>.inner { -webkit-animation: checkboxSpin 1.2s ease-in-out 0s infinite; animation: checkboxSpin 1.2s ease-in-out 0s infinite; }

@-webkit-keyframes checkboxSpin {
	0% { -webkit-transform: rotate(0); opacity: 1; }
	50% { opacity: 0; }
	100% { -webkit-transform: rotate(360deg); opacity: 1; }
}

@keyframes checkboxSpin {
	0% { transform: rotate(0); opacity: 1; }
	50% { opacity: 0; }
	100% { transform: rotate(360deg); opacity: 1; }
}

/* Auth */

.auth { max-width: 400px; padding: 50px 10px 0 10px; margin: 0 auto; }
.auth>* { margin: 0; }
.auth>.input { position: relative; height: auto; padding: 10px 12px; font-size: 16px; }
.auth>.input[type="password"] { margin-top: -1px; margin-bottom: 10px; }
.auth>.checkbox { margin: 10px 0; }
.auth>.button { width: 100%; margin: 0; }

/* Forms */

.form-group {
	margin-bottom: 20px;
	position: relative;
	width: 100%;
	max-width: 768px;
	margin-left: auto;
	margin-right: auto;
}

.form-group:last-child {
	margin-bottom: 0;
}

.form-group-header {
	margin-bottom: 10px;
	position: relative;
	width: 100%;
	max-width: 768px;
	margin-left: auto;
	margin-right: auto;

	color: var(--border-color);
	text-transform: uppercase;
	font-weight: 600;
	font-size: 12px;
	letter-spacing: 1px;
	line-height: 32px;
}

.form-group-action {
	float: right;
	text-decoration: none;
	color: #66bb6a;
}

.form-group-action:hover,
.form-group-action:focus {
	text-shadow: 0 0 3px var(--border-color);
	z-index: 2;
	text-decoration: none;
}

.form-submit {
	text-align: center;
}

.form-group[data-label]:before {
	content: attr(data-label);
	position: absolute;
	left: 4px;
	top: -7px;
	font-size: 12px;
	text-transform: uppercase;
	font-weight: 600;
	background-color: var(--main-bg-color);
	padding: 0 2px 0 4px;
	color: var(--border-color);
	letter-spacing: 1px;
}

.form-group[data-label].error:before {
	color: var(--error-color);
}

/* Tab */

.tab {
	list-style-type: none;
	margin: 0;
	padding: 0;
	overflow: hidden;
	display: -webkit-flex;
	display: flex;
	justify-content: center;
}

.tab>* {
	display: inline-block;
	-webkit-flex: 0 0 auto;
	flex: 0 0 auto;
	padding: 0 10px;
	font-weight: 200;
	line-height: 30px;
	font-size: 16px;
}

/* Controls */

.main-content>.form { padding: 20px 0; }

.io-wizard { position: relative; }
.io-wizard input { padding-right: 68px; }
.io-wizard .button-wrap {
	position: absolute;
	right: 2px;
	top: 2px;
	bottom: 2px;
	z-index: 5;
	white-space: nowrap;
}

.icon {
	display: inline-block;
	width: 100%;
	height: 100%;
	background-position: center;
	background-repeat: no-repeat;
}

.button.icon {
	height: 32px;
	width: 32px;
	border-radius: 50%;
	padding: 0;
	margin: 0;
	overflow: hidden;
	background-size: 28px;
	vertical-align: middle;
}

.button.icon:hover,
.button.icon:focus {
	z-index: 2;
}

.button.icon.small {
	height: 24px;
	width: 24px;
	background-size: 20px;
}

select.button.icon {
	color: transparent;
}

select.button.icon>option {
	text-transform: none;
	color: var(--text-color);
}

.icon-settings {
	background-image: url('data:image/svg+xml,<svg fill="%23519ed4" height="32" viewBox="0 0 24 24" width="32" xmlns="http://www.w3.org/2000/svg"><path d="M0 0h24v24H0z" fill="none"/><path d="M19.43 12.98c.04-.32.07-.64.07-.98s-.03-.66-.07-.98l2.11-1.65c.19-.15.24-.42.12-.64l-2-3.46c-.12-.22-.39-.3-.61-.22l-2.49 1c-.52-.4-1.08-.73-1.69-.98l-.38-2.65C14.46 2.18 14.25 2 14 2h-4c-.25 0-.46.18-.49.42l-.38 2.65c-.61.25-1.17.59-1.69.98l-2.49-1c-.23-.09-.49 0-.61.22l-2 3.46c-.13.22-.07.49.12.64l2.11 1.65c-.04.32-.07.65-.07.98s.03.66.07.98l-2.11 1.65c-.19.15-.24.42-.12.64l2 3.46c.12.22.39.3.61.22l2.49-1c.52.4 1.08.73 1.69.98l.38 2.65c.03.24.24.42.49.42h4c.25 0 .46-.18.49-.42l.38-2.65c.61-.25 1.17-.59 1.69-.98l2.49 1c.23.09.49 0 .61-.22l2-3.46c.12-.22.07-.49-.12-.64l-2.11-1.65zM12 15.5c-1.93 0-3.5-1.57-3.5-3.5s1.57-3.5 3.5-3.5 3.5 1.57 3.5 3.5-1.57 3.5-3.5 3.5z"/></svg>');
}

.icon-close {
	background-image: url('data:image/svg+xml,<svg fill="%23519ed4" height="32" viewBox="0 0 24 24" width="32" xmlns="http://www.w3.org/2000/svg"><path d="M0 0h24v24H0z" fill="none"/><path d="M14.59 8L12 10.59 9.41 8 8 9.41 10.59 12 8 14.59 9.41 16 12 13.41 14.59 16 16 14.59 13.41 12 16 9.41 14.59 8zM12 2C6.47 2 2 6.47 2 12s4.47 10 10 10 10-4.47 10-10S17.53 2 12 2zm0 18c-4.41 0-8-3.59-8-8s3.59-8 8-8 8 3.59 8 8-3.59 8-8 8z"/></svg>');
}

.icon-move {
	background-image: url('data:image/svg+xml,<svg fill="%23519ed4" height="32" viewBox="0 0 24 24" width="32" xmlns="http://www.w3.org/2000/svg"><path d="M16 17.01V10h-2v7.01h-3L15 21l4-3.99h-3zM9 3L5 6.99h3V14h2V6.99h3L9 3z"/><path d="M0 0h24v24H0z" fill="none"/></svg>');
}

.icon-more {
	background-image: url('data:image/svg+xml,<svg fill="%23519ed4" height="32" viewBox="0 0 24 24" width="32" xmlns="http://www.w3.org/2000/svg"><path d="M0 0h24v24H0z" fill="none"/><path d="M6 10c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm12 0c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2zm-6 0c-1.1 0-2 .9-2 2s.9 2 2 2 2-.9 2-2-.9-2-2-2z"/></svg>');
}

.icon-add {
	background-image: url('data:image/svg+xml,<svg fill="%23519ed4" height="32" viewBox="0 0 24 24" width="32" xmlns="http://www.w3.org/2000/svg"><path d="M19 13h-6v6h-2v-6H5v-2h6V5h2v6h6v2z"/><path d="M0 0h24v24H0z" fill="none"/></svg>');
}

/* Help */

.help {
	cursor: help;
}

/* Import/Edit */

.settings-edit { display: -webkit-flex; display: flex; -webkit-flex-direction: column; flex-direction: column; position: absolute; top: 30px; right: 0; bottom: 0; left: 0; }
.settings-edit .input { white-space: pre-wrap; word-wrap: break-word; outline: none; resize: none; border: 0; padding: 10px; -webkit-flex: 1 1 auto; flex: 1 1 auto; }
.settings-edit .input:focus { box-shadow: none; }

/* Cards */

.card-stack { text-align: center; margin: 5px -10px; }
.card-stack[data-header]:before { content: attr(data-header); color: var(--border-color); text-transform: uppercase; letter-spacing: 0.05em; font-weight: 600; font-size: 12px; letter-spacing: 1px; text-align: center; display: block; }

.card { border: 1px solid var(--border-color); margin: 3px; text-align: left; cursor: pointer; display: inline-block; vertical-align: top; position: relative; color: inherit; }
.card:hover, .card:focus { text-decoration: none; outline: none; }
.card .card-name, .card .card-status { white-space: nowrap; overflow: hidden; margin: 0 3px; }
.card .card-name { line-height: 24px; font-size: 16px; font-weight: 200; }
.card .card-action { position: absolute; top: 0; right: 0; }
.card .card-status { line-height: 18px; font-size: 14px; }
.card .card-footer { margin-top: 2px; border-top: 1px solid var(--border-color); }
.card .card-status .text { color: var(--border-color); display: block; }
.card .card-status .text.onair { color: #66bb6a; }
.card .card-status .text.error, .card .card-status .text.scrambled, .card .card-status .text.cc { color: #ef5350; }
.card .card-image { height: 100px; background-position: center; background-repeat: no-repeat; background-size: cover; margin: 0 3px; display: none; }

@media (max-width: 768px) {
	.card { margin: 1px; }
}

@keyframes card-inactive-bounce {
	0% { box-shadow: none; }
	50% { box-shadow: 0 0 5px var(--border-color); }
	100% { box-shadow: none; }
}
.card.card-true-3, .card.card-false-0 { animation: card-inactive-bounce 1s; }
.card.card-false-0 { border-style: dotted; }
.card.card-false-0 .card-name { color: var(--border-color); }

@keyframes card-onair-bounce {
	0% { box-shadow: none; }
	50% { box-shadow: 0 0 5px #66bb6a; }
	100% { box-shadow: none; }
}
.card.card-true-2 { border-color: #66bb6a; animation: card-onair-bounce 1s; }

@keyframes card-error-bounce {
	0% { box-shadow: none; }
	50% { box-shadow: 0 0 5px #ef5350; }
	100% { box-shadow: none; }
}
.card.card-true-0, .card.card-true-1 { border-color: #ef5350; animation: card-error-bounce 1s; }

@keyframes card-selected-bounce {
	0% { box-shadow: none; }
	50% { box-shadow: 0 0 5px #42a5f5; }
	100% { box-shadow: none; }
}
.card.selected, .card:focus { border-color: #42a5f5; animation: card-selected-bounce 1s; }
.card.selected:hover { border-color: #1565c0; }
.card.selected:before { content: " "; float: right; border-top: 10px solid #42a5f5; border-left: 10px solid transparent; }
.card.selected:hover:before { border-top-color: #1565c0; }

.card.updated:before { content: "\25cf"; color: #42a5f5; font-size: 24px; position: absolute; right: 5px; top: -5px; }

.card.stream-mpts .card-status .text[data-name]:after { content: attr(data-name); padding-left: 5px; }

.card-hide-disabled .card.card-false-0 { display: none; }
.card-hide-inactive .card.card-true-3 { display: none; }
.card-hide-wo-error .card.card-true-2 { display: none; }

.signal-level {
	background: url('data:image/svg+xml;utf8,<svg width="100%" height="16" xmlns="http://www.w3.org/2000/svg"><defs><linearGradient id="g1"><stop stop-color="%23F44336" offset="0%"/><stop stop-color="%23FFFF00" offset="25%"/><stop stop-color="%2300E676" offset="75%"/><stop stop-color="%2300E676" offset="100%"/></linearGradient></defs><rect fill="url(%23g1)" x="0" y="0" width="100%" height="100%"/></svg>');
	background-repeat: no-repeat;
	height: 16px;
}

.dvb-scan .dvb-status { margin-top: 5px; }

.dvb-status { width: auto; }
.dvb-status>*:nth-child(1) { width: 10px; }
.dvb-status>*:nth-child(2) { width: auto; }
.dvb-status>*:nth-child(3) { width: 30px; text-align: right; }

.info-status { text-align: left; }
.info-status>* { white-space: pre; padding-right: 10px; }
.info-status>*:last-child { float: right; padding-right: 0; }
.info-status>*.er { color: #ef5350; }
.info-status>*.ok { color: #66bb6a; }

@media (min-width: 768px) {
	.info-status>*:nth-child(1):before { content: "SIGNAL"; }
	.info-status>*:nth-child(2):before { content: "CARRIER"; }
	.info-status>*:nth-child(3):before { content: "FEC"; }
	.info-status>*:nth-child(4):before { content: "SYNC"; }
	.info-status>*:nth-child(5):before { content: "LOCK"; }
}

@media (max-width: 768px) {
	.info-status>*:nth-child(1) { padding-right: 0; }
	.info-status>*:nth-child(2) { padding-right: 0; }
	.info-status>*:nth-child(3) { padding-right: 0; }
	.info-status>*:nth-child(4) { padding-right: 0; }
	.info-status>*:nth-child(1):before { content: "S"; }
	.info-status>*:nth-child(2):before { content: "C"; }
	.info-status>*:nth-child(3):before { content: "V"; }
	.info-status>*:nth-child(4):before { content: "Y"; }
	.info-status>*:nth-child(5):before { content: "L"; }
}

/* */

.table .action {
	text-align: right;
	width: 0;
	padding: 0;
}

/* Help */

.modal.help {
	padding: 20px 0;
	margin: 0;
	width: auto;
	position: fixed;
	top: 40px;
	right: 40px;
	bottom: 40px;
	left: 40px;
	display: flex;
	flex-direction: column;
}

.modal.help>iframe {
	border: none;
	flex: 1;
}

@media (max-width: 768px) {
	.modal.help {
		margin: 0;
		top: 0;
		right: 0;
		bottom: 0;
		left: 0;
	}
}

/* Log */

.log { margin: 0; padding: 5px 0 40px 0; }
.log .log-item { white-space: nowrap; margin: 2px 0; }
.log .log-item.log-0 { color: #66bb6a; }
.log .log-item.log-1 { color: #ef5350; }
.log .log-item.log-2 { color: #ffca28; }
.log .log-item.log-3 { color: #777; }
.log .log-item>* { display: inline-block; white-space: pre; }
.log .log-item>.log-time { margin-right: 5px; }
.log .log-item>.log-text { margin-left: 5px; }
.log-footer { background-color: inherit; padding-left: 10px; position: fixed; right: 0; bottom: 0; left: 0; }
.log-footer .button { margin: 0; }
.log-footer .checkbox { display: inline-block; width: auto; margin-right: 10px; }

]==])