index.html 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271
  1. <!DOCTYPE html>
  2. <html lang="en" xml:lang="en">
  3. <head>
  4. <link rel="icon" href="favicon.ico?v=$COMMIT_HASH" type="image/x-icon">
  5. <link rel="apple-touch-icon" href="watermeter.svg?v=$COMMIT_HASH" />
  6. <title>AI on the edge</title>
  7. <meta charset="UTF-8" />
  8. <meta name="viewport" content="width=device-width, initial-scale=1">
  9. <link rel="stylesheet" href="style.css?v=$COMMIT_HASH" type="text/css" >
  10. <link href="firework.css?v=$COMMIT_HASH" rel="stylesheet">
  11. <script type="text/javascript" src="common.js?v=$COMMIT_HASH"></script>
  12. <script type="text/javascript" src="readconfigcommon.js?v=$COMMIT_HASH"></script>
  13. <script type="text/javascript" src="readconfigparam.js?v=$COMMIT_HASH"></script>
  14. <script type="text/javascript" src="jquery-3.6.0.min.js?v=$COMMIT_HASH"></script>
  15. <script type="text/javascript" src="firework.js?v=$COMMIT_HASH"></script>
  16. <script>
  17. var streamPopup;
  18. var streamFlashlight = false;
  19. var streamWindowFeatures =
  20. 'channelmode=no,directories=no,fullscreen=no,' +
  21. 'location=no,dependent=yes,menubar=no,resizable=no,scrollbars=no,' +
  22. 'status=no,toolbar=no,titlebar=no,' +
  23. 'left=10,top=260,width=640px,height=480px';
  24. function loadPage(page) {
  25. console.log("loadPage( " + page + " )");
  26. if (streamPopup) { // Ensure that stream popup is closed because it's blocking web interface
  27. streamPopup.close();
  28. }
  29. asyncPageLoad(page);
  30. }
  31. async function asyncPageLoad(page ) {
  32. console.log(" loading " + page + " ...");
  33. document.cookie = "page="+page + "; path=/";
  34. document.getElementById('maincontent').src = page;
  35. [].forEach.call(document.querySelectorAll('.submenu'), function (el) {
  36. el.style.visibility = 'hidden';
  37. });
  38. }
  39. function resetMenu() {
  40. [].forEach.call(document.querySelectorAll('.submenu'), function (el) {
  41. el.style.visibility = 'visible';
  42. });
  43. }
  44. function getCookie(cname) {
  45. let name = cname + "=";
  46. let decodedCookie = decodeURIComponent(document.cookie);
  47. let ca = decodedCookie.split(';');
  48. for(let i = 0; i <ca.length; i++) {
  49. let c = ca[i];
  50. while (c.charAt(0) == ' ') {
  51. c = c.substring(1);
  52. }
  53. if (c.indexOf(name) == 0) {
  54. return c.substring(name.length, c.length);
  55. }
  56. }
  57. return "";
  58. }
  59. </script>
  60. <style>
  61. /* Add these styles to your existing CSS file or in a <style> tag in the head */
  62. .footer {
  63. display: flex;
  64. justify-content: space-between;
  65. align-items: center;
  66. padding: 20px;
  67. background-color: #d8d8d8;
  68. margin-top: 20px;
  69. }
  70. .footer-section {
  71. display: flex;
  72. align-items: center;
  73. }
  74. .footer-section img {
  75. width: 24px;
  76. height: 24px;
  77. margin-left: 10px;
  78. }
  79. .donation-cards img {
  80. height: 20px;
  81. margin-right: 5px;
  82. }
  83. </style>
  84. </head>
  85. <body>
  86. <div class="main">
  87. <table style="border: none; width:100%">
  88. <tr>
  89. <td style="padding-right: 10px;"><img style="width:64px; height:64px" src="favicon.ico?v=$COMMIT_HASH"></td>
  90. <td><h1 id="id_title"> Digitizer - AI on the edge</h1>
  91. <h2>An ESP32 all inclusive neural network recognition system for meter Digitization</h2>
  92. </td>
  93. </tr>
  94. </table>
  95. <div class="menu" onmouseover="resetMenu()">
  96. <ul>
  97. <li><a href="#" onclick="loadPage('overview.html?v=$COMMIT_HASH');">Overview</a></li>
  98. <li><a>Settings <i class="arrow down"></i></a>
  99. <ul class="submenu" style="width: 260px">
  100. <li style="width: 260px"><a href="#" onclick="loadPage('prevalue_set.html?v=$COMMIT_HASH');">Set "Previous Value"</a></li>
  101. <li style="width: 260px"><a>References <i class="arrow right"></i></a>
  102. <ul style="width: 350px">
  103. <li style="width: 350px"><a href="#" onclick="loadPage('edit_reference.html?v=$COMMIT_HASH');">Reference Image and Camera Settings</a></li>
  104. <li style="width: 350px"><a href="#" onclick="loadPage('edit_alignment.html?v=$COMMIT_HASH');">Alignment Markers</a></li>
  105. </ul>
  106. </li>
  107. <li style="width: 260px"><a>Regions of Interest (ROIs)<i class="arrow right"></i></a>
  108. <ul>
  109. <li><a href="#" onclick="loadPage('edit_digits.html?v=$COMMIT_HASH');">Digit ROI</a></li>
  110. <li><a href="#" onclick="loadPage('edit_analog.html?v=$COMMIT_HASH');">Analog ROI</a></li>
  111. </ul>
  112. </li>
  113. <li style="width: 260px"><a href="#" onclick="loadPage('edit_config.html?v=$COMMIT_HASH');">Configuration</a></li>
  114. </ul>
  115. <li><a>Data<i class="arrow down"></i></a>
  116. <ul class="submenu">
  117. <li><a href="#" onclick="loadPage(getDomainname() + '/value?full');">Recognition</a></li>
  118. <li><a>Livestream <i class="arrow right"></i></a>
  119. <ul>
  120. <li><a href="#" onclick="start_livestream(false);">Live Stream (Light off)</a></li>
  121. <li><a href="#" onclick="start_livestream(true);">Live Stream (Light on)</a></li>
  122. </ul>
  123. </li>
  124. <li><a href="#" onclick="loadPage('graph.html?v=$COMMIT_HASH');">Data Graph</a></li>
  125. <li><a href="#" onclick="loadPage('data.html?v=$COMMIT_HASH');">Data Table</a></li>
  126. <li><a href="#" onclick="loadPage(getDomainname() + '/fileserver/log/data/');">Data Files</a></li>
  127. </ul>
  128. </li>
  129. <li><a>System <i class="arrow down"></i></a>
  130. <ul class="submenu">
  131. <li><a href="#" onclick="loadPage('backup.html?v=$COMMIT_HASH');">Backup/Restore</a></li>
  132. <li><a href="#" onclick="loadPage('ota_page.html?v=$COMMIT_HASH');">OTA Update</a></li>
  133. <li><a href="#" onclick="loadPage('log.html?v=$COMMIT_HASH');">Log Viewer</a></li>
  134. <li><a href="#" onclick="loadPage(getDomainname() + '/fileserver/');">File Server</a></li>
  135. <li><a href="#" onclick="loadPage('reboot_page.html?v=$COMMIT_HASH');">Reboot</a></li>
  136. <li><a href="#" onclick="loadPage('info.html?v=$COMMIT_HASH');">Info</a></li>
  137. <li><a href="https://jomjol.github.io/AI-on-the-edge-device-docs/" target="_blank">Help</a></li>
  138. </ul>
  139. </li>
  140. <li id="ManualControl" style="display:none;"><a>Manual Control <i class="arrow down"></i></a> <!-- Workaround: Hide menu if no entry is available -->
  141. <ul class="submenu" style="width: 300px">
  142. <!--<li><a href="#" onclick="flow_start()">Start Round</a></li>--> <!-- Needs to be adapted on code side first to ensure proper user feedback -->
  143. <li id="HASendDiscovery" style="width: 300px" style="display:none;"><a href="#" onclick="HA_send_discovery()">Resend Homeassistant Discovery</a></li>
  144. </ul>
  145. </li>
  146. </ul>
  147. </div>
  148. <iframe title="maincontent" name="maincontent" class="iframe" id="maincontent"></iframe>
  149. <span id="Version" style="font-size: 10px; margin-top: -5px;padding-left: 10px;">Loading version...</span>
  150. <div class="footer">
  151. <div class="footer-section">
  152. <span>Support & Contact Us</span>
  153. <a href="https://github.com/jomjol/AI-on-the-edge-device" target="_blank" title="GitHub">
  154. <img src="https://github.com/jomjol/AI-on-the-edge-device/images/github-logo.png" alt="GitHub">
  155. </a>
  156. <img src="https://github.com/jomjol/AI-on-the-edge-device/images/gmail-logo.png" alt="Email">
  157. </a>
  158. <a href="https://github.com/jomjol/AI-on-the-edge-device/discussions" target="_blank" title="GitHub">
  159. <img src="https://github.com/jomjol/AI-on-the-edge-device/images/discussion-logo" alt="GitHub">
  160. </a>
  161. </div>
  162. <div class="footer-section">
  163. <span>Donations</span>
  164. <a href="https://www.paypal.com/donate?hosted_button_id=8TRSVYNYKDSWL" target="_blank" title="Donate via PayPal">
  165. <img src="https://github.com/jomjol/AI-on-the-edge-device/images/paypal.png" alt="PayPal" style="width: 60px; height: auto;">
  166. </a>
  167. </div>
  168. </div>
  169. <script type="text/javascript">
  170. LoadHostname();
  171. LoadFwVersion();
  172. LoadWebUiVersion();
  173. HA_send_discovery_visibility();
  174. if (getCookie("page") == "" || getCookie("page") == "reboot_page.html?v=$COMMIT_HASH") {
  175. document.cookie = "page=overview.html?v=$COMMIT_HASH" + "; path=/";
  176. }
  177. console.log("Loading page: " + getCookie("page"));
  178. document.getElementById('maincontent').src = getCookie("page");
  179. /*
  180. function flow_start() {
  181. var url = getDomainname() + '/flow_start';
  182. var xhttp = new XMLHttpRequest();
  183. xhttp.onreadystatechange = function() {
  184. if (this.readyState == 4 && this.status == 200) {
  185. if (xhttp.responseText.substring(0,3) == "001") {
  186. firework.launch('Flow start triggered', 'success', 5000);
  187. window.location.reload();
  188. }
  189. else if (xhttp.responseText.substring(0,3) == "002") {
  190. firework.launch('Flow start scheduled. Start after round is completed', 'success', 5000);
  191. }
  192. else if (xhttp.responseText.substring(0,3) == "099") {
  193. firework.launch('Flow start triggered, but start not possible (no flow task available)', 'danger', 5000);
  194. }
  195. }
  196. }
  197. xhttp.open("GET", url, true);
  198. xhttp.send();
  199. }
  200. */
  201. function HA_send_discovery_visibility() {
  202. loadConfig(domainname);
  203. ParseConfig();
  204. category = getConfigCategory();
  205. param = getConfigParameters();
  206. if (category["MQTT"]["enabled"] && param["MQTT"]["HomeassistantDiscovery"].value1 == "true") {
  207. document.getElementById("ManualControl").style.display="";
  208. document.getElementById("HASendDiscovery").style.display="";
  209. }
  210. }
  211. function HA_send_discovery() {
  212. console.log("Homeassistant Discovery topic sending scheduled");
  213. var url = getDomainname() + '/mqtt_publish_discovery';
  214. var xhttp = new XMLHttpRequest();
  215. xhttp.onreadystatechange = function() {
  216. if (this.readyState == 4 && this.status == 200) {
  217. firework.launch('Sending Homeassistant discovery topics scheduled. It will get sent in the step "Publish to MQTT" of the next digitization round', 'success', 5000);
  218. }
  219. }
  220. xhttp.open("GET", url, true);
  221. xhttp.send();
  222. }
  223. function start_livestream(streamFlashlight) {
  224. if (streamPopup) {
  225. streamPopup.close();
  226. }
  227. if (streamFlashlight) {
  228. streamPopup = window.open(getDomainname() + '/stream?flashlight=true','LivestreamWithlight',streamWindowFeatures);
  229. }
  230. else {
  231. streamPopup = window.open(getDomainname() + '/stream','Livestream',streamWindowFeatures);
  232. }
  233. streamPopup.focus();
  234. }
  235. </script>
  236. </body>
  237. </html>