custom.js 73 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400
  1. var he = require('he');
  2. var Promise = require('es6-promise').Promise;
  3. window.bootstrap = require('bootstrap');
  4. import Cookies from 'js-cookie';
  5. if (!String.prototype.format) {
  6. Object.assign(String.prototype, {
  7. format() {
  8. const args = arguments;
  9. return this.replace(/{(\d+)}/g, function (match, number) {
  10. return typeof args[number] !== 'undefined' ? args[number] : match;
  11. });
  12. },
  13. });
  14. }
  15. if (!String.prototype.encodeHTML) {
  16. Object.assign(String.prototype, {
  17. encodeHTML() {
  18. return he.encode(this).replace(/\n/g, '<br />');
  19. },
  20. });
  21. }
  22. Object.assign(Date.prototype, {
  23. toLocalShort() {
  24. const opt = { dateStyle: 'short', timeStyle: 'short' };
  25. return this.toLocaleString(undefined, opt);
  26. },
  27. });
  28. function get_control_option_value(obj) {
  29. let ctrl,id,val,opt;
  30. let radio = false;
  31. let checked = false;
  32. if (typeof (obj) === 'string') {
  33. id = obj;
  34. ctrl = $(`#${id}`);
  35. } else {
  36. id = $(obj).attr('id');
  37. ctrl = $(obj);
  38. }
  39. if(ctrl.attr('type') === 'checkbox'){
  40. opt = $(obj).checked?id.replace('cmd_opt_', ''):'';
  41. val = true;
  42. }
  43. else {
  44. opt = id.replace('cmd_opt_', '');
  45. val = $(obj).val();
  46. val = `${val.includes(" ") ? '"' : ''}${val}${val.includes(" ") ? '"' : ''}`;
  47. }
  48. return { opt, val };
  49. }
  50. function handleNVSVisible() {
  51. let nvs_previous_checked = isEnabled(Cookies.get("show-nvs"));
  52. $('input#show-nvs')[0].checked = nvs_previous_checked;
  53. if ($('input#show-nvs')[0].checked || recovery) {
  54. $('*[href*="-nvs"]').show();
  55. } else {
  56. $('*[href*="-nvs"]').hide();
  57. }
  58. }
  59. function concatenateOptions(options) {
  60. let commandLine = ' ';
  61. for (const [option, value] of Object.entries(options)) {
  62. if (option !== 'n' && option !== 'o') {
  63. commandLine += `-${option} `;
  64. if (value !== true) {
  65. commandLine += `${value} `;
  66. }
  67. }
  68. }
  69. return commandLine;
  70. }
  71. function isEnabled(val) {
  72. return val != undefined && typeof val === 'string' && val.match("[Yy1]");
  73. }
  74. const nvsTypes = {
  75. NVS_TYPE_U8: 0x01,
  76. /*! < Type uint8_t */
  77. NVS_TYPE_I8: 0x11,
  78. /*! < Type int8_t */
  79. NVS_TYPE_U16: 0x02,
  80. /*! < Type uint16_t */
  81. NVS_TYPE_I16: 0x12,
  82. /*! < Type int16_t */
  83. NVS_TYPE_U32: 0x04,
  84. /*! < Type uint32_t */
  85. NVS_TYPE_I32: 0x14,
  86. /*! < Type int32_t */
  87. NVS_TYPE_U64: 0x08,
  88. /*! < Type uint64_t */
  89. NVS_TYPE_I64: 0x18,
  90. /*! < Type int64_t */
  91. NVS_TYPE_STR: 0x21,
  92. /*! < Type string */
  93. NVS_TYPE_BLOB: 0x42,
  94. /*! < Type blob */
  95. NVS_TYPE_ANY: 0xff /*! < Must be last */,
  96. };
  97. const btIcons = {
  98. bt_playing: { 'label': '', 'icon': 'media_bluetooth_on' },
  99. bt_disconnected: { 'label': '', 'icon': 'media_bluetooth_off' },
  100. bt_neutral: { 'label': '', 'icon': 'bluetooth' },
  101. bt_connecting: { 'label': '', 'icon': 'bluetooth_searching' },
  102. bt_connected: { 'label': '', 'icon': 'bluetooth_connected' },
  103. bt_disabled: { 'label': '', 'icon': 'bluetooth_disabled' },
  104. play_arrow: { 'label': '', 'icon': 'play_circle_filled' },
  105. pause: { 'label': '', 'icon': 'pause_circle' },
  106. stop: { 'label': '', 'icon': 'stop_circle' },
  107. '': { 'label': '', 'icon': '' }
  108. };
  109. const batIcons = [
  110. { icon: "battery_0_bar", label: '▪', ranges: [{ f: 5.8, t: 6.8 }, { f: 8.8, t: 10.2 }] },
  111. { icon: "battery_2_bar", label: '▪▪', ranges: [{ f: 6.8, t: 7.4 }, { f: 10.2, t: 11.1 }] },
  112. { icon: "battery_3_bar", label: '▪▪▪', ranges: [{ f: 7.4, t: 7.5 }, { f: 11.1, t: 11.25 }] },
  113. { icon: "battery_4_bar", label: '▪▪▪▪', ranges: [{ f: 7.5, t: 7.8 }, { f: 11.25, t: 11.7 }] }
  114. ];
  115. const btStateIcons = [
  116. { desc: 'Idle', sub: ['bt_neutral'] },
  117. { desc: 'Discovering', sub: ['bt_connecting'] },
  118. { desc: 'Discovered', sub: ['bt_connecting'] },
  119. { desc: 'Unconnected', sub: ['bt_disconnected'] },
  120. { desc: 'Connecting', sub: ['bt_connecting'] },
  121. {
  122. desc: 'Connected',
  123. sub: ['bt_connected', 'play_arrow', 'bt_playing', 'pause', 'stop'],
  124. },
  125. { desc: 'Disconnecting', sub: ['bt_disconnected'] },
  126. ];
  127. const pillcolors = {
  128. MESSAGING_INFO: 'badge-success',
  129. MESSAGING_WARNING: 'badge-warning',
  130. MESSAGING_ERROR: 'badge-danger',
  131. };
  132. const connectReturnCode = {
  133. OK: 0,
  134. FAIL: 1,
  135. DISC: 2,
  136. LOST: 3,
  137. RESTORE: 4,
  138. ETH: 5
  139. }
  140. const taskStates = {
  141. 0: 'eRunning',
  142. /*! < A task is querying the state of itself, so must be running. */
  143. 1: 'eReady',
  144. /*! < The task being queried is in a read or pending ready list. */
  145. 2: 'eBlocked',
  146. /*! < The task being queried is in the Blocked state. */
  147. 3: 'eSuspended',
  148. /*! < The task being queried is in the Suspended state, or is in the Blocked state with an infinite time out. */
  149. 4: 'eDeleted',
  150. };
  151. let flashState = {
  152. NONE: 0,
  153. REBOOT_TO_RECOVERY: 2,
  154. SET_FWURL: 5,
  155. FLASHING: 6,
  156. DONE: 7,
  157. UPLOADING: 8,
  158. ERROR: 9,
  159. UPLOADCOMPLETE: 10,
  160. _state: -1,
  161. olderRecovery: false,
  162. statusText: '',
  163. flashURL: '',
  164. flashFileName: '',
  165. statusPercent: 0,
  166. Completed: false,
  167. recovery: false,
  168. prevRecovery: false,
  169. updateModal: new bootstrap.Modal(document.getElementById('otadiv'), {}),
  170. reset: function () {
  171. this.olderRecovery = false;
  172. this.statusText = '';
  173. this.statusPercent = -1;
  174. this.flashURL = '';
  175. this.flashFileName = undefined;
  176. this.UpdateProgress();
  177. $('#rTable tr.release').removeClass('table-success table-warning');
  178. $('.flact').prop('disabled', false);
  179. $('#flashfilename').value = null;
  180. $('#fw-url-input').value = null;
  181. if (!this.isStateError()) {
  182. $('span#flash-status').html('');
  183. $('#fwProgressLabel').parent().removeClass('bg-danger');
  184. }
  185. this._state = this.NONE
  186. return this;
  187. },
  188. isStateUploadComplete: function () {
  189. return this._state == this.UPLOADCOMPLETE;
  190. },
  191. isStateError: function () {
  192. return this._state == this.ERROR;
  193. },
  194. isStateNone: function () {
  195. return this._state == this.NONE;
  196. },
  197. isStateRebootRecovery: function () {
  198. return this._state == this.REBOOT_TO_RECOVERY;
  199. },
  200. isStateSetUrl: function () {
  201. return this._state == this.SET_FWURL;
  202. },
  203. isStateFlashing: function () {
  204. return this._state == this.FLASHING;
  205. },
  206. isStateDone: function () {
  207. return this._state == this.DONE;
  208. },
  209. isStateUploading: function () {
  210. return this._state == this.UPLOADING;
  211. },
  212. init: function () {
  213. this._state = this.NONE;
  214. return this;
  215. },
  216. SetStateError: function () {
  217. this._state = this.ERROR;
  218. $('#fwProgressLabel').parent().addClass('bg-danger');
  219. return this;
  220. },
  221. SetStateNone: function () {
  222. this._state = this.NONE;
  223. return this;
  224. },
  225. SetStateRebootRecovery: function () {
  226. this._state = this.REBOOT_TO_RECOVERY;
  227. // Reboot system to recovery mode
  228. this.SetStatusText('Starting recovery mode.')
  229. $.ajax({
  230. url: '/recovery.json',
  231. context: this,
  232. dataType: 'text',
  233. method: 'POST',
  234. cache: false,
  235. contentType: 'application/json; charset=utf-8',
  236. data: JSON.stringify({
  237. timestamp: Date.now(),
  238. }),
  239. error: function (xhr, _ajaxOptions, thrownError) {
  240. this.setOTAError(`Unexpected error while trying to restart to recovery. (status=${xhr.status ?? ''}, error=${thrownError ?? ''} ) `);
  241. },
  242. complete: function (response) {
  243. this.SetStatusText('Waiting for system to boot.')
  244. },
  245. });
  246. return this;
  247. },
  248. SetStateSetUrl: function () {
  249. this._state = this.SET_FWURL;
  250. this.statusText = 'Sending firmware download location.';
  251. let confData = {
  252. fwurl: {
  253. value: this.flashURL,
  254. type: 33,
  255. }
  256. };
  257. post_config(confData);
  258. return this;
  259. },
  260. SetStateFlashing: function () {
  261. this._state = this.FLASHING;
  262. return this;
  263. },
  264. SetStateDone: function () {
  265. this._state = this.DONE;
  266. this.reset();
  267. return this;
  268. },
  269. SetStateUploading: function () {
  270. this._state = this.UPLOADING;
  271. return this.SetStatusText('Sending file to device.');
  272. },
  273. SetStateUploadComplete: function () {
  274. this._state = this.UPLOADCOMPLETE;
  275. return this;
  276. },
  277. isFlashExecuting: function () {
  278. return true === (this._state != this.UPLOADING && (this.statusText !== '' || this.statusPercent >= 0));
  279. },
  280. toString: function () {
  281. let keys = Object.keys(this);
  282. return keys.find(x => this[x] === this._state);
  283. },
  284. setOTATargets: function () {
  285. this.flashURL = '';
  286. this.flashFileName = '';
  287. this.flashURL = $('#fw-url-input').val();
  288. let fileInput = $('#flashfilename')[0].files;
  289. if (fileInput.length > 0) {
  290. this.flashFileName = fileInput[0];
  291. }
  292. if (this.flashFileName.length == 0 && this.flashURL.length == 0) {
  293. this.setOTAError('Invalid url or file. Cannot start OTA');
  294. }
  295. return this;
  296. },
  297. setOTAError: function (message) {
  298. this.SetStateError().SetStatusPercent(0).SetStatusText(message).reset();
  299. return this;
  300. },
  301. ShowDialog: function () {
  302. if (!this.isStateNone()) {
  303. this.updateModal.show();
  304. $('.flact').prop('disabled', true);
  305. }
  306. return this;
  307. },
  308. SetStatusPercent: function (pct) {
  309. var pctChanged = (this.statusPercent != pct);
  310. this.statusPercent = pct;
  311. if (pctChanged) {
  312. if (!this.isStateUploading() && !this.isStateFlashing()) {
  313. this.SetStateFlashing();
  314. }
  315. if (pct == 100) {
  316. if (this.isStateFlashing()) {
  317. this.SetStateDone();
  318. }
  319. else if (this.isStateUploading()) {
  320. this.statusPercent = 0;
  321. this.SetStateFlashing();
  322. }
  323. }
  324. this.UpdateProgress().ShowDialog();
  325. }
  326. return this;
  327. },
  328. SetStatusText: function (txt) {
  329. var changed = (this.statusText != txt);
  330. this.statusText = txt;
  331. if (changed) {
  332. $('span#flash-status').html(this.statusText);
  333. this.ShowDialog();
  334. }
  335. return this;
  336. },
  337. UpdateProgress: function () {
  338. $('.progress-bar')
  339. .css('width', this.statusPercent + '%')
  340. .attr('aria-valuenow', this.statusPercent)
  341. .text(this.statusPercent + '%')
  342. $('.progress-bar').html((this.isStateDone() ? 100 : this.statusPercent) + '%');
  343. return this;
  344. },
  345. StartOTA: function () {
  346. this.logEvent(this.StartOTA.name);
  347. $('#fwProgressLabel').parent().removeClass('bg-danger');
  348. this.setOTATargets();
  349. if (this.isStateError()) {
  350. return this;
  351. }
  352. if (!recovery) {
  353. this.SetStateRebootRecovery();
  354. }
  355. else {
  356. this.SetStateFlashing().TargetReadyStartOTA();
  357. }
  358. return this;
  359. },
  360. UploadLocalFile: function () {
  361. this.SetStateUploading();
  362. const xhttp = new XMLHttpRequest();
  363. xhttp.context = this;
  364. var boundHandleUploadProgressEvent = this.HandleUploadProgressEvent.bind(this);
  365. var boundsetOTAError = this.setOTAError.bind(this);
  366. xhttp.upload.addEventListener("progress", boundHandleUploadProgressEvent, false);
  367. xhttp.onreadystatechange = function () {
  368. if (xhttp.readyState === 4) {
  369. if (xhttp.status === 0 || xhttp.status === 404) {
  370. boundsetOTAError(`Upload Failed. Recovery version might not support uploading. Please use web update instead.`);
  371. }
  372. }
  373. };
  374. xhttp.open('POST', '/flash.json', true);
  375. xhttp.send(this.flashFileName);
  376. },
  377. TargetReadyStartOTA: function () {
  378. if (recovery && this.prevRecovery && !this.isStateRebootRecovery() && !this.isStateFlashing()) {
  379. // this should only execute once, while being in a valid state
  380. return this;
  381. }
  382. this.logEvent(this.TargetReadyStartOTA.name);
  383. if (!recovery) {
  384. console.error('Event TargetReadyStartOTA fired in the wrong mode ');
  385. return this;
  386. }
  387. this.prevRecovery = true;
  388. if (this.flashFileName !== '') {
  389. this.UploadLocalFile();
  390. }
  391. else if (this.flashURL != '') {
  392. this.SetStateSetUrl();
  393. }
  394. else {
  395. this.setOTAError('Invalid URL or file name while trying to start the OTa process')
  396. }
  397. },
  398. HandleUploadProgressEvent: function (data) {
  399. this.logEvent(this.HandleUploadProgressEvent.name);
  400. this.SetStateUploading().SetStatusPercent(Math.round(data.loaded / data.total * 100)).SetStatusText('Uploading file to device');
  401. },
  402. EventTargetStatus: function (data) {
  403. if (!this.isStateNone()) {
  404. this.logEvent(this.EventTargetStatus.name);
  405. }
  406. if (data.ota_pct ?? -1 >= 0) {
  407. this.olderRecovery = true;
  408. this.SetStatusPercent(data.ota_pct);
  409. }
  410. if ((data.ota_dsc ?? '') != '') {
  411. this.olderRecovery = true;
  412. this.SetStatusText(data.ota_dsc);
  413. }
  414. if (data.recovery != undefined) {
  415. this.recovery = data.recovery === 1 ? true : false;
  416. }
  417. if (this.isStateRebootRecovery() && this.recovery) {
  418. this.TargetReadyStartOTA();
  419. }
  420. },
  421. EventOTAMessageClass: function (data) {
  422. this.logEvent(this.EventOTAMessageClass.name);
  423. var otaData = JSON.parse(data);
  424. this.SetStatusPercent(otaData.ota_pct).SetStatusText(otaData.ota_dsc);
  425. },
  426. logEvent: function (fun) {
  427. console.log(`${fun}, flash state ${this.toString()}, recovery: ${this.recovery}, ota pct: ${this.statusPercent}, ota desc: ${this.statusText}`);
  428. }
  429. };
  430. window.hideSurrounding = function (obj) {
  431. $(obj).parent().parent().hide();
  432. }
  433. let presetsloaded = false;
  434. let is_i2c_locked = false;
  435. let statusInterval = 2000;
  436. let messageInterval = 2500;
  437. function post_config(data) {
  438. let confPayload = {
  439. timestamp: Date.now(),
  440. config: data
  441. };
  442. $.ajax({
  443. url: '/config.json',
  444. dataType: 'text',
  445. method: 'POST',
  446. cache: false,
  447. contentType: 'application/json; charset=utf-8',
  448. data: JSON.stringify(confPayload),
  449. error: handleExceptionResponse,
  450. });
  451. }
  452. window.hFlash = function () {
  453. // reset file upload selection if any;
  454. $('#flashfilename').value = null
  455. flashState.StartOTA();
  456. }
  457. window.handleReboot = function (link) {
  458. if (link == 'reboot_ota') {
  459. $('#reboot_ota_nav').removeClass('active').prop("disabled", true); delayReboot(500, '', 'reboot_ota');
  460. }
  461. else {
  462. $('#reboot_nav').removeClass('active'); delayReboot(500, '', link);
  463. }
  464. }
  465. function parseSqueezeliteCommandLine(commandLine) {
  466. const options = {};
  467. let output, name;
  468. let otherValues = '';
  469. const argRegex = /("[^"]+"|'[^']+'|\S+)/g;
  470. const args = commandLine.match(argRegex);
  471. let i = 0;
  472. while (i < args.length) {
  473. const arg = args[i];
  474. if (arg.startsWith('-')) {
  475. const option = arg.slice(1);
  476. if (option === '') {
  477. otherValues += args.slice(i).join(' ');
  478. break;
  479. }
  480. let value = true;
  481. if (i + 1 < args.length && !args[i + 1].startsWith('-')) {
  482. value = args[i + 1].replace(/"/g, '').replace(/'/g, '');
  483. i++;
  484. }
  485. options[option] = value;
  486. } else {
  487. otherValues += arg + ' ';
  488. }
  489. i++;
  490. }
  491. otherValues = otherValues.trim();
  492. output = getOutput(options);
  493. name = getName(options);
  494. let otherOptions={btname:null,n:null};
  495. // assign o and n options to otheroptions if present
  496. if (options.o && output.toUpperCase() === 'BT') {
  497. let temp = parseSqueezeliteCommandLine(options.o);
  498. if(temp.name) {
  499. otherOptions.btname = temp.name;
  500. }
  501. delete options.o;
  502. }
  503. if (options.n) {
  504. otherOptions['n'] = options.n;
  505. delete options.n;
  506. }
  507. return { name, output, options, otherValues,otherOptions };
  508. }
  509. function getOutput(options) {
  510. let output;
  511. if (options.o){
  512. output = options.o.replace(/"/g, '').replace(/'/g, '');
  513. /* set output as the first alphanumerical word in the command line */
  514. if (output.indexOf(' ') > 0) {
  515. output = output.substring(0, output.indexOf(' '));
  516. }
  517. }
  518. return output;
  519. }
  520. function getName(options) {
  521. let name;
  522. /* if n option present, assign to name variable */
  523. if (options.n){
  524. name = options.n.replace(/"/g, '').replace(/'/g, '');
  525. }
  526. return name;
  527. }
  528. function isConnected() {
  529. return ConnectedTo.hasOwnProperty('ip') && ConnectedTo.ip != '0.0.0.0' && ConnectedTo.ip != '';
  530. }
  531. function getIcon(icons) {
  532. return isConnected() ? icons.icon : icons.label;
  533. }
  534. function handlebtstate(data) {
  535. let icon = '';
  536. let tt = '';
  537. if (data.bt_status !== undefined && data.bt_sub_status !== undefined) {
  538. const iconindex = btStateIcons[data.bt_status].sub[data.bt_sub_status];
  539. if (iconindex) {
  540. icon = btIcons[iconindex];
  541. tt = btStateIcons[data.bt_status].desc;
  542. } else {
  543. icon = btIcons.bt_connected;
  544. tt = 'Output status';
  545. }
  546. }
  547. $('#o_type').attr('title', tt);
  548. $('#o_bt').html(isConnected() ? icon.label : icon.text);
  549. }
  550. function handleTemplateTypeRadio(outtype) {
  551. $('#o_type').children('span').css({ display: 'none' });
  552. let changed = false;
  553. if (outtype === 'bt') {
  554. changed = output !== 'bt' && output !== '';
  555. output = 'bt';
  556. } else if (outtype === 'spdif') {
  557. changed = output !== 'spdif' && output !== '';
  558. output = 'spdif';
  559. } else {
  560. changed = output !== 'i2s' && output !== '';
  561. output = 'i2s';
  562. }
  563. $('#' + output).prop('checked', true);
  564. $('#o_' + output).css({ display: 'inline' });
  565. if (changed) {
  566. Object.keys(commandDefaults[output]).forEach(function (key) {
  567. $(`#cmd_opt_${key}`).val(commandDefaults[output][key]);
  568. });
  569. }
  570. }
  571. function handleExceptionResponse(xhr, _ajaxOptions, thrownError) {
  572. console.log(xhr.status);
  573. console.log(thrownError);
  574. if (thrownError !== '') {
  575. showLocalMessage(thrownError, 'MESSAGING_ERROR');
  576. }
  577. }
  578. function HideCmdMessage(cmdname) {
  579. $('#toast_' + cmdname)
  580. .removeClass('table-success')
  581. .removeClass('table-warning')
  582. .removeClass('table-danger')
  583. .addClass('table-success')
  584. .removeClass('show');
  585. $('#msg_' + cmdname).html('');
  586. }
  587. function showCmdMessage(cmdname, msgtype, msgtext, append = false) {
  588. let color = 'table-success';
  589. if (msgtype === 'MESSAGING_WARNING') {
  590. color = 'table-warning';
  591. } else if (msgtype === 'MESSAGING_ERROR') {
  592. color = 'table-danger';
  593. }
  594. $('#toast_' + cmdname)
  595. .removeClass('table-success')
  596. .removeClass('table-warning')
  597. .removeClass('table-danger')
  598. .addClass(color)
  599. .addClass('show');
  600. let escapedtext = msgtext
  601. .substring(0, msgtext.length - 1)
  602. .encodeHTML()
  603. .replace(/\n/g, '<br />');
  604. escapedtext =
  605. ($('#msg_' + cmdname).html().length > 0 && append
  606. ? $('#msg_' + cmdname).html() + '<br/>'
  607. : '') + escapedtext;
  608. $('#msg_' + cmdname).html(escapedtext);
  609. }
  610. let releaseURL =
  611. 'https://api.github.com/repos/sle118/squeezelite-esp32/releases';
  612. let recovery = false;
  613. let messagesHeld = false;
  614. let commandBTSinkName = '';
  615. const commandHeader = 'squeezelite ';
  616. const commandDefaults = {
  617. i2s: { b: "500:2000", C: "30", W: "", Z: "96000", o: "I2S" },
  618. spdif: { b: "500:2000", C: "30", W: "", Z: "48000", o: "SPDIF" },
  619. bt: { b: "500:2000", C: "30", W: "", Z: "44100", o: "BT" },
  620. };
  621. let validOptions = {
  622. codecs: ['flac', 'pcm', 'mp3', 'ogg', 'aac', 'wma', 'alac', 'dsd', 'mad', 'mpg']
  623. };
  624. //let blockFlashButton = false;
  625. let apList = null;
  626. //let selectedSSID = '';
  627. //let checkStatusInterval = null;
  628. let messagecount = 0;
  629. let messageseverity = 'MESSAGING_INFO';
  630. let SystemConfig = {};
  631. let LastCommandsState = null;
  632. var output = '';
  633. let hostName = '';
  634. let versionName = 'Squeezelite-ESP32';
  635. let prevmessage = '';
  636. let project_name = versionName;
  637. let depth = 16;
  638. let board_model = '';
  639. let platform_name = versionName;
  640. let preset_name = '';
  641. let btSinkNamesOptSel = '#cfg-audio-bt_source-sink_name';
  642. let ConnectedTo = {};
  643. let ConnectingToSSID = {};
  644. let lmsBaseUrl;
  645. let prevLMSIP = '';
  646. const ConnectingToActions = {
  647. 'CONN': 0, 'MAN': 1, 'STS': 2,
  648. }
  649. Promise.prototype.delay = function (duration) {
  650. return this.then(
  651. function (value) {
  652. return new Promise(function (resolve) {
  653. setTimeout(function () {
  654. resolve(value);
  655. }, duration);
  656. });
  657. },
  658. function (reason) {
  659. return new Promise(function (_resolve, reject) {
  660. setTimeout(function () {
  661. reject(reason);
  662. }, duration);
  663. });
  664. }
  665. );
  666. };
  667. function getConfigJson(slimMode) {
  668. const config = {};
  669. $('input.nvs').each(function (_index, entry) {
  670. if (!slimMode) {
  671. const nvsType = parseInt(entry.attributes.nvs_type.value, 10);
  672. if (entry.id !== '') {
  673. config[entry.id] = {};
  674. if (
  675. nvsType === nvsTypes.NVS_TYPE_U8 ||
  676. nvsType === nvsTypes.NVS_TYPE_I8 ||
  677. nvsType === nvsTypes.NVS_TYPE_U16 ||
  678. nvsType === nvsTypes.NVS_TYPE_I16 ||
  679. nvsType === nvsTypes.NVS_TYPE_U32 ||
  680. nvsType === nvsTypes.NVS_TYPE_I32 ||
  681. nvsType === nvsTypes.NVS_TYPE_U64 ||
  682. nvsType === nvsTypes.NVS_TYPE_I64
  683. ) {
  684. config[entry.id].value = parseInt(entry.value);
  685. } else {
  686. config[entry.id].value = entry.value;
  687. }
  688. config[entry.id].type = nvsType;
  689. }
  690. } else {
  691. config[entry.id] = entry.value;
  692. }
  693. });
  694. const key = $('#nvs-new-key').val();
  695. const val = $('#nvs-new-value').val();
  696. if (key !== '') {
  697. if (!slimMode) {
  698. config[key] = {};
  699. config[key].value = val;
  700. config[key].type = 33;
  701. } else {
  702. config[key] = val;
  703. }
  704. }
  705. return config;
  706. }
  707. function handleHWPreset(allfields, reboot) {
  708. const selJson = JSON.parse(allfields[0].value);
  709. var cmd = allfields[0].attributes.cmdname.value;
  710. console.log(`selected model: ${selJson.name}`);
  711. let confPayload = {
  712. timestamp: Date.now(),
  713. config: { model_config: { value: selJson.name, type: 33 } }
  714. };
  715. for (const [name, value] of Object.entries(selJson.config)) {
  716. const storedval = (typeof value === 'string' || value instanceof String) ? value : JSON.stringify(value);
  717. confPayload.config[name] = {
  718. value: storedval,
  719. type: 33,
  720. }
  721. showCmdMessage(
  722. cmd,
  723. 'MESSAGING_INFO',
  724. `Setting ${name}=${storedval} `,
  725. true
  726. );
  727. }
  728. showCmdMessage(
  729. cmd,
  730. 'MESSAGING_INFO',
  731. `Committing `,
  732. true
  733. );
  734. $.ajax({
  735. url: '/config.json',
  736. dataType: 'text',
  737. method: 'POST',
  738. cache: false,
  739. contentType: 'application/json; charset=utf-8',
  740. data: JSON.stringify(confPayload),
  741. error: function (xhr, _ajaxOptions, thrownError) {
  742. handleExceptionResponse(xhr, _ajaxOptions, thrownError);
  743. showCmdMessage(
  744. cmd,
  745. 'MESSAGING_ERROR',
  746. `Unexpected error ${(thrownError !== '') ? thrownError : 'with return status = ' + xhr.status} `,
  747. true
  748. );
  749. },
  750. success: function (response) {
  751. showCmdMessage(
  752. cmd,
  753. 'MESSAGING_INFO',
  754. `Saving complete `,
  755. true
  756. );
  757. console.log(response);
  758. if (reboot) {
  759. delayReboot(2500, cmd);
  760. }
  761. },
  762. });
  763. }
  764. // pull json file from https://gist.githubusercontent.com/sle118/dae585e157b733a639c12dc70f0910c5/raw/b462691f69e2ad31ac95c547af6ec97afb0f53db/squeezelite-esp32-presets.json and
  765. function loadPresets() {
  766. if ($("#cfg-hw-preset-model_config").length == 0) return;
  767. if (presetsloaded) return;
  768. presetsloaded = true;
  769. $('#cfg-hw-preset-model_config').html('<option>--</option>');
  770. $.getJSON(
  771. 'https://gist.githubusercontent.com/sle118/dae585e157b733a639c12dc70f0910c5/raw/',
  772. { _: new Date().getTime() },
  773. function (data) {
  774. $.each(data, function (key, val) {
  775. $('#cfg-hw-preset-model_config').append(`<option value='${JSON.stringify(val).replace(/"/g, '\"').replace(/\'/g, '\"')}'>${val.name}</option>`);
  776. if (preset_name !== '' && preset_name == val.name) {
  777. $('#cfg-hw-preset-model_config').val(preset_name);
  778. }
  779. });
  780. if (preset_name !== '') {
  781. ('#prev_preset').show().val(preset_name);
  782. }
  783. }
  784. ).fail(function (jqxhr, textStatus, error) {
  785. const err = textStatus + ', ' + error;
  786. console.log('Request Failed: ' + err);
  787. }
  788. );
  789. }
  790. function delayReboot(duration, cmdname, ota = 'reboot') {
  791. const url = '/' + ota + '.json';
  792. $('tbody#tasks').empty();
  793. $('#tasks_sect').css('visibility', 'collapse');
  794. Promise.resolve({ cmdname: cmdname, url: url })
  795. .delay(duration)
  796. .then(function (data) {
  797. if (data.cmdname.length > 0) {
  798. showCmdMessage(
  799. data.cmdname,
  800. 'MESSAGING_WARNING',
  801. 'System is rebooting.\n',
  802. true
  803. );
  804. } else {
  805. showLocalMessage('System is rebooting.\n', 'MESSAGING_WARNING');
  806. }
  807. console.log('now triggering reboot');
  808. $("button[onclick*='handleReboot']").addClass('rebooting');
  809. $.ajax({
  810. url: data.url,
  811. dataType: 'text',
  812. method: 'POST',
  813. cache: false,
  814. contentType: 'application/json; charset=utf-8',
  815. data: JSON.stringify({
  816. timestamp: Date.now(),
  817. }),
  818. error: handleExceptionResponse,
  819. complete: function () {
  820. console.log('reboot call completed');
  821. Promise.resolve(data)
  822. .delay(6000)
  823. .then(function (rdata) {
  824. if (rdata.cmdname.length > 0) {
  825. HideCmdMessage(rdata.cmdname);
  826. }
  827. getCommands();
  828. getConfig();
  829. });
  830. },
  831. });
  832. });
  833. }
  834. // eslint-disable-next-line no-unused-vars
  835. window.saveAutoexec1 = function (apply) {
  836. showCmdMessage('cfg-audio-tmpl', 'MESSAGING_INFO', 'Saving.\n', false);
  837. let commandLine = `${commandHeader} -o ${output} `;
  838. $('.sqcmd').each(function () {
  839. let { opt, val } = get_control_option_value($(this));
  840. if ((opt && opt.length>0 ) && typeof(val) == 'boolean' || val.length > 0) {
  841. const optStr=opt===':'?opt:(` -${opt} `);
  842. val = typeof(val) == 'boolean'?'':val;
  843. commandLine += `${optStr} ${val}`;
  844. }
  845. });
  846. const resample=$('#cmd_opt_R input[name=resample]:checked');
  847. if (resample.length>0 && resample.attr('suffix')!=='') {
  848. commandLine += resample.attr('suffix');
  849. // now check resample_i option and if checked, add suffix to command line
  850. if ($('#resample_i').is(":checked") && resample.attr('aint') =='true') {
  851. commandLine += $('#resample_i').attr('suffix');
  852. }
  853. }
  854. if (output === 'bt') {
  855. showCmdMessage(
  856. 'cfg-audio-tmpl',
  857. 'MESSAGING_INFO',
  858. 'Remember to configure the Bluetooth audio device name.\n',
  859. true
  860. );
  861. }
  862. commandLine += concatenateOptions(options);
  863. const data = {
  864. timestamp: Date.now(),
  865. };
  866. data.config = {
  867. autoexec1: { value: commandLine, type: 33 },
  868. // autoexec: {
  869. // value: $('#disable-squeezelite').prop('checked') ? '0' : '1',
  870. // type: 33,
  871. // },
  872. };
  873. $.ajax({
  874. url: '/config.json',
  875. dataType: 'text',
  876. method: 'POST',
  877. cache: false,
  878. contentType: 'application/json; charset=utf-8',
  879. data: JSON.stringify(data),
  880. error: handleExceptionResponse,
  881. complete: function (response) {
  882. if (
  883. response.responseText &&
  884. JSON.parse(response.responseText).result === 'OK'
  885. ) {
  886. showCmdMessage('cfg-audio-tmpl', 'MESSAGING_INFO', 'Done.\n', true);
  887. if (apply) {
  888. delayReboot(1500, 'cfg-audio-tmpl');
  889. }
  890. } else if (JSON.parse(response.responseText).result) {
  891. showCmdMessage(
  892. 'cfg-audio-tmpl',
  893. 'MESSAGING_WARNING',
  894. JSON.parse(response.responseText).Result + '\n',
  895. true
  896. );
  897. } else {
  898. showCmdMessage(
  899. 'cfg-audio-tmpl',
  900. 'MESSAGING_ERROR',
  901. response.statusText + '\n'
  902. );
  903. }
  904. console.log(response.responseText);
  905. },
  906. });
  907. console.log('sent data:', JSON.stringify(data));
  908. }
  909. window.handleDisconnect = function () {
  910. $.ajax({
  911. url: '/connect.json',
  912. dataType: 'text',
  913. method: 'DELETE',
  914. cache: false,
  915. contentType: 'application/json; charset=utf-8',
  916. data: JSON.stringify({
  917. timestamp: Date.now(),
  918. }),
  919. });
  920. }
  921. function setPlatformFilter(val) {
  922. if ($('.upf').filter(function () { return $(this).text().toUpperCase() === val.toUpperCase() }).length > 0) {
  923. $('#splf').val(val).trigger('input');
  924. return true;
  925. }
  926. return false;
  927. }
  928. window.handleConnect = function () {
  929. ConnectingToSSID.ssid = $('#manual_ssid').val();
  930. ConnectingToSSID.pwd = $('#manual_pwd').val();
  931. ConnectingToSSID.dhcpname = $('#dhcp-name2').val();
  932. $("*[class*='connecting']").hide();
  933. $('#ssid-wait').text(ConnectingToSSID.ssid);
  934. $('.connecting').show();
  935. $.ajax({
  936. url: '/connect.json',
  937. dataType: 'text',
  938. method: 'POST',
  939. cache: false,
  940. contentType: 'application/json; charset=utf-8',
  941. data: JSON.stringify({
  942. timestamp: Date.now(),
  943. ssid: ConnectingToSSID.ssid,
  944. pwd: ConnectingToSSID.pwd
  945. }),
  946. error: handleExceptionResponse,
  947. });
  948. // now we can re-set the intervals regardless of result
  949. }
  950. function renderError(opt,error){
  951. const fieldname = `cmd_opt_${opt}`;
  952. let errorFieldName=`${fieldname}-error`;
  953. let errorField=$(`#${errorFieldName}`);
  954. let field=$(`#${fieldname}`);
  955. if (!errorField || errorField.length ==0) {
  956. field.after(`<div id="${errorFieldName}" class="invalid-feedback"></div>`);
  957. errorField=$(`#${errorFieldName}`);
  958. }
  959. if(error.length ==0){
  960. errorField.hide();
  961. field.removeClass('is-invalid');
  962. field.addClass('is-valid');
  963. errorField.text('');
  964. }
  965. else {
  966. errorField.show();
  967. errorField.text(error);
  968. field.removeClass('is-valid');
  969. field.addClass('is-invalid');
  970. }
  971. return errorField;
  972. }
  973. $(document).ready(function () {
  974. $('.material-icons').each(function (_index, entry) {
  975. entry.attributes['icon'] = entry.textContent;
  976. });
  977. setIcons(true);
  978. handleNVSVisible();
  979. flashState.init();
  980. $('#fw-url-input').on('input', function () {
  981. if ($(this).val().length > 8 && ($(this).val().startsWith('http://') || $(this).val().startsWith('https://'))) {
  982. $('#start-flash').show();
  983. }
  984. else {
  985. $('#start-flash').hide();
  986. }
  987. });
  988. $('.upSrch').on('input', function () {
  989. const val = this.value;
  990. $("#rTable tr").removeClass(this.id + '_hide');
  991. if (val.length > 0) {
  992. $(`#rTable td:nth-child(${$(this).parent().index() + 1})`).filter(function () {
  993. return !$(this).text().toUpperCase().includes(val.toUpperCase());
  994. }).parent().addClass(this.id + '_hide');
  995. }
  996. $('[class*="_hide"]').hide();
  997. $('#rTable tr').not('[class*="_hide"]').show()
  998. });
  999. setTimeout(refreshAP, 1500);
  1000. /* add validation for cmd_opt_c, which accepts a comma separated list.
  1001. getting known codecs from validOptions.codecs array
  1002. use bootstrap classes to highlight the error with an overlay message */
  1003. $('#options input').on('input', function () {
  1004. const { opt, val } = get_control_option_value(this);
  1005. if (opt === 'c' || opt === 'e') {
  1006. const fieldname = `cmd_opt_${opt}_codec-error`;
  1007. const values = val.split(',').map(function (item) {
  1008. return item.trim();
  1009. });
  1010. /* get a list of invalid codecs */
  1011. const invalid = values.filter(function (item) {
  1012. return !validOptions.codecs.includes(item);
  1013. });
  1014. renderError(opt,invalid.length > 0 ? `Invalid codec(s) ${invalid.join(', ')}` : '');
  1015. }
  1016. /* add validation for cmd_opt_m, which accepts a mac_address */
  1017. if (opt === 'm') {
  1018. const mac_regex = /^([0-9A-Fa-f]{2}[:-]){5}([0-9A-Fa-f]{2})$/;
  1019. renderError(opt,mac_regex.test(val) ? '' : 'Invalid MAC address');
  1020. }
  1021. if (opt === 'r') {
  1022. const rateRegex = /^(\d+\.?\d*|\.\d+)-(\d+\.?\d*|\.\d+)$|^(\d+\.?\d*)$|^(\d+\.?\d*,)+\d+\.?\d*$/;
  1023. renderError(opt,rateRegex.test(val)?'':`Invalid rate(s) ${val}. Acceptable format: <maxrate>|<minrate>-<maxrate>|<rate1>,<rate2>,<rate3>`);
  1024. }
  1025. }
  1026. );
  1027. $('#WifiConnectDialog')[0].addEventListener('shown.bs.modal', function (event) {
  1028. $("*[class*='connecting']").hide();
  1029. if (event?.relatedTarget) {
  1030. ConnectingToSSID.Action = ConnectingToActions.CONN;
  1031. if ($(event.relatedTarget).children('td:eq(1)').text() == ConnectedTo.ssid) {
  1032. ConnectingToSSID.Action = ConnectingToActions.STS;
  1033. }
  1034. else {
  1035. if (!$(event.relatedTarget).is(':last-child')) {
  1036. ConnectingToSSID.ssid = $(event.relatedTarget).children('td:eq(1)').text();
  1037. $('#manual_ssid').val(ConnectingToSSID.ssid);
  1038. }
  1039. else {
  1040. ConnectingToSSID.Action = ConnectingToActions.MAN;
  1041. ConnectingToSSID.ssid = '';
  1042. $('#manual_ssid').val(ConnectingToSSID.ssid);
  1043. }
  1044. }
  1045. }
  1046. if (ConnectingToSSID.Action !== ConnectingToActions.STS) {
  1047. $('.connecting-init').show();
  1048. $('#manual_ssid').trigger('focus');
  1049. }
  1050. else {
  1051. handleWifiDialog();
  1052. }
  1053. });
  1054. $('#WifiConnectDialog')[0].addEventListener('hidden.bs.modal', function () {
  1055. $('#WifiConnectDialog input').val('');
  1056. });
  1057. $('#uCnfrm')[0].addEventListener('shown.bs.modal', function () {
  1058. $('#selectedFWURL').text($('#fw-url-input').val());
  1059. });
  1060. $('input#show-commands')[0].checked = LastCommandsState === 1;
  1061. $('a[href^="#tab-commands"]').hide();
  1062. $('#load-nvs').on('click', function () {
  1063. $('#nvsfilename').trigger('click');
  1064. });
  1065. $('#nvsfilename').on('change', function () {
  1066. if (typeof window.FileReader !== 'function') {
  1067. throw "The file API isn't supported on this browser.";
  1068. }
  1069. if (!this.files) {
  1070. throw 'This browser does not support the `files` property of the file input.';
  1071. }
  1072. if (!this.files[0]) {
  1073. return undefined;
  1074. }
  1075. const file = this.files[0];
  1076. let fr = new FileReader();
  1077. fr.onload = function (e) {
  1078. let data = {};
  1079. try {
  1080. data = JSON.parse(e.target.result);
  1081. } catch (ex) {
  1082. alert('Parsing failed!\r\n ' + ex);
  1083. }
  1084. $('input.nvs').each(function (_index, entry) {
  1085. $(this).parent().removeClass('bg-warning').removeClass('bg-success');
  1086. if (data[entry.id]) {
  1087. if (data[entry.id] !== entry.value) {
  1088. console.log(
  1089. 'Changed ' + entry.id + ' ' + entry.value + '==>' + data[entry.id]
  1090. );
  1091. $(this).parent().addClass('bg-warning');
  1092. $(this).val(data[entry.id]);
  1093. }
  1094. else {
  1095. $(this).parent().addClass('bg-success');
  1096. }
  1097. }
  1098. });
  1099. var changed = $("input.nvs").children('.bg-warning');
  1100. if (changed) {
  1101. alert('Highlighted values were changed. Press Commit to change on the device');
  1102. }
  1103. }
  1104. fr.readAsText(file);
  1105. this.value = null;
  1106. }
  1107. );
  1108. $('#clear-syslog').on('click', function () {
  1109. messagecount = 0;
  1110. messageseverity = 'MESSAGING_INFO';
  1111. $('#msgcnt').text('');
  1112. $('#syslogTable').html('');
  1113. });
  1114. $('#ok-credits').on('click', function () {
  1115. $('#credits').slideUp('fast', function () { });
  1116. $('#app').slideDown('fast', function () { });
  1117. });
  1118. $('#acredits').on('click', function (event) {
  1119. event.preventDefault();
  1120. $('#app').slideUp('fast', function () { });
  1121. $('#credits').slideDown('fast', function () { });
  1122. });
  1123. $('input#show-commands').on('click', function () {
  1124. this.checked = this.checked ? 1 : 0;
  1125. if (this.checked) {
  1126. $('a[href^="#tab-commands"]').show();
  1127. LastCommandsState = 1;
  1128. } else {
  1129. LastCommandsState = 0;
  1130. $('a[href^="#tab-commands"]').hide();
  1131. }
  1132. });
  1133. $('#disable-squeezelite').on('click', function () {
  1134. // this.checked = this.checked ? 1 : 0;
  1135. // $('#disable-squeezelite').prop('checked')
  1136. if (this.checked) {
  1137. // Store the current value before overwriting it
  1138. const currentValue = $('#cmd_opt_s').val();
  1139. $('#cmd_opt_s').data('originalValue', currentValue);
  1140. // Overwrite the value with '-disable'
  1141. $('#cmd_opt_s').val('-disable');
  1142. } else {
  1143. // Retrieve the original value
  1144. const originalValue = $('#cmd_opt_s').data('originalValue');
  1145. // Restore the original value if it exists, otherwise set it to an empty string
  1146. $('#cmd_opt_s').val(originalValue ? originalValue : '');
  1147. }
  1148. });
  1149. $('input#show-nvs').on('click', function () {
  1150. this.checked = this.checked ? 1 : 0;
  1151. Cookies.set("show-nvs", this.checked ? 'Y' : 'N');
  1152. handleNVSVisible();
  1153. });
  1154. $('#btn_reboot_recovery').on('click', function () {
  1155. handleReboot('recovery');
  1156. });
  1157. $('#btn_reboot').on('click', function () {
  1158. handleReboot('reboot');
  1159. });
  1160. $('#btn_flash').on('click', function () {
  1161. hFlash();
  1162. });
  1163. $('#save-autoexec1').on('click', function () {
  1164. saveAutoexec1(false);
  1165. });
  1166. $('#commit-autoexec1').on('click', function () {
  1167. saveAutoexec1(true);
  1168. });
  1169. $('#btn_disconnect').on('click', function () {
  1170. ConnectedTo = {};
  1171. refreshAPHTML2();
  1172. $.ajax({
  1173. url: '/connect.json',
  1174. dataType: 'text',
  1175. method: 'DELETE',
  1176. cache: false,
  1177. contentType: 'application/json; charset=utf-8',
  1178. data: JSON.stringify({
  1179. timestamp: Date.now(),
  1180. }),
  1181. });
  1182. });
  1183. $('#btnJoin').on('click', function () {
  1184. handleConnect();
  1185. });
  1186. $('#reboot_nav').on('click', function () {
  1187. handleReboot('reboot');
  1188. });
  1189. $('#reboot_ota_nav').on('click', function () {
  1190. handleReboot('reboot_ota');
  1191. });
  1192. $('#save-as-nvs').on('click', function () {
  1193. const config = getConfigJson(true);
  1194. const a = document.createElement('a');
  1195. a.href = URL.createObjectURL(
  1196. new Blob([JSON.stringify(config, null, 2)], {
  1197. type: 'text/plain',
  1198. })
  1199. );
  1200. a.setAttribute(
  1201. 'download',
  1202. 'nvs_config_' + hostName + '_' + Date.now() + 'json'
  1203. );
  1204. document.body.appendChild(a);
  1205. a.click();
  1206. document.body.removeChild(a);
  1207. });
  1208. $('#save-nvs').on('click', function () {
  1209. post_config(getConfigJson(false));
  1210. });
  1211. $('#fwUpload').on('click', function () {
  1212. const fileInput = document.getElementById('flashfilename').files;
  1213. if (fileInput.length === 0) {
  1214. alert('No file selected!');
  1215. } else {
  1216. $('#fw-url-input').value = null;
  1217. flashState.StartOTA();
  1218. }
  1219. });
  1220. $('[name=output-tmpl]').on('click', function () {
  1221. handleTemplateTypeRadio(this.id);
  1222. });
  1223. $('#chkUpdates').on('click', function () {
  1224. $('#rTable').html('');
  1225. $.getJSON(releaseURL, function (data) {
  1226. let i = 0;
  1227. const branches = [];
  1228. data.forEach(function (release) {
  1229. const namecomponents = release.name.split('#');
  1230. const branch = namecomponents[3];
  1231. if (!branches.includes(branch)) {
  1232. branches.push(branch);
  1233. }
  1234. });
  1235. let fwb = '';
  1236. branches.forEach(function (branch) {
  1237. fwb += '<option value="' + branch + '">' + branch + '</option>';
  1238. });
  1239. $('#fwbranch').append(fwb);
  1240. data.forEach(function (release) {
  1241. let url = '';
  1242. release.assets.forEach(function (asset) {
  1243. if (asset.name.match(/\.bin$/)) {
  1244. url = asset.browser_download_url;
  1245. }
  1246. });
  1247. const namecomponents = release.name.split('#');
  1248. const ver = namecomponents[0];
  1249. const cfg = namecomponents[2];
  1250. const branch = namecomponents[3];
  1251. var bits = ver.substr(ver.lastIndexOf('-') + 1);
  1252. bits = (bits == '32' || bits == '16') ? bits : '';
  1253. let body = release.body;
  1254. body = body.replace(/'/gi, '"');
  1255. body = body.replace(
  1256. /[\s\S]+(### Revision Log[\s\S]+)### ESP-IDF Version Used[\s\S]+/,
  1257. '$1'
  1258. );
  1259. body = body.replace(/- \(.+?\) /g, '- ').encodeHTML();
  1260. $('#rTable').append(`<tr class='release ' fwurl='${url}'>
  1261. <td data-bs-toggle='tooltip' title='${body}'>${ver}</td><td>${new Date(release.created_at).toLocalShort()}
  1262. </td><td class='upf'>${cfg}</td><td>${branch}</td><td>${bits}</td></tr>`
  1263. );
  1264. });
  1265. if (i > 7) {
  1266. $('#releaseTable').append(
  1267. "<tr id='showall'>" +
  1268. "<td colspan='6'>" +
  1269. "<input type='button' id='showallbutton' class='btn btn-info' value='Show older releases' />" +
  1270. '</td>' +
  1271. '</tr>'
  1272. );
  1273. $('#showallbutton').on('click', function () {
  1274. $('tr.hide').removeClass('hide');
  1275. $('tr#showall').addClass('hide');
  1276. });
  1277. }
  1278. $('#searchfw').css('display', 'inline');
  1279. if (!setPlatformFilter(platform_name)) {
  1280. setPlatformFilter(project_name)
  1281. }
  1282. $('#rTable tr.release').on('click', function () {
  1283. var url = this.attributes['fwurl'].value;
  1284. if (lmsBaseUrl) {
  1285. url = url.replace(/.*\/download\//, lmsBaseUrl + '/plugins/SqueezeESP32/firmware/');
  1286. }
  1287. $('#fw-url-input').val(url);
  1288. $('#start-flash').show();
  1289. $('#rTable tr.release').removeClass('table-success table-warning');
  1290. $(this).addClass('table-success table-warning');
  1291. });
  1292. }).fail(function () {
  1293. alert('failed to fetch release history!');
  1294. });
  1295. });
  1296. $('#fwcheck').on('click', function () {
  1297. $('#releaseTable').html('');
  1298. $('#fwbranch').empty();
  1299. $.getJSON(releaseURL, function (data) {
  1300. let i = 0;
  1301. const branches = [];
  1302. data.forEach(function (release) {
  1303. const namecomponents = release.name.split('#');
  1304. const branch = namecomponents[3];
  1305. if (!branches.includes(branch)) {
  1306. branches.push(branch);
  1307. }
  1308. });
  1309. let fwb;
  1310. branches.forEach(function (branch) {
  1311. fwb += '<option value="' + branch + '">' + branch + '</option>';
  1312. });
  1313. $('#fwbranch').append(fwb);
  1314. data.forEach(function (release) {
  1315. let url = '';
  1316. release.assets.forEach(function (asset) {
  1317. if (asset.name.match(/\.bin$/)) {
  1318. url = asset.browser_download_url;
  1319. }
  1320. });
  1321. const namecomponents = release.name.split('#');
  1322. const ver = namecomponents[0];
  1323. const idf = namecomponents[1];
  1324. const cfg = namecomponents[2];
  1325. const branch = namecomponents[3];
  1326. let body = release.body;
  1327. body = body.replace(/'/gi, '"');
  1328. body = body.replace(
  1329. /[\s\S]+(### Revision Log[\s\S]+)### ESP-IDF Version Used[\s\S]+/,
  1330. '$1'
  1331. );
  1332. body = body.replace(/- \(.+?\) /g, '- ');
  1333. const trclass = i++ > 6 ? ' hide' : '';
  1334. $('#releaseTable').append(
  1335. "<tr class='release" +
  1336. trclass +
  1337. "'>" +
  1338. "<td data-bs-toggle='tooltip' title='" +
  1339. body +
  1340. "'>" +
  1341. ver +
  1342. '</td>' +
  1343. '<td>' +
  1344. new Date(release.created_at).toLocalShort() +
  1345. '</td>' +
  1346. '<td>' +
  1347. cfg +
  1348. '</td>' +
  1349. '<td>' +
  1350. idf +
  1351. '</td>' +
  1352. '<td>' +
  1353. branch +
  1354. '</td>' +
  1355. "<td><input type='button' class='btn btn-success' value='Select' data-bs-url='" +
  1356. url +
  1357. "' onclick='setURL(this);' /></td>" +
  1358. '</tr>'
  1359. );
  1360. });
  1361. if (i > 7) {
  1362. $('#releaseTable').append(
  1363. "<tr id='showall'>" +
  1364. "<td colspan='6'>" +
  1365. "<input type='button' id='showallbutton' class='btn btn-info' value='Show older releases' />" +
  1366. '</td>' +
  1367. '</tr>'
  1368. );
  1369. $('#showallbutton').on('click', function () {
  1370. $('tr.hide').removeClass('hide');
  1371. $('tr#showall').addClass('hide');
  1372. });
  1373. }
  1374. $('#searchfw').css('display', 'inline');
  1375. }).fail(function () {
  1376. alert('failed to fetch release history!');
  1377. });
  1378. });
  1379. $('#updateAP').on('click', function () {
  1380. refreshAP();
  1381. console.log('refresh AP');
  1382. });
  1383. // first time the page loads: attempt to get the connection status and start the wifi scan
  1384. getConfig();
  1385. getCommands();
  1386. getMessages();
  1387. checkStatus();
  1388. });
  1389. // eslint-disable-next-line no-unused-vars
  1390. window.setURL = function (button) {
  1391. let url = button.dataset.url;
  1392. $('[data-bs-url^="http"]')
  1393. .addClass('btn-success')
  1394. .removeClass('btn-danger');
  1395. $('[data-bs-url="' + url + '"]')
  1396. .addClass('btn-danger')
  1397. .removeClass('btn-success');
  1398. // if user can proxy download through LMS, modify the URL
  1399. if (lmsBaseUrl) {
  1400. url = url.replace(/.*\/download\//, lmsBaseUrl + '/plugins/SqueezeESP32/firmware/');
  1401. }
  1402. $('#fwurl').val(url);
  1403. }
  1404. function rssiToIcon(rssi) {
  1405. if (rssi >= -55) {
  1406. return { 'label': '****', 'icon': `signal_wifi_statusbar_4_bar` };
  1407. } else if (rssi >= -60) {
  1408. return { 'label': '***', 'icon': `network_wifi_3_bar` };
  1409. } else if (rssi >= -65) {
  1410. return { 'label': '**', 'icon': `network_wifi_2_bar` };
  1411. } else if (rssi >= -70) {
  1412. return { 'label': '*', 'icon': `network_wifi_1_bar` };
  1413. } else {
  1414. return { 'label': '.', 'icon': `signal_wifi_statusbar_null` };
  1415. }
  1416. }
  1417. function refreshAP() {
  1418. if (ConnectedTo?.urc === connectReturnCode.ETH) return;
  1419. $.ajaxSetup({
  1420. timeout: 3000 //Time in milliseconds
  1421. });
  1422. $.getJSON('/scan.json', async function () {
  1423. await sleep(2000);
  1424. $.getJSON('/ap.json', function (data) {
  1425. if (data.length > 0) {
  1426. // sort by signal strength
  1427. data.sort(function (a, b) {
  1428. const x = a.rssi;
  1429. const y = b.rssi;
  1430. // eslint-disable-next-line no-nested-ternary
  1431. return x < y ? 1 : x > y ? -1 : 0;
  1432. });
  1433. apList = data;
  1434. refreshAPHTML2(apList);
  1435. }
  1436. });
  1437. });
  1438. }
  1439. function formatAP(ssid, rssi, auth) {
  1440. const rssi_icon = rssiToIcon(rssi);
  1441. const auth_icon = { label: auth == 0 ? '🔓' : '🔒', icon: auth == 0 ? 'no_encryption' : 'lock' };
  1442. return `<tr data-bs-toggle="modal" data-bs-target="#WifiConnectDialog"><td></td><td>${ssid}</td><td>
  1443. <span class="material-icons" style="fill:white; display: inline" aria-label="${rssi_icon.label}" icon="${rssi_icon.icon}" >${getIcon(rssi_icon)}</span>
  1444. </td><td>
  1445. <span class="material-icons" aria-label="${auth_icon.label}" icon="${auth_icon.icon}">${getIcon(auth_icon)}</span>
  1446. </td></tr>`;
  1447. }
  1448. function refreshAPHTML2(data) {
  1449. let h = '';
  1450. $('#wifiTable tr td:first-of-type').text('');
  1451. $('#wifiTable tr').removeClass('table-success table-warning');
  1452. if (data) {
  1453. data.forEach(function (e) {
  1454. h += formatAP(e.ssid, e.rssi, e.auth);
  1455. });
  1456. $('#wifiTable').html(h);
  1457. }
  1458. if ($('.manual_add').length == 0) {
  1459. $('#wifiTable').append(formatAP('Manual add', 0, 0));
  1460. $('#wifiTable tr:last').addClass('table-light text-dark').addClass('manual_add');
  1461. }
  1462. if (ConnectedTo.ssid && (ConnectedTo.urc === connectReturnCode.OK || ConnectedTo.urc === connectReturnCode.RESTORE)) {
  1463. const wifiSelector = `#wifiTable td:contains("${ConnectedTo.ssid}")`;
  1464. if ($(wifiSelector).filter(function () { return $(this).text() === ConnectedTo.ssid; }).length == 0) {
  1465. $('#wifiTable').prepend(`${formatAP(ConnectedTo.ssid, ConnectedTo.rssi ?? 0, 0)}`);
  1466. }
  1467. $(wifiSelector).filter(function () { return $(this).text() === ConnectedTo.ssid; }).siblings().first().html('&check;').parent().addClass((ConnectedTo.urc === connectReturnCode.OK ? 'table-success' : 'table-warning'));
  1468. $('span#foot-if').html(`SSID: <strong>${ConnectedTo.ssid}</strong>, IP: <strong>${ConnectedTo.ip}</strong>`);
  1469. $('#wifiStsIcon').html(rssiToIcon(ConnectedTo.rssi));
  1470. }
  1471. else if (ConnectedTo?.urc !== connectReturnCode.ETH) {
  1472. $('span#foot-if').html('');
  1473. }
  1474. }
  1475. function refreshETH() {
  1476. if (ConnectedTo.urc === connectReturnCode.ETH) {
  1477. $('span#foot-if').html(`Network: Ethernet, IP: <strong>${ConnectedTo.ip}</strong>`);
  1478. }
  1479. }
  1480. function showTask(task) {
  1481. console.debug(
  1482. this.toLocaleString() +
  1483. '\t' +
  1484. task.nme +
  1485. '\t' +
  1486. task.cpu +
  1487. '\t' +
  1488. taskStates[task.st] +
  1489. '\t' +
  1490. task.minstk +
  1491. '\t' +
  1492. task.bprio +
  1493. '\t' +
  1494. task.cprio +
  1495. '\t' +
  1496. task.num
  1497. );
  1498. $('tbody#tasks').append(
  1499. '<tr class="table-primary"><th scope="row">' +
  1500. task.num +
  1501. '</th><td>' +
  1502. task.nme +
  1503. '</td><td>' +
  1504. task.cpu +
  1505. '</td><td>' +
  1506. taskStates[task.st] +
  1507. '</td><td>' +
  1508. task.minstk +
  1509. '</td><td>' +
  1510. task.bprio +
  1511. '</td><td>' +
  1512. task.cprio +
  1513. '</td></tr>'
  1514. );
  1515. }
  1516. function btExists(name) {
  1517. return getBTSinkOpt(name).length > 0;
  1518. }
  1519. function getBTSinkOpt(name) {
  1520. return $(`${btSinkNamesOptSel} option:contains('${name}')`);
  1521. }
  1522. function getMessages() {
  1523. $.ajaxSetup({
  1524. timeout: messageInterval //Time in milliseconds
  1525. });
  1526. $.getJSON('/messages.json', async function (data) {
  1527. for (const msg of data) {
  1528. const msgAge = msg.current_time - msg.sent_time;
  1529. var msgTime = new Date();
  1530. msgTime.setTime(msgTime.getTime() - msgAge);
  1531. switch (msg.class) {
  1532. case 'MESSAGING_CLASS_OTA':
  1533. flashState.EventOTAMessageClass(msg.message);
  1534. break;
  1535. case 'MESSAGING_CLASS_STATS':
  1536. // for task states, check structure : task_state_t
  1537. var statsData = JSON.parse(msg.message);
  1538. console.debug(
  1539. msgTime.toLocalShort() +
  1540. ' - Number of running tasks: ' +
  1541. statsData.ntasks
  1542. );
  1543. console.debug(
  1544. msgTime.toLocalShort() +
  1545. '\tname' +
  1546. '\tcpu' +
  1547. '\tstate' +
  1548. '\tminstk' +
  1549. '\tbprio' +
  1550. '\tcprio' +
  1551. '\tnum'
  1552. );
  1553. if (statsData.tasks) {
  1554. if ($('#tasks_sect').css('visibility') === 'collapse') {
  1555. $('#tasks_sect').css('visibility', 'visible');
  1556. }
  1557. $('tbody#tasks').html('');
  1558. statsData.tasks
  1559. .sort(function (a, b) {
  1560. return b.cpu - a.cpu;
  1561. })
  1562. .forEach(showTask, msgTime);
  1563. } else if ($('#tasks_sect').css('visibility') === 'visible') {
  1564. $('tbody#tasks').empty();
  1565. $('#tasks_sect').css('visibility', 'collapse');
  1566. }
  1567. break;
  1568. case 'MESSAGING_CLASS_SYSTEM':
  1569. showMessage(msg, msgTime);
  1570. break;
  1571. case 'MESSAGING_CLASS_CFGCMD':
  1572. var msgparts = msg.message.split(/([^\n]*)\n(.*)/gs);
  1573. showCmdMessage(msgparts[1], msg.type, msgparts[2], true);
  1574. break;
  1575. case 'MESSAGING_CLASS_BT':
  1576. if ($("#cfg-audio-bt_source-sink_name").is('input')) {
  1577. var attr = $("#cfg-audio-bt_source-sink_name")[0].attributes;
  1578. var attrs = '';
  1579. for (var j = 0; j < attr.length; j++) {
  1580. if (attr.item(j).name != "type") {
  1581. attrs += `${attr.item(j).name} = "${attr.item(j).value}" `;
  1582. }
  1583. }
  1584. var curOpt = $("#cfg-audio-bt_source-sink_name")[0].value;
  1585. $("#cfg-audio-bt_source-sink_name").replaceWith(`<select id="cfg-audio-bt_source-sink_name" ${attrs}><option value="${curOpt}" data-bs-description="${curOpt}">${curOpt}</option></select> `);
  1586. }
  1587. JSON.parse(msg.message).forEach(function (btEntry) {
  1588. //<input type="text" class="form-control bg-success" placeholder="name" hasvalue="true" longopts="sink_name" shortopts="n" checkbox="false" cmdname="cfg-audio-bt_source" id="cfg-audio-bt_source-sink_name" name="cfg-audio-bt_source-sink_name">
  1589. //<select hasvalue="true" longopts="jack_behavior" shortopts="j" checkbox="false" cmdname="cfg-audio-general" id="cfg-audio-general-jack_behavior" name="cfg-audio-general-jack_behavior" class="form-control "><option>--</option><option>Headphones</option><option>Subwoofer</option></select>
  1590. if (!btExists(btEntry.name)) {
  1591. $("#cfg-audio-bt_source-sink_name").append(`<option>${btEntry.name}</option>`);
  1592. showMessage({ type: msg.type, message: `BT Audio device found: ${btEntry.name} RSSI: ${btEntry.rssi} ` }, msgTime);
  1593. }
  1594. getBTSinkOpt(btEntry.name).attr('data-bs-description', `${btEntry.name} (${btEntry.rssi}dB)`)
  1595. .attr('rssi', btEntry.rssi)
  1596. .attr('value', btEntry.name)
  1597. .text(`${btEntry.name} [${btEntry.rssi}dB]`).trigger('change');
  1598. });
  1599. $(btSinkNamesOptSel).append($(`${btSinkNamesOptSel} option`).remove().sort(function (a, b) {
  1600. console.log(`${parseInt($(a).attr('rssi'))} < ${parseInt($(b).attr('rssi'))} ? `);
  1601. return parseInt($(a).attr('rssi')) < parseInt($(b).attr('rssi')) ? 1 : -1;
  1602. }));
  1603. break;
  1604. default:
  1605. break;
  1606. }
  1607. }
  1608. setTimeout(getMessages, messageInterval);
  1609. }).fail(function (xhr, ajaxOptions, thrownError) {
  1610. if (xhr.status == 404) {
  1611. $('.orec').hide(); // system commands won't be available either
  1612. messagesHeld = true;
  1613. }
  1614. else {
  1615. handleExceptionResponse(xhr, ajaxOptions, thrownError);
  1616. }
  1617. if (xhr.status == 0 && xhr.readyState == 0) {
  1618. // probably a timeout. Target is rebooting?
  1619. setTimeout(getMessages, messageInterval * 2); // increase duration if a failure happens
  1620. }
  1621. else if (!messagesHeld) {
  1622. // 404 here means we rebooted to an old recovery
  1623. setTimeout(getMessages, messageInterval); // increase duration if a failure happens
  1624. }
  1625. }
  1626. );
  1627. /*
  1628. Minstk is minimum stack space left
  1629. Bprio is base priority
  1630. cprio is current priority
  1631. nme is name
  1632. st is task state. I provided a "typedef" that you can use to convert to text
  1633. cpu is cpu percent used
  1634. */
  1635. }
  1636. function handleRecoveryMode(data) {
  1637. const locRecovery = data.recovery ?? 0;
  1638. if (locRecovery === 1) {
  1639. recovery = true;
  1640. $('.recovery_element').show();
  1641. $('.ota_element').hide();
  1642. $('#boot-button').html('Reboot');
  1643. $('#boot-form').attr('action', '/reboot_ota.json');
  1644. } else {
  1645. if (!recovery && messagesHeld) {
  1646. messagesHeld = false;
  1647. setTimeout(getMessages, messageInterval); // increase duration if a failure happens
  1648. }
  1649. recovery = false;
  1650. $('.recovery_element').hide();
  1651. $('.ota_element').show();
  1652. $('#boot-button').html('Recovery');
  1653. $('#boot-form').attr('action', '/recovery.json');
  1654. }
  1655. }
  1656. function hasConnectionChanged(data) {
  1657. // gw: "192.168.10.1"
  1658. // ip: "192.168.10.225"
  1659. // netmask: "255.255.255.0"
  1660. // ssid: "MyTestSSID"
  1661. return (data.urc !== ConnectedTo.urc ||
  1662. data.ssid !== ConnectedTo.ssid ||
  1663. data.gw !== ConnectedTo.gw ||
  1664. data.netmask !== ConnectedTo.netmask ||
  1665. data.ip !== ConnectedTo.ip || data.rssi !== ConnectedTo.rssi)
  1666. }
  1667. function handleWifiDialog(data) {
  1668. if ($('#WifiConnectDialog').is(':visible')) {
  1669. if (ConnectedTo.ip) {
  1670. $('#ipAddress').text(ConnectedTo.ip);
  1671. }
  1672. if (ConnectedTo.ssid) {
  1673. $('#connectedToSSID').text(ConnectedTo.ssid);
  1674. }
  1675. if (ConnectedTo.gw) {
  1676. $('#gateway').text(ConnectedTo.gw);
  1677. }
  1678. if (ConnectedTo.netmask) {
  1679. $('#netmask').text(ConnectedTo.netmask);
  1680. }
  1681. if (ConnectingToSSID.Action === undefined || (ConnectingToSSID.Action && ConnectingToSSID.Action == ConnectingToActions.STS)) {
  1682. $("*[class*='connecting']").hide();
  1683. $('.connecting-status').show();
  1684. }
  1685. if (SystemConfig.ap_ssid) {
  1686. $('#apName').text(SystemConfig.ap_ssid.value);
  1687. }
  1688. if (SystemConfig.ap_pwd) {
  1689. $('#apPass').text(SystemConfig.ap_pwd.value);
  1690. }
  1691. if (!data) {
  1692. return;
  1693. }
  1694. else {
  1695. switch (data.urc) {
  1696. case connectReturnCode.OK:
  1697. if (data.ssid && data.ssid === ConnectingToSSID.ssid) {
  1698. $("*[class*='connecting']").hide();
  1699. $('.connecting-success').show();
  1700. ConnectingToSSID.Action = ConnectingToActions.STS;
  1701. }
  1702. break;
  1703. case connectReturnCode.FAIL:
  1704. //
  1705. if (ConnectingToSSID.Action != ConnectingToActions.STS && ConnectingToSSID.ssid == data.ssid) {
  1706. $("*[class*='connecting']").hide();
  1707. $('.connecting-fail').show();
  1708. }
  1709. break;
  1710. case connectReturnCode.LOST:
  1711. break;
  1712. case connectReturnCode.RESTORE:
  1713. if (ConnectingToSSID.Action != ConnectingToActions.STS && ConnectingToSSID.ssid != data.ssid) {
  1714. $("*[class*='connecting']").hide();
  1715. $('.connecting-fail').show();
  1716. }
  1717. break;
  1718. case connectReturnCode.DISC:
  1719. // that's a manual disconnect
  1720. // if ($('#wifi-status').is(':visible')) {
  1721. // $('#wifi-status').slideUp('fast', function() {});
  1722. // $('span#foot-wifi').html('');
  1723. // }
  1724. break;
  1725. default:
  1726. break;
  1727. }
  1728. }
  1729. }
  1730. }
  1731. function setIcons(offline) {
  1732. $('.material-icons').each(function (_index, entry) {
  1733. entry.textContent = entry.attributes[offline ? 'aria-label' : 'icon'].value;
  1734. });
  1735. }
  1736. function handleNetworkStatus(data) {
  1737. setIcons(!isConnected());
  1738. if (hasConnectionChanged(data) || !data.urc) {
  1739. ConnectedTo = data;
  1740. $(".if_eth").hide();
  1741. $('.if_wifi').hide();
  1742. if (!data.urc || ConnectedTo.urc != connectReturnCode.ETH) {
  1743. $('.if_wifi').show();
  1744. refreshAPHTML2();
  1745. }
  1746. else {
  1747. $(".if_eth").show();
  1748. refreshETH();
  1749. }
  1750. }
  1751. handleWifiDialog(data);
  1752. }
  1753. function batteryToIcon(voltage) {
  1754. /* Assuming Li-ion 18650s as a power source, 3.9V per cell, or above is treated
  1755. as full charge (>75% of capacity). 3.4V is empty. The gauge is loosely
  1756. following the graph here:
  1757. https://learn.adafruit.com/li-ion-and-lipoly-batteries/voltages
  1758. using the 0.2C discharge profile for the rest of the values.
  1759. */
  1760. for (const iconEntry of batIcons) {
  1761. for (const entryRanges of iconEntry.ranges) {
  1762. if (inRange(voltage, entryRanges.f, entryRanges.t)) {
  1763. return { label: iconEntry.label, icon: iconEntry.icon };
  1764. }
  1765. }
  1766. }
  1767. return { label: '▪▪▪▪', icon: "battery_full" };
  1768. }
  1769. function checkStatus() {
  1770. $.ajaxSetup({
  1771. timeout: statusInterval //Time in milliseconds
  1772. });
  1773. $.getJSON('/status.json', function (data) {
  1774. handleRecoveryMode(data);
  1775. handleNVSVisible();
  1776. handleNetworkStatus(data);
  1777. handlebtstate(data);
  1778. flashState.EventTargetStatus(data);
  1779. if(data.depth) {
  1780. depth = data.depth;
  1781. if(depth==16){
  1782. $('#cmd_opt_R').show();
  1783. }
  1784. else{
  1785. $('#cmd_opt_R').hide();
  1786. }
  1787. }
  1788. if (data.project_name && data.project_name !== '') {
  1789. project_name = data.project_name;
  1790. }
  1791. if (data.platform_name && data.platform_name !== '') {
  1792. platform_name = data.platform_name;
  1793. }
  1794. if (board_model === '') board_model = project_name;
  1795. if (board_model === '') board_model = 'Squeezelite-ESP32';
  1796. if (data.version && data.version !== '') {
  1797. versionName = data.version;
  1798. $("#navtitle").html(`${board_model}${recovery ? '<br>[recovery]' : ''}`);
  1799. $('span#foot-fw').html(`fw: <strong>${versionName}</strong>, mode: <strong>${recovery ? "Recovery" : project_name}</strong>`);
  1800. } else {
  1801. $('span#flash-status').html('');
  1802. }
  1803. if (data.Voltage) {
  1804. const bat_icon = batteryToIcon(data.Voltage);
  1805. $('#battery').html(`${getIcon(bat_icon)}`);
  1806. $('#battery').attr("aria-label", bat_icon.label);
  1807. $('#battery').attr("icon", bat_icon.icon);
  1808. $('#battery').show();
  1809. } else {
  1810. $('#battery').hide();
  1811. }
  1812. if ((data.message ?? '') != '' && prevmessage != data.message) {
  1813. // supporting older recovery firmwares - messages will come from the status.json structure
  1814. prevmessage = data.message;
  1815. showLocalMessage(data.message, 'MESSAGING_INFO')
  1816. }
  1817. is_i2c_locked = data.is_i2c_locked;
  1818. if (is_i2c_locked) {
  1819. $('flds-cfg-hw-preset').hide();
  1820. }
  1821. else {
  1822. $('flds-cfg-hw-preset').show();
  1823. }
  1824. $("button[onclick*='handleReboot']").removeClass('rebooting');
  1825. if (typeof lmsBaseUrl == "undefined" || data.lms_ip != prevLMSIP && data.lms_ip && data.lms_port) {
  1826. const baseUrl = 'http://' + data.lms_ip + ':' + data.lms_port;
  1827. prevLMSIP = data.lms_ip;
  1828. $.ajax({
  1829. url: baseUrl + '/plugins/SqueezeESP32/firmware/-check.bin',
  1830. type: 'HEAD',
  1831. dataType: 'text',
  1832. cache: false,
  1833. error: function () {
  1834. // define the value, so we don't check it any more.
  1835. lmsBaseUrl = '';
  1836. },
  1837. success: function () {
  1838. lmsBaseUrl = baseUrl;
  1839. }
  1840. });
  1841. }
  1842. $('#o_jack').css({ display: Number(data.Jack) ? 'inline' : 'none' });
  1843. setTimeout(checkStatus, statusInterval);
  1844. }).fail(function (xhr, ajaxOptions, thrownError) {
  1845. handleExceptionResponse(xhr, ajaxOptions, thrownError);
  1846. if (xhr.status == 0 && xhr.readyState == 0) {
  1847. // probably a timeout. Target is rebooting?
  1848. setTimeout(checkStatus, messageInterval * 2); // increase duration if a failure happens
  1849. }
  1850. else {
  1851. setTimeout(checkStatus, messageInterval); // increase duration if a failure happens
  1852. }
  1853. });
  1854. }
  1855. // eslint-disable-next-line no-unused-vars
  1856. window.runCommand = function (button, reboot) {
  1857. let cmdstring = button.attributes.cmdname.value;
  1858. showCmdMessage(
  1859. button.attributes.cmdname.value,
  1860. 'MESSAGING_INFO',
  1861. 'Executing.',
  1862. false
  1863. );
  1864. const fields = document.getElementById('flds-' + cmdstring);
  1865. const allfields = fields?.querySelectorAll('select,input');
  1866. if (cmdstring === 'cfg-hw-preset') return handleHWPreset(allfields, reboot);
  1867. cmdstring += ' ';
  1868. if (fields) {
  1869. for (const field of allfields) {
  1870. let qts = '';
  1871. let opt = '';
  1872. let attr = field.attributes;
  1873. let isSelect = $(field).is('select');
  1874. const hasValue = attr?.hasvalue?.value === 'true';
  1875. const validVal = (isSelect && field.value !== '--') || (!isSelect && field.value !== '');
  1876. if (!hasValue || hasValue && validVal) {
  1877. if (attr?.longopts?.value !== 'undefined') {
  1878. opt += '--' + attr?.longopts?.value;
  1879. } else if (attr?.shortopts?.value !== 'undefined') {
  1880. opt = '-' + attr.shortopts.value;
  1881. }
  1882. if (attr?.hasvalue?.value === 'true') {
  1883. if (attr?.value !== '') {
  1884. qts = /\s/.test(field.value) ? '"' : '';
  1885. cmdstring += opt + ' ' + qts + field.value + qts + ' ';
  1886. }
  1887. } else {
  1888. // this is a checkbox
  1889. if (field?.checked) {
  1890. cmdstring += opt + ' ';
  1891. }
  1892. }
  1893. }
  1894. }
  1895. }
  1896. console.log(cmdstring);
  1897. const data = {
  1898. timestamp: Date.now(),
  1899. };
  1900. data.command = cmdstring;
  1901. $.ajax({
  1902. url: '/commands.json',
  1903. dataType: 'text',
  1904. method: 'POST',
  1905. cache: false,
  1906. contentType: 'application/json; charset=utf-8',
  1907. data: JSON.stringify(data),
  1908. error: function (xhr, _ajaxOptions, thrownError) {
  1909. var cmd = JSON.parse(this.data).command;
  1910. if (xhr.status == 404) {
  1911. showCmdMessage(
  1912. cmd.substr(0, cmd.indexOf(' ')),
  1913. 'MESSAGING_ERROR',
  1914. `${recovery ? 'Limited recovery mode active. Unsupported action ' : 'Unexpected error while processing command'}`,
  1915. true
  1916. );
  1917. }
  1918. else {
  1919. handleExceptionResponse(xhr, _ajaxOptions, thrownError);
  1920. showCmdMessage(
  1921. cmd.substr(0, cmd.indexOf(' ') - 1),
  1922. 'MESSAGING_ERROR',
  1923. `Unexpected error ${(thrownError !== '') ? thrownError : 'with return status = ' + xhr.status}`,
  1924. true
  1925. );
  1926. }
  1927. },
  1928. success: function (response) {
  1929. $('.orec').show();
  1930. console.log(response);
  1931. if (
  1932. JSON.parse(response).Result === 'Success' &&
  1933. reboot
  1934. ) {
  1935. delayReboot(2500, button.attributes.cmdname.value);
  1936. }
  1937. },
  1938. });
  1939. }
  1940. function getLongOps(data, name, longopts) {
  1941. return data.values[name] !== undefined ? data.values[name][longopts] : "";
  1942. }
  1943. function getCommands() {
  1944. $.ajaxSetup({
  1945. timeout: 7000 //Time in milliseconds
  1946. });
  1947. $.getJSON('/commands.json', function (data) {
  1948. console.log(data);
  1949. $('.orec').show();
  1950. data.commands.forEach(function (command) {
  1951. if ($('#flds-' + command.name).length === 0) {
  1952. const cmdParts = command.name.split('-');
  1953. const isConfig = cmdParts[0] === 'cfg';
  1954. const targetDiv = '#tab-' + cmdParts[0] + '-' + cmdParts[1];
  1955. let innerhtml = '';
  1956. innerhtml += `<div class="card text-white mb-3"><div class="card-header">${command.help.encodeHTML().replace(/\n/g, '<br />')}</div><div class="card-body"><fieldset id="flds-${command.name}">`;
  1957. if (command.argtable) {
  1958. command.argtable.forEach(function (arg) {
  1959. let placeholder = arg.datatype || '';
  1960. const ctrlname = command.name + '-' + arg.longopts;
  1961. const curvalue = getLongOps(data, command.name, arg.longopts);
  1962. let attributes = 'hasvalue=' + arg.hasvalue + ' ';
  1963. attributes += 'longopts="' + arg.longopts + '" ';
  1964. attributes += 'shortopts="' + arg.shortopts + '" ';
  1965. attributes += 'checkbox=' + arg.checkbox + ' ';
  1966. attributes += 'cmdname="' + command.name + '" ';
  1967. attributes +=
  1968. 'id="' +
  1969. ctrlname +
  1970. '" name="' +
  1971. ctrlname +
  1972. '" hasvalue="' +
  1973. arg.hasvalue +
  1974. '" ';
  1975. let extraclass = arg.mincount > 0 ? 'bg-success' : '';
  1976. if (arg.glossary === 'hidden') {
  1977. attributes += ' style="visibility: hidden;"';
  1978. }
  1979. if (arg.checkbox) {
  1980. innerhtml += `<div class="form-check"><label class="form-check-label"><input type="checkbox" ${attributes} class="form-check-input ${extraclass}" value="" >${arg.glossary.encodeHTML()}</label>`;
  1981. } else {
  1982. innerhtml += `<div class="form-group" ><label for="${ctrlname}">${arg.glossary.encodeHTML()}</label>`;
  1983. if (placeholder.includes('|')) {
  1984. extraclass = placeholder.startsWith('+') ? ' multiple ' : '';
  1985. placeholder = placeholder
  1986. .replace('<', '')
  1987. .replace('=', '')
  1988. .replace('>', '');
  1989. innerhtml += `<select ${attributes} class="form-control ${extraclass}" >`;
  1990. placeholder = '--|' + placeholder;
  1991. placeholder.split('|').forEach(function (choice) {
  1992. innerhtml += '<option >' + choice + '</option>';
  1993. });
  1994. innerhtml += '</select>';
  1995. } else {
  1996. innerhtml += `<input type="text" class="form-control ${extraclass}" placeholder="${placeholder}" ${attributes}>`;
  1997. }
  1998. }
  1999. innerhtml += `${arg.checkbox ? '</div>' : ''}<small class="form-text text-muted">Previous value: ${arg.checkbox ? (curvalue ? 'Checked' : 'Unchecked') : (curvalue || '')}</small>${arg.checkbox ? '' : '</div>'}`;
  2000. });
  2001. }
  2002. innerhtml += `<div style="margin-top: 16px;">
  2003. <div class="toast hide" role="alert" aria-live="assertive" aria-atomic="true" id="toast_${command.name}">
  2004. <div class="toast-header">
  2005. <strong class="mr-auto">Result</strong
  2006. <button type="button" class="btn-close" data-bs-dismiss="toast" aria-label="Close"></button>
  2007. </div>
  2008. <div class="toast-body" id="msg_${command.name}"></div>
  2009. </div>`;
  2010. if (isConfig) {
  2011. innerhtml +=
  2012. `<button type="submit" class="btn btn-info sclk" id="btn-save-${command.name}" cmdname="${command.name}">Save</button>
  2013. <button type="submit" class="btn btn-warning cclk" id="btn-commit-${command.name}" cmdname="${command.name}">Apply</button>`;
  2014. } else {
  2015. innerhtml += `<button type="submit" class="btn btn-success sclk" id="btn-run-${command.name}" cmdname="${command.name}">Execute</button>`;
  2016. }
  2017. innerhtml += '</div></fieldset></div></div>';
  2018. if (isConfig) {
  2019. $(targetDiv).append(innerhtml);
  2020. } else {
  2021. $('#commands-list').append(innerhtml);
  2022. }
  2023. }
  2024. });
  2025. $(".sclk").off('click').on('click', function () { runCommand(this, false); });
  2026. $(".cclk").off('click').on('click', function () { runCommand(this, true); });
  2027. data.commands.forEach(function (command) {
  2028. $('[cmdname=' + command.name + ']:input').val('');
  2029. $('[cmdname=' + command.name + ']:checkbox').prop('checked', false);
  2030. if (command.argtable) {
  2031. command.argtable.forEach(function (arg) {
  2032. const ctrlselector = '#' + command.name + '-' + arg.longopts;
  2033. const ctrlValue = getLongOps(data, command.name, arg.longopts);
  2034. if (arg.checkbox) {
  2035. $(ctrlselector)[0].checked = ctrlValue;
  2036. } else {
  2037. if (ctrlValue !== undefined) {
  2038. $(ctrlselector)
  2039. .val(ctrlValue)
  2040. .trigger('change');
  2041. }
  2042. if (
  2043. $(ctrlselector)[0].value.length === 0 &&
  2044. (arg.datatype || '').includes('|')
  2045. ) {
  2046. $(ctrlselector)[0].value = '--';
  2047. }
  2048. }
  2049. });
  2050. }
  2051. });
  2052. loadPresets();
  2053. }).fail(function (xhr, ajaxOptions, thrownError) {
  2054. if (xhr.status == 404) {
  2055. $('.orec').hide();
  2056. }
  2057. else {
  2058. handleExceptionResponse(xhr, ajaxOptions, thrownError);
  2059. }
  2060. $('#commands-list').empty();
  2061. });
  2062. }
  2063. function getConfig() {
  2064. $.ajaxSetup({
  2065. timeout: 7000 //Time in milliseconds
  2066. });
  2067. $.getJSON('/config.json', function (entries) {
  2068. $('#nvsTable tr').remove();
  2069. const data = (entries.config ? entries.config : entries);
  2070. SystemConfig = data;
  2071. commandBTSinkName = '';
  2072. Object.keys(data)
  2073. .sort()
  2074. .forEach(function (key) {
  2075. let val = data[key].value;
  2076. if (key === 'autoexec1') {
  2077. /* call new function to parse the squeezelite options */
  2078. processSqueezeliteCommandLine(val);
  2079. } else if (key === 'host_name') {
  2080. val = val.replaceAll('"', '');
  2081. $('input#dhcp-name1').val(val);
  2082. $('input#dhcp-name2').val(val);
  2083. if ($('#cmd_opt_n').length == 0) {
  2084. $('#cmd_opt_n').val(val);
  2085. }
  2086. document.title = val;
  2087. hostName = val;
  2088. } else if (key === 'rel_api') {
  2089. releaseURL = val;
  2090. }
  2091. else if (key === 'enable_airplay') {
  2092. $("#s_airplay").css({ display: isEnabled(val) ? 'inline' : 'none' })
  2093. }
  2094. else if (key === 'enable_cspot') {
  2095. $("#s_cspot").css({ display: isEnabled(val) ? 'inline' : 'none' })
  2096. }
  2097. else if (key == 'preset_name') {
  2098. preset_name = val;
  2099. }
  2100. else if (key == 'board_model') {
  2101. board_model = val;
  2102. }
  2103. $('tbody#nvsTable').append(
  2104. '<tr>' +
  2105. '<td>' +
  2106. key +
  2107. '</td>' +
  2108. "<td class='value'>" +
  2109. "<input type='text' class='form-control nvs' id='" +
  2110. key +
  2111. "' nvs_type=" +
  2112. data[key].type +
  2113. ' >' +
  2114. '</td>' +
  2115. '</tr>'
  2116. );
  2117. $('input#' + key).val(data[key].value);
  2118. });
  2119. if(commandBTSinkName.length > 0) {
  2120. // persist the sink name found in the autoexec1 command line
  2121. $('#cfg-audio-bt_source-sink_name').val(commandBTSinkName);
  2122. }
  2123. $('tbody#nvsTable').append(
  2124. "<tr><td><input type='text' class='form-control' id='nvs-new-key' placeholder='new key'></td><td><input type='text' class='form-control' id='nvs-new-value' placeholder='new value' nvs_type=33 ></td></tr>"
  2125. );
  2126. if (entries.gpio) {
  2127. $('#pins').show();
  2128. $('tbody#gpiotable tr').remove();
  2129. entries.gpio.forEach(function (gpioEntry) {
  2130. $('tbody#gpiotable').append(
  2131. '<tr class=' +
  2132. (gpioEntry.fixed ? 'table-secondary' : 'table-primary') +
  2133. '><th scope="row">' +
  2134. gpioEntry.group +
  2135. '</th><td>' +
  2136. gpioEntry.name +
  2137. '</td><td>' +
  2138. gpioEntry.gpio +
  2139. '</td><td>' +
  2140. (gpioEntry.fixed ? 'Fixed' : 'Configuration') +
  2141. '</td></tr>'
  2142. );
  2143. });
  2144. }
  2145. else {
  2146. $('#pins').hide();
  2147. }
  2148. }).fail(function (xhr, ajaxOptions, thrownError) {
  2149. handleExceptionResponse(xhr, ajaxOptions, thrownError);
  2150. });
  2151. }
  2152. function processSqueezeliteCommandLine(val) {
  2153. const parsed = parseSqueezeliteCommandLine(val);
  2154. if (parsed.output.toUpperCase().startsWith('I2S')) {
  2155. handleTemplateTypeRadio('i2s');
  2156. } else if (parsed.output.toUpperCase().startsWith('SPDIF')) {
  2157. handleTemplateTypeRadio('spdif');
  2158. } else if (parsed.output.toUpperCase().startsWith('BT')) {
  2159. if(parsed.otherOptions.btname){
  2160. commandBTSinkName= parsed.otherOptions.btname;
  2161. }
  2162. handleTemplateTypeRadio('bt');
  2163. }
  2164. Object.keys(parsed.options).forEach(function (key) {
  2165. const option = parsed.options[key];
  2166. if (!$(`#cmd_opt_${key}`).hasOwnProperty('checked')) {
  2167. $(`#cmd_opt_${key}`).val(option);
  2168. } else {
  2169. $(`#cmd_opt_${key}`)[0].checked = option;
  2170. }
  2171. });
  2172. if (parsed.options.hasOwnProperty('u')) {
  2173. // parse -u v[:i] and check the appropriate radio button with id #resample_v
  2174. const [resampleValue, resampleInterpolation] = parsed.options.u.split(':');
  2175. $(`#resample_${resampleValue}`).prop('checked', true);
  2176. // if resampleinterpolation is set, check resample_i checkbox
  2177. if (resampleInterpolation) {
  2178. $('#resample_i').prop('checked', true);
  2179. }
  2180. }
  2181. if (parsed.options.hasOwnProperty('s')) {
  2182. // parse -u v[:i] and check the appropriate radio button with id #resample_v
  2183. if(parsed.options.s === '-disable'){
  2184. $('#disable-squeezelite')[0].checked = true;
  2185. }
  2186. else {
  2187. $('#disable-squeezelite')[0].checked = false;
  2188. }
  2189. }
  2190. }
  2191. function showLocalMessage(message, severity) {
  2192. const msg = {
  2193. message: message,
  2194. type: severity,
  2195. };
  2196. showMessage(msg, new Date());
  2197. }
  2198. function showMessage(msg, msgTime) {
  2199. let color = 'table-success';
  2200. if (msg.type === 'MESSAGING_WARNING') {
  2201. color = 'table-warning';
  2202. if (messageseverity === 'MESSAGING_INFO') {
  2203. messageseverity = 'MESSAGING_WARNING';
  2204. }
  2205. } else if (msg.type === 'MESSAGING_ERROR') {
  2206. if (
  2207. messageseverity === 'MESSAGING_INFO' ||
  2208. messageseverity === 'MESSAGING_WARNING'
  2209. ) {
  2210. messageseverity = 'MESSAGING_ERROR';
  2211. }
  2212. color = 'table-danger';
  2213. }
  2214. if (++messagecount > 0) {
  2215. $('#msgcnt').removeClass('badge-success');
  2216. $('#msgcnt').removeClass('badge-warning');
  2217. $('#msgcnt').removeClass('badge-danger');
  2218. $('#msgcnt').addClass(pillcolors[messageseverity]);
  2219. $('#msgcnt').text(messagecount);
  2220. }
  2221. $('#syslogTable').append(
  2222. "<tr class='" +
  2223. color +
  2224. "'>" +
  2225. '<td>' +
  2226. msgTime.toLocalShort() +
  2227. '</td>' +
  2228. '<td>' +
  2229. msg.message.encodeHTML() +
  2230. '</td>' +
  2231. '</tr>'
  2232. );
  2233. }
  2234. function inRange(x, min, max) {
  2235. return (x - min) * (x - max) <= 0;
  2236. }
  2237. function sleep(ms) {
  2238. return new Promise(resolve => setTimeout(resolve, ms));
  2239. }