// Custom scripts let currentSettings = null; function backupSettings() { const data = "data:text/json;charset=utf-8," + encodeURIComponent(JSON.stringify(currentSettings)); const a = document.createElement("a"); a.setAttribute("href", data); a.setAttribute("download", "iGateConfigurationBackup.json"); a.click(); } document.getElementById("backup").onclick = backupSettings; document.getElementById("restore").onclick = function (e) { e.preventDefault(); document.querySelector("input[type=file]").click(); }; document.querySelector("input[type=file]").onchange = function () { const files = document.querySelector("input[type=file]").files; if (!files.length) return; const file = files.item(0); const reader = new FileReader(); reader.readAsText(file); reader.onload = () => { const data = JSON.parse(reader.result); loadSettings(data); }; }; function fetchSettings() { fetch("/configuration.json") .then((response) => response.json()) .then((settings) => { loadSettings(settings); }) .catch((err) => { console.error(err); alert(`Failed to load configuration`); }); } function loadSettings(settings) { currentSettings = settings; // General document.getElementById("callsign").value = settings.callsign; document.getElementById("beacon.comment").value = settings.beacon.comment; document.getElementById("beacon.path").value = settings.beacon.path; document.getElementById("beacon.symbol").value = settings.beacon.symbol; document.getElementById("beacon.overlay").value = settings.beacon.overlay; document.getElementById("personalNote").value = settings.personalNote; document.getElementById("action.symbol").value = settings.beacon.overlay + settings.beacon.symbol; document.querySelector(".list-networks").innerHTML = ""; // Networks const wifiNetworks = settings.wifi.AP || []; const networksContainer = document.querySelector(".list-networks"); let networkCount = 0; wifiNetworks.forEach((network) => { const networkElement = document.createElement("div"); networkElement.classList.add("row", "network", "border-bottom", "py-2"); // Increment the name, id, and for attributes const attributeName = `wifi.AP.${networkCount}`; networkElement.innerHTML = `
`; networksContainer.appendChild(networkElement); networkCount++; }); document.getElementById("startupDelay").value = settings.startupDelay; // APRS-IS document.getElementById("aprs_is.active").checked = settings.aprs_is.active; document.getElementById("aprs_is.messagesToRF").checked = settings.aprs_is.messagesToRF; document.getElementById("aprs_is.objectsToRF").checked = settings.aprs_is.objectsToRF; document.getElementById("aprs_is.server").value = settings.aprs_is.server; document.getElementById("aprs_is.port").value = settings.aprs_is.port; document.getElementById("aprs_is.filter").value = settings.aprs_is.filter; document.getElementById("aprs_is.passcode").value = settings.aprs_is.passcode; APRSISCheckbox.checked = settings.aprs_is.active; APRSISGateMessages.disabled = !APRSISCheckbox.checked; APRSISGateObjects.disabled = !APRSISCheckbox.checked; APRSISServer.disabled = !APRSISCheckbox.checked; APRSISPort.disabled = !APRSISCheckbox.checked; APRSISPasscode.disabled = !APRSISCheckbox.checked; APRSISFilter.disabled = !APRSISCheckbox.checked; // Beacon document.getElementById("beacon.latitude").value = settings.beacon.latitude; document.getElementById("beacon.longitude").value = settings.beacon.longitude; document.getElementById("beacon.interval").value = settings.beacon.interval; document.getElementById("other.rememberStationTime").value = settings.other.rememberStationTime; document.getElementById("beacon.sendViaAPRSIS").checked = settings.beacon.sendViaAPRSIS; document.getElementById("beacon.sendViaRF").checked = settings.beacon.sendViaRF; document.getElementById("beacon.beaconFreq").value = settings.beacon.beaconFreq; BeaconingViaRFCheckbox.checked = settings.beacon.sendViaRF; BeaconingFrequency.disabled = !BeaconingViaRFCheckbox.checked; document.getElementById("beacon.statusActive").checked = settings.beacon.statusActive; document.getElementById("beacon.statusPacket").value = settings.beacon.statusPacket; StatusCheckbox.checked = settings.beacon.statusActive; StatusPacket.disabled = !StatusCheckbox.checked; document.getElementById("beacon.gpsActive").checked = settings.beacon.gpsActive; document.getElementById("beacon.gpsAmbiguity").checked = settings.beacon.gpsAmbiguity; // Black List document.getElementById("blacklist").value = settings.blacklist; // Digi document.getElementById("digi.mode").value = settings.digi.mode; document.getElementById("digi.ecoMode").value = settings.digi.ecoMode; // LoRa document.getElementById("lora.rxActive").checked = settings.lora.rxActive; document.getElementById("lora.rxFreq").value = settings.lora.rxFreq; document.getElementById("lora.rxSpreadingFactor").value = settings.lora.rxSpreadingFactor; document.getElementById("lora.rxCodingRate4").value = settings.lora.rxCodingRate4; document.getElementById("lora.rxSignalBandwidth").value = settings.lora.rxSignalBandwidth; document.getElementById("lora.txActive").checked = settings.lora.txActive; document.getElementById("lora.txFreq").value = settings.lora.txFreq; document.getElementById("lora.txSpreadingFactor").value = settings.lora.txSpreadingFactor; document.getElementById("lora.txCodingRate4").value = settings.lora.txCodingRate4; document.getElementById("lora.txSignalBandwidth").value = settings.lora.txSignalBandwidth; document.getElementById("lora.power").value = settings.lora.power; // Display document.getElementById("display.alwaysOn").checked = settings.display.alwaysOn; document.getElementById("display.turn180").checked = settings.display.turn180; document.getElementById("display.timeout").value = settings.display.timeout; DisplayAlwaysOnCheckbox.checked = settings.display.alwaysOn; DisplayTimeout.disabled = DisplayAlwaysOnCheckbox.checked; // BATTERY document.getElementById("battery.sendInternalVoltage").checked = settings.battery.sendInternalVoltage; document.getElementById("battery.monitorInternalVoltage").checked = settings.battery.monitorInternalVoltage; document.getElementById("battery.internalSleepVoltage").value = settings.battery.internalSleepVoltage.toFixed(1); MonitorInternalVoltageCheckbox.checked = settings.battery.monitorInternalVoltage; MonitorInternalSleepVoltage.disabled = !MonitorInternalVoltageCheckbox.checked; document.getElementById("battery.sendExternalVoltage").checked = settings.battery.sendExternalVoltage; document.getElementById("battery.externalVoltagePin").value = settings.battery.externalVoltagePin; document.getElementById("battery.voltageDividerR1").value = settings.battery.voltageDividerR1.toFixed(1); document.getElementById("battery.voltageDividerR2").value = settings.battery.voltageDividerR2.toFixed(1); SendExternalVoltageCheckbox.checked = settings.battery.sendExternalVoltage; ExternalVoltagePin.disabled = !SendExternalVoltageCheckbox.checked; ExternalVoltageDividerR1.disabled = !SendExternalVoltageCheckbox.checked; ExternalVoltageDividerR2.disabled = !SendExternalVoltageCheckbox.checked; document.getElementById("battery.monitorExternalVoltage").checked = settings.battery.monitorExternalVoltage; document.getElementById("battery.externalSleepVoltage").value = settings.battery.externalSleepVoltage.toFixed(1); MonitorExternalVoltageCheckbox.checked = settings.battery.monitorExternalVoltage; MonitorExternalSleepVoltage.disabled = !MonitorExternalVoltageCheckbox.checked; document.getElementById("battery.sendVoltageAsTelemetry").checked = settings.battery.sendVoltageAsTelemetry; // TELEMETRY WX SENSOR document.getElementById("wxsensor.active").checked = settings.wxsensor.active; document.getElementById("wxsensor.heightCorrection").value = settings.wxsensor.heightCorrection; document.getElementById("wxsensor.temperatureCorrection").value = settings.wxsensor.temperatureCorrection.toFixed(1); TelemetryCheckbox.checked = settings.wxsensor.active; TelemetryHeightCorrection.disabled = !TelemetryCheckbox.checked; TelemetryTempCorrection.disabled = !TelemetryCheckbox.checked; // SYSLOG document.getElementById("syslog.active").checked = settings.syslog.active; document.getElementById("syslog.server").value = settings.syslog.server; document.getElementById("syslog.port").value = settings.syslog.port; document.getElementById("syslog.logBeaconOverTCPIP").checked = settings.syslog.logBeaconOverTCPIP; SyslogCheckbox.checked = settings.syslog.active; SyslogServer.disabled = !SyslogCheckbox.checked; SyslogPort.disabled = !SyslogCheckbox.checked; SyslogBeaconOverTCPIP.disabled = !SyslogCheckbox.checked; // TNC if (settings.tnc) { document.getElementById("tnc.enableServer").checked = settings.tnc.enableServer; document.getElementById("tnc.enableSerial").checked = settings.tnc.enableSerial; document.getElementById("tnc.acceptOwn").checked = settings.tnc.acceptOwn; document.getElementById("tnc.aprsBridgeActive").checked = settings.tnc.aprsBridgeActive; } // MQTT document.getElementById("mqtt.active").checked = settings.mqtt.active; document.getElementById("mqtt.server").value = settings.mqtt.server; document.getElementById("mqtt.topic").value = settings.mqtt.topic; document.getElementById("mqtt.username").value = settings.mqtt.username; document.getElementById("mqtt.password").value = settings.mqtt.password; document.getElementById("mqtt.port").value = settings.mqtt.port; document.getElementById("mqtt.beaconOverMqtt").checked = settings.mqtt.beaconOverMqtt; MqttCheckbox.checked = settings.mqtt.active; MqttServer.disabled = !MqttCheckbox.check; MqttTopic.disabled = !MqttCheckbox.check; MqttUsername.disabled = !MqttCheckbox.check; MqttPassword.disabled = !MqttCheckbox.check; MqttPort.disabled = !MqttCheckbox.check; MqttBeaconOverMqtt.disabled = !MqttCheckbox.check; // Reboot document.getElementById("other.rebootMode").checked = settings.other.rebootMode; document.getElementById("other.rebootModeTime").value = settings.other.rebootModeTime; RebootModeCheckbox.checked = settings.other.rebootMode; RebootModeTime.disabled = !RebootModeCheckbox.check; // WiFi Auto AP document.getElementById("wifi.autoAP.password").value = settings.wifi.autoAP.password; document.getElementById("wifi.autoAP.timeout").value = settings.wifi.autoAP.timeout; // OTA document.getElementById("ota.username").value = settings.ota.username; document.getElementById("ota.password").value = settings.ota.password; // Webadmin document.getElementById("webadmin.active").checked = settings.webadmin.active; document.getElementById("webadmin.username").value = settings.webadmin.username; document.getElementById("webadmin.password").value = settings.webadmin.password; WebadminCheckbox.checked = settings.webadmin.active; WebadminUsername.disabled = !WebadminCheckbox.check; WebadminPassword.disabled = !WebadminCheckbox.check; // Management over APRS document.getElementById("remoteManagement.managers").value = settings.remoteManagement.managers; document.getElementById("remoteManagement.rfOnly").checked = settings.remoteManagement.rfOnly; // NTP document.getElementById("ntp.server").value = settings.ntp.server; document.getElementById("ntp.gmtCorrection").value = settings.ntp.gmtCorrection; // Experimental document.getElementById("other.backupDigiMode").checked = settings.other.backupDigiMode; updateImage(); } function showToast(message) { const el = document.querySelector('#toast'); el.querySelector('.toast-body').innerHTML = message; (new bootstrap.Toast(el)).show(); } document.getElementById('send-beacon').addEventListener('click', function (e) { e.preventDefault(); fetch("/action?type=send-beacon", { method: "POST" }); showToast("Your beacon will be sent in a moment.
This action will be ignored if you have APRSIS and LoRa TX disabled!"); }); document.getElementById('reboot').addEventListener('click', function (e) { e.preventDefault(); fetch("/action?type=reboot", { method: "POST" }); showToast("Your device will be rebooted in a while"); }); function updateImage() { const value = document.getElementById("beacon.overlay").value + document.getElementById("beacon.symbol").value; const image = document.querySelector("img"); switch (value) { case "L&": image.src = ""; break; case "L#": image.src = ""; break; case "L_": image.src = ""; break; case "La": image.src = ""; break; } } // Beaconing Switches const BeaconingViaRFCheckbox = document.querySelector('input[name="beacon.sendViaRF"]'); const BeaconingFrequency = document.querySelector('select[name="beacon.beaconFreq"]'); BeaconingViaRFCheckbox.addEventListener("change", function() { BeaconingFrequency.disabled = !this.checked; }); // Status Switch const StatusCheckbox = document.querySelector('input[name="beacon.statusActive"]'); const StatusPacket = document.querySelector('input[name="beacon.statusPacket"]'); StatusCheckbox.addEventListener("change", function() { StatusPacket.disabled = !this.checked; }); // APRS-IS Switches const APRSISCheckbox = document.querySelector('input[name="aprs_is.active"]'); const APRSISGateMessages = document.querySelector('input[name="aprs_is.messagesToRF"]'); const APRSISGateObjects = document.querySelector('input[name="aprs_is.objectsToRF"]'); const APRSISServer = document.querySelector('input[name="aprs_is.server"]'); const APRSISPort = document.querySelector('input[name="aprs_is.port"]'); const APRSISPasscode = document.querySelector('input[name="aprs_is.passcode"]'); const APRSISFilter = document.querySelector('input[name="aprs_is.filter"]'); APRSISCheckbox.addEventListener("change", function() { APRSISGateMessages.disabled = !this.checked; APRSISGateObjects.disabled = !this.checked; APRSISServer.disabled = !this.checked; APRSISPort.disabled = !this.checked; APRSISPasscode.disabled = !this.checked; APRSISFilter.disabled = !this.checked; }); // Display Switches const DisplayAlwaysOnCheckbox = document.querySelector('input[name="display.alwaysOn"]'); const DisplayTimeout = document.querySelector('input[name="display.timeout"]'); DisplayAlwaysOnCheckbox.addEventListener("change", function () { DisplayTimeout.disabled = this.checked; }); // Battery Switches const MonitorInternalVoltageCheckbox = document.querySelector('input[name="battery.monitorInternalVoltage"]'); const MonitorInternalSleepVoltage = document.querySelector('input[name="battery.internalSleepVoltage"]'); MonitorInternalVoltageCheckbox.addEventListener("change", function () { MonitorInternalSleepVoltage.disabled = !this.checked; }); const MonitorExternalVoltageCheckbox = document.querySelector('input[name="battery.monitorExternalVoltage"]'); const MonitorExternalSleepVoltage = document.querySelector('input[name="battery.externalSleepVoltage"]'); MonitorExternalVoltageCheckbox.addEventListener("change", function () { MonitorExternalSleepVoltage.disabled = !this.checked; }); const SendExternalVoltageCheckbox = document.querySelector('input[name="battery.sendExternalVoltage"]'); const ExternalVoltagePin = document.querySelector('input[name="battery.externalVoltagePin"]'); const ExternalVoltageDividerR1 = document.querySelector('input[name="battery.voltageDividerR1"]'); const ExternalVoltageDividerR2 = document.querySelector('input[name="battery.voltageDividerR2"]'); SendExternalVoltageCheckbox.addEventListener("change", function () { ExternalVoltagePin.disabled = !this.checked; ExternalVoltageDividerR1.disabled = !this.checked; ExternalVoltageDividerR2.disabled = !this.checked; }); // Telemetry Switches const TelemetryCheckbox = document.querySelector('input[name="wxsensor.active"]'); const TelemetryHeightCorrection = document.querySelector('input[name="wxsensor.heightCorrection"]'); const TelemetryTempCorrection = document.querySelector('input[name="wxsensor.temperatureCorrection"]'); TelemetryCheckbox.addEventListener("change", function () { TelemetryHeightCorrection.disabled = !this.checked; TelemetryTempCorrection.disabled = !this.checked; }); // Syslog Switches const SyslogCheckbox = document.querySelector('input[name="syslog.active"]'); const SyslogServer = document.querySelector('input[name="syslog.server"]'); const SyslogPort = document.querySelector('input[name="syslog.port"]'); const SyslogBeaconOverTCPIP = document.querySelector('input[name="syslog.logBeaconOverTCPIP"]'); SyslogCheckbox.addEventListener("change", function () { SyslogServer.disabled = !this.checked; SyslogPort.disabled = !this.checked; SyslogBeaconOverTCPIP.disabled = !this.checked }); // MQTT Switches const MqttCheckbox = document.querySelector('input[name="mqtt.active"]'); const MqttServer = document.querySelector('input[name="mqtt.server"]'); const MqttTopic = document.querySelector('input[name="mqtt.topic"]'); const MqttUsername = document.querySelector('input[name="mqtt.username"]'); const MqttPassword = document.querySelector('input[name="mqtt.password"]'); const MqttPort = document.querySelector('input[name="mqtt.port"]'); const MqttBeaconOverMqtt = document.querySelector('input[name="mqtt.beaconOverMqtt"]'); MqttCheckbox.addEventListener("change", function () { MqttServer.disabled = !this.checked; MqttTopic.disabled = !this.checked; MqttUsername.disabled = !this.checked; MqttPassword.disabled = !this.checked; MqttPort.disabled = !this.checked; MqttBeaconOverMqtt.disabled = !this.checked; }); // Reboot Switches const RebootModeCheckbox = document.querySelector('input[name="other.rebootMode"]'); const RebootModeTime = document.querySelector('input[name="other.rebootModeTime"]'); RebootModeCheckbox.addEventListener("change", function() { RebootModeTime.disabled = !this.checked; }); // Web Admin Switches const WebadminCheckbox = document.querySelector('input[name="webadmin.active"]'); const WebadminUsername = document.querySelector('input[name="webadmin.username"]'); const WebadminPassword = document.querySelector('input[name="webadmin.password"]'); WebadminCheckbox.addEventListener("change", function () { WebadminUsername.disabled = !this.checked; WebadminPassword.disabled = !this.checked; }); document.querySelector(".new button").addEventListener("click", function () { const networksContainer = document.querySelector(".list-networks"); let networkCount = document.querySelectorAll(".network").length; const networkElement = document.createElement("div"); networkElement.classList.add("row", "network", "border-bottom", "py-2"); // Increment the name, id, and for attributes const attributeName = `wifi.AP.${networkCount}`; networkElement.innerHTML = `
`; networksContainer.appendChild(networkElement); networkCount++; // Add the new network element to the end of the document document.querySelector(".new").before(networkElement); }); document .getElementById("action.symbol") .addEventListener("change", function () { const value = document.getElementById("action.symbol").value; document.getElementById("beacon.overlay").value = value[0]; document.getElementById("beacon.symbol").value = value[1]; updateImage(); }); const form = document.querySelector("form"); const saveModal = new bootstrap.Modal(document.getElementById("saveModal"), { backdrop: "static", keyboard: false, }); const savedModal = new bootstrap.Modal( document.getElementById("savedModal"), {} ); function checkConnection() { const controller = new AbortController(); setTimeout(() => controller.abort(), 2000); fetch("/status?_t=" + Date.now(), { signal: controller.signal }) .then(() => { saveModal.hide(); savedModal.show(); setTimeout(function () { savedModal.hide(); }, 3000); fetchSettings(); }) .catch((err) => { setTimeout(checkConnection, 0); }); } form.addEventListener("submit", async (event) => { event.preventDefault(); document.getElementById("wifi.APs").value = document.querySelectorAll(".network").length; fetch(form.action, { method: form.method, body: new FormData(form), }); saveModal.show(); setTimeout(checkConnection, 2000); }); fetchSettings(); function loadReceivedPackets(packets) { if (packets) { document.querySelector('#received-packets tbody').innerHTML = ''; const container = document.querySelector("#received-packets tbody"); container.innerHTML = ''; const date = new Date(); packets.forEach((packet) => { const element = document.createElement("tr"); element.innerHTML = ` ${packet.rxTime} ${packet.packet} ${packet.RSSI} ${packet.SNR} `; container.appendChild(element); }) } setTimeout(fetchReceivedPackets, 15000); } function fetchReceivedPackets() { fetch("/received-packets.json") .then((response) => response.json()) .then((packets) => { loadReceivedPackets(packets); }) .catch((err) => { console.error(err); console.error(`Failed to load received packets`); }); } document.querySelector('a[href="/received-packets"]').addEventListener('click', function (e) { e.preventDefault(); document.getElementById('received-packets').classList.remove('d-none'); document.getElementById('configuration').classList.add('d-none'); document.querySelector('button[type=submit]').remove(); fetchReceivedPackets(); })