FreeDATA/gui/preload-main.js
dj2ls 3501b4f952 waiting for buffer
it seems we need to wait a little bit for our modulation data. There are some moments where audio output is stuck a little bit. Maybe python is sometimes faster than the buffer allocation or its a threading problem
2021-12-26 18:41:43 +01:00

1285 lines
51 KiB
JavaScript

const path = require('path')
const {
ipcRenderer
} = require('electron')
const sock = require('./sock.js');
const daemon = require('./daemon.js');
const fs = require('fs');
const {
locatorToLatLng,
distance,
bearingDistance,
latLngToLocator
} = require('qth-locator');
const os = require('os');
// https://stackoverflow.com/a/26227660
var appDataFolder = process.env.APPDATA || (process.platform == 'darwin' ? process.env.HOME + '/Library/Application Support' : process.env.HOME + "/.config")
var configFolder = path.join(appDataFolder, "FreeDATA");
var configPath = path.join(configFolder, 'config.json')
const config = require(configPath);
// START INTERVALL COMMAND EXECUTION FOR STATES
setInterval(daemon.getDaemonState, 1000)
setInterval(sock.getTncState, 150)
setInterval(sock.getRxBuffer, 1000)
setInterval(sock.getMsgRxBuffer, 1000)
// UPDATE FFT DEMO
/*
updateFFT = function(fft) {
var fft = Array.from({
length: 2048
}, () => Math.floor(Math.random() * 10));
spectrum.addData(fft);
console.log(fft)
}
setInterval(updateFFT, 250)
*/
// WINDOW LISTENER
window.addEventListener('DOMContentLoaded', () => {
/*
// ENABLE BOOTSTRAP POPOVERS EVERYWHERE
// https://getbootstrap.com/docs/5.0/components/popovers/#example-enable-popovers-everywhere
var popoverTriggerList = [].slice.call(document.querySelectorAll('[data-bs-toggle="popover"]'))
var popoverList = popoverTriggerList.map(function (popoverTriggerEl) {
return new bootstrap.Popover(popoverTriggerEl)
})
*/
// ENABLE TOOLTIPS EVERYWHERE
// https://getbootstrap.com/docs/5.1/components/tooltips/
var tooltipTriggerList = [].slice.call(document.querySelectorAll('[data-bs-toggle="tooltip"]'))
var tooltipList = tooltipTriggerList.map(function (tooltipTriggerEl) {
return new bootstrap.Tooltip(tooltipTriggerEl)
})
// LOAD SETTINGS
document.getElementById("tnc_adress").value = config.tnc_host
document.getElementById("tnc_port").value = config.tnc_port
document.getElementById("myCall").value = config.mycall
document.getElementById("myGrid").value = config.mygrid
document.getElementById('hamlib_deviceid').value = config.deviceid
document.getElementById('hamlib_deviceport').value = config.deviceport
document.getElementById('hamlib_serialspeed').value = config.serialspeed
document.getElementById('hamlib_ptt_protocol').value = config.pttprotocol
document.getElementById('hamlib_deviceport_advanced').value = config.deviceport
document.getElementById('hamlib_serialspeed_advanced').value = config.serialspeed
document.getElementById('hamlib_ptt_protocol_advanced').value = config.pttprotocol
document.getElementById('hamlib_pttspeed_advanced').value = config.pttspeed
document.getElementById('hamlib_ptt_port_advanced').value = config.pttport
document.getElementById('hamlib_databits_advanced').value = config.data_bits
document.getElementById('hamlib_stopbits_advanced').value = config.stop_bits
document.getElementById('hamlib_handshake_advanced').value = config.handshake
document.getElementById("beaconInterval").value = config.beacon_interval
if (config.spectrum == 'waterfall') {
document.getElementById("waterfall-scatter-switch1").checked = true
document.getElementById("waterfall-scatter-switch2").checked = false
document.getElementById("scatter").style.visibility = 'hidden';
document.getElementById("waterfall").style.visibility = 'visible';
document.getElementById("waterfall").style.height = '100%';
} else {
document.getElementById("waterfall-scatter-switch1").checked = false
document.getElementById("waterfall-scatter-switch2").checked = true
document.getElementById("scatter").style.visibility = 'visible';
document.getElementById("waterfall").style.visibility = 'hidden';
document.getElementById("waterfall").style.height = '0px';
}
// remote tnc
if (config.tnclocation == 'remote') {
document.getElementById("local-remote-switch1").checked = false
document.getElementById("local-remote-switch2").checked = true
document.getElementById("remote-tnc-field").style.visibility = 'visible';
} else {
document.getElementById("local-remote-switch1").checked = true
document.getElementById("local-remote-switch2").checked = false
document.getElementById("remote-tnc-field").style.visibility = 'hidden';
}
// Create spectrum object on canvas with ID "waterfall"
global.spectrum = new Spectrum(
"waterfall", {
spectrumPercent: 0
});
// SETUP OF SCATTER DIAGRAM
/*
global.data = {
datasets: [{
label: 'Scatter Dataset',
data: [{
x: 0,
y: 0
}],
backgroundColor: 'rgb(255, 99, 132)'
}],
};
var ctx = document.getElementById('scatter').getContext('2d');
global.myChart = new Chart(ctx, {
type: 'scatter',
data: data,
options: {
animation: false,
legend: {
display: false
},
scales: {
display: false,
grid: {
display: false
},
x: {
type: 'linear',
position: 'bottom',
display: false,
min: -50,
max: 50
},
y: {
display: false,
min: -50,
max: 50
}
}
}
});
*/
/*
// advanced Hamlib Settings
document.getElementById("advancedHamlibSettingsButton").addEventListener("click", () => {
advancedHamlibSettingsModal
});
*/
// on click waterfall scatter toggle view
// waterfall
document.getElementById("waterfall-scatter-switch1").addEventListener("click", () => {
document.getElementById("scatter").style.visibility = 'hidden';
document.getElementById("waterfall").style.visibility = 'visible';
document.getElementById("waterfall").style.height = '100%';
config.spectrum = 'waterfall'
fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
});
// scatter
document.getElementById("waterfall-scatter-switch2").addEventListener("click", () => {
document.getElementById("scatter").style.visibility = 'visible';
document.getElementById("waterfall").style.visibility = 'hidden';
document.getElementById("waterfall").style.height = '0px';
config.spectrum = 'scatter'
fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
});
// on click remote tnc toggle view
document.getElementById("local-remote-switch1").addEventListener("click", () => {
document.getElementById("local-remote-switch1").checked = true
document.getElementById("local-remote-switch2").checked = false
document.getElementById("remote-tnc-field").style.visibility = 'hidden';
config.tnclocation = 'localhost'
fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
});
document.getElementById("local-remote-switch2").addEventListener("click", () => {
document.getElementById("local-remote-switch1").checked = false
document.getElementById("local-remote-switch2").checked = true
document.getElementById("remote-tnc-field").style.visibility = 'visible';
config.tnclocation = 'remote'
fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
});
// on change port and host
document.getElementById("tnc_adress").addEventListener("change", () => {
console.log(document.getElementById("tnc_adress").value)
config.tnc_host = document.getElementById("tnc_adress").value
config.daemon_host = document.getElementById("tnc_adress").value
fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
});
// on change ping callsign
document.getElementById("dxCall").addEventListener("change", () => {
document.getElementById("dataModalDxCall").value = document.getElementById("dxCall").value
});
// on change ping callsign
document.getElementById("dataModalDxCall").addEventListener("change", () => {
document.getElementById("dxCall").value = document.getElementById("dataModalDxCall").value
});
document.getElementById("tnc_port").addEventListener("change", () => {
config.tnc_port = document.getElementById("tnc_port").value
config.daemon_port = parseInt(document.getElementById("tnc_port").value) + 1
fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
});
// saveMyCall button clicked
document.getElementById("saveMyCall").addEventListener("click", () => {
callsign = document.getElementById("myCall").value
config.mycall = callsign.toUpperCase()
fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
sock.saveMyCall(callsign)
});
// saveMyGrid button clicked
document.getElementById("saveMyGrid").addEventListener("click", () => {
grid = document.getElementById("myGrid").value
config.mygrid = grid
fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
sock.saveMyGrid(grid)
});
// startPing button clicked
document.getElementById("sendPing").addEventListener("click", () => {
var dxcallsign = document.getElementById("dxCall").value
dxcallsign = dxcallsign.toUpperCase()
sock.sendPing(dxcallsign)
});
// dataModalstartPing button clicked
document.getElementById("dataModalSendPing").addEventListener("click", () => {
var dxcallsign = document.getElementById("dataModalDxCall").value
dxcallsign = dxcallsign.toUpperCase()
sock.sendPing(dxcallsign)
});
// sendCQ button clicked
document.getElementById("sendCQ").addEventListener("click", () => {
sock.sendCQ()
});
// Start beacon button clicked
document.getElementById("startBeacon").addEventListener("click", () => {
interval = document.getElementById("beaconInterval").value
config.beacon_interval = interval
fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
sock.startBeacon(interval)
});
// Stop beacon button clicked
document.getElementById("stopBeacon").addEventListener("click", () => {
sock.stopBeacon()
});
// startTNC button clicked
document.getElementById("startTNC").addEventListener("click", () => {
var rx_audio = document.getElementById("audio_input_selectbox").value
var tx_audio = document.getElementById("audio_output_selectbox").value
var deviceid = document.getElementById("hamlib_deviceid").value
var deviceport = document.getElementById("hamlib_deviceport").value
var serialspeed = document.getElementById("hamlib_serialspeed").value
var pttprotocol = document.getElementById("hamlib_ptt_protocol").value
var pttport = document.getElementById("hamlib_ptt_port_advanced").value
var pttspeed = document.getElementById('hamlib_pttspeed_advanced').value
var data_bits = document.getElementById('hamlib_databits_advanced').value
var stop_bits = document.getElementById('hamlib_stopbits_advanced').value
var handshake = document.getElementById('hamlib_handshake_advanced').value
config.deviceid = deviceid
config.deviceport = deviceport
config.serialspeed = serialspeed
config.pttprotocol = pttprotocol
config.pttport = pttport
config.pttspeed = pttspeed
config.data_bits = data_bits
config.stop_bits = stop_bits
config.handshake = handshake
fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
// collapse settings screen
var collapseFirstRow = new bootstrap.Collapse(document.getElementById('collapseFirstRow'), {toggle: false})
collapseFirstRow.hide()
var collapseSecondRow = new bootstrap.Collapse(document.getElementById('collapseSecondRow'), {toggle: false})
collapseSecondRow.hide()
var collapseThirdRow = new bootstrap.Collapse(document.getElementById('collapseThirdRow'), {toggle: false})
collapseThirdRow.show()
var collapseFourthRow = new bootstrap.Collapse(document.getElementById('collapseFourthRow'), {toggle: false})
collapseFourthRow.show()
daemon.startTNC(rx_audio, tx_audio, deviceid, deviceport, pttprotocol, pttport, serialspeed, pttspeed, data_bits, stop_bits, handshake)
setTimeout(function() {
sock.saveMyCall(config.mycall);
}, 3000);
setTimeout(function() {
sock.saveMyGrid(config.mygrid);
}, 3500);
})
// stopTNC button clicked
document.getElementById("stopTNC").addEventListener("click", () => {
daemon.stopTNC()
// collapse settings screen
var collapseFirstRow = new bootstrap.Collapse(document.getElementById('collapseFirstRow'), {toggle: false})
collapseFirstRow.show()
var collapseSecondRow = new bootstrap.Collapse(document.getElementById('collapseSecondRow'), {toggle: false})
collapseSecondRow.show()
var collapseThirdRow = new bootstrap.Collapse(document.getElementById('collapseThirdRow'), {toggle: false})
collapseThirdRow.hide()
var collapseFourthRow = new bootstrap.Collapse(document.getElementById('collapseFourthRow'), {toggle: false})
collapseFourthRow.hide()
})
// openDataModule button clicked
// not necessesary at this time beacuse bootstrap handles this
// document.getElementById("openDataModule").addEventListener("click", () => {
// var transmitFileSidebar = document.getElementById('transmitFileSidebar')
// var bstransmitFileSidebar = new bootstrap.Offcanvas(transmitFileSidebar)
// bstransmitFileSidebar.show()
//})
// openReceivedFiles button clicked
// not necessesary at this time beacuse bootstrap handles this
//document.getElementById("openReceivedFiles").addEventListener("click", () => {
// var transmitFileSidebar = document.getElementById('transmitFileSidebar')
// var bstransmitFileSidebar = new bootstrap.Offcanvas(transmitFileSidebar)
// bstransmitFileSidebar.show()
//})
// TEST HAMLIB
document.getElementById("testHamlib").addEventListener("click", () => {
var deviceid = document.getElementById("hamlib_deviceid").value
var deviceport = document.getElementById("hamlib_deviceport").value
var serialspeed = document.getElementById("hamlib_serialspeed").value
var pttprotocol = document.getElementById("hamlib_ptt_protocol").value
var pttport = deviceport
var pttspeed = serialspeed
var data_bits = "8"
var stop_bits = "1"
var handshake = "None"
daemon.testHamlib(deviceid, deviceport, serialspeed, pttprotocol, pttport, pttspeed, data_bits, stop_bits, handshake)
})
// TEST HAMLIB ADVANCED
document.getElementById("testHamlibAdvanced").addEventListener("click", () => {
var deviceid = document.getElementById("hamlib_deviceid").value
var deviceport = document.getElementById("hamlib_deviceport_advanced").value
document.getElementById("hamlib_deviceport").value = deviceport
var serialspeed = document.getElementById("hamlib_serialspeed_advanced").value
document.getElementById("hamlib_serialspeed").value = serialspeed
var pttprotocol = document.getElementById("hamlib_ptt_protocol_advanced").value
document.getElementById("hamlib_ptt_protocol").value = pttprotocol
var pttport = document.getElementById("hamlib_deviceport").value
var pttspeed = document.getElementById("hamlib_serialspeed_advanced").value
var data_bits = document.getElementById("hamlib_databits_advanced").value
var stop_bits = document.getElementById("hamlib_stopbits_advanced").value
var handshake = document.getElementById("hamlib_handshake_advanced").value
daemon.testHamlib(deviceid, deviceport, serialspeed, pttprotocol, pttport, pttspeed, data_bits, stop_bits, handshake)
})
// START TRANSMISSION
document.getElementById("startTransmission").addEventListener("click", () => {
//document.getElementById("transmitFileSidebar").style.width = "0px";
/* not neccessary at this time because handled by bootstap inside html
var transmitFileSidebar = document.getElementById('transmitFileSidebar')
var bstransmitFileSidebar = new bootstrap.Offcanvas(transmitFileSidebar)
bstransmitFileSidebar.show()
*/
var fileList = document.getElementById("dataModalFile").files;
var reader = new FileReader();
//reader.readAsBinaryString(fileList[0]);
reader.readAsDataURL(fileList[0]);
reader.onload = function(e) {
// binary data
var data = e.target.result
console.log(data)
let Data = {
command: "sendFile",
dxcallsign: document.getElementById("dataModalDxCall").value.toUpperCase(),
mode: document.getElementById("datamode").value,
frames: document.getElementById("framesperburst").value,
filetype: fileList[0].type,
filename: fileList[0].name,
data: data,
checksum: '123123123',
};
ipcRenderer.send('run-tnc-command', Data);
};
reader.onerror = function(e) {
// error occurred
console.log('Error : ' + e.type);
};
})
// START TRANSMISSION
document.getElementById("stopTransmission").addEventListener("click", () => {
let Data = {
command: "stopTransmission"
};
ipcRenderer.send('run-tnc-command', Data);
})
// OPEN CHAT MODULE
document.getElementById("openRFChat").addEventListener("click", () => {
let Data = {
command: "openRFChat"
};
ipcRenderer.send('request-show-chat-window', Data);
})
})
ipcRenderer.on('action-update-tnc-state', (event, arg) => {
// update FFT
if (typeof(arg.fft) !== 'undefined') {
var array = JSON.parse("[" + arg.fft + "]");
spectrum.addData(array[0]);
}
// TOE TIME OF EXECUTION --> How many time needs a command to be executed until data arrives
// deactivated this feature, beacuse its useless at this time. maybe it is getting more interesting, if we are working via network
// but for this we need to find a nice place for this on the screen
/*
if (typeof(arg.toe) == 'undefined') {
var toe = 0
} else {
var toe = arg.toe
}
document.getElementById("toe").innerHTML = toe + ' ms'
*/
// DATA STATE
global.rxBufferLengthTnc = arg.rx_buffer_length
// SCATTER DIAGRAM PLOTTING
//global.myChart.destroy();
//console.log(arg.scatter.length)
const config = {
plugins: {
legend: {
display: false,
},
tooltip: {
enabled: false
},
annotation: {
annotations: {
line1: {
type: 'line',
yMin: 0,
yMax: 0,
borderColor: 'rgb(255, 99, 132)',
borderWidth: 2,
},
line2: {
type: 'line',
xMin: 0,
xMax: 0,
borderColor: 'rgb(255, 99, 132)',
borderWidth: 2,
}
}
},
},
animations: false,
scales: {
x: {
type: 'linear',
position: 'bottom',
display: true,
min: -80,
max: 80,
ticks: {
display: false
}
},
y: {
display: true,
min: -80,
max: 80,
ticks: {
display: false,
}
}
}
}
var data = arg.scatter
var newdata = {
datasets: [{
//label: 'constellation diagram',
data: data,
options: config,
backgroundColor: 'rgb(255, 99, 132)'
}],
};
if (typeof(arg.scatter) == 'undefined') {
var scatterSize = 0
} else {
var scatterSize = arg.scatter.length
}
if (global.data != newdata && scatterSize > 0) {
try {
global.myChart.destroy();
} catch (e) {
// myChart not yet created
}
global.data = newdata
var ctx = document.getElementById('scatter').getContext('2d');
global.myChart = new Chart(ctx, {
type: 'scatter',
data: global.data,
options: config
});
}
// PTT STATE
if (arg.ptt_state == 'True') {
document.getElementById("ptt_state").className = "btn btn-danger";
} else if (arg.ptt_state == 'False') {
document.getElementById("ptt_state").className = "btn btn-success";
} else {
document.getElementById("ptt_state").className = "btn btn-secondary"
}
// BUSY STATE
if (arg.busy_state == 'BUSY') {
document.getElementById("busy_state").className = "btn btn-danger";
document.getElementById("startTransmission").disabled = true
document.getElementById("stopTransmission").disabled = false
} else if (arg.busy_state == 'IDLE') {
document.getElementById("busy_state").className = "btn btn-success";
} else {
document.getElementById("busy_state").className = "btn btn-secondary"
document.getElementById("startTransmission").disabled = true
document.getElementById("stopTransmission").disabled = false
}
// ARQ STATE
if (arg.arq_state == 'DATA') {
document.getElementById("arq_state").className = "btn btn-warning";
document.getElementById("startTransmission").disabled = true
document.getElementById("stopTransmission").disabled = false
} else if (arg.arq_state == 'IDLE') {
document.getElementById("arq_state").className = "btn btn-secondary";
document.getElementById("startTransmission").disabled = false
document.getElementById("stopTransmission").disabled = true
} else {
document.getElementById("arq_state").className = "btn btn-secondary"
document.getElementById("startTransmission").disabled = true
document.getElementById("stopTransmission").disabled = false
}
// BEACON STATE
if (arg.beacon_state == 'True') {
document.getElementById("startBeacon").className = "btn btn-success spinner-grow"
document.getElementById("startBeacon").disabled = true
document.getElementById("beaconInterval").disabled = true
document.getElementById("stopBeacon").disabled = false
} else if (arg.beacon_state == 'False') {
document.getElementById("startBeacon").className = "btn btn-success"
document.getElementById("startBeacon").disabled = false
document.getElementById("beaconInterval").disabled = false
document.getElementById("stopBeacon").disabled = true
} else {
document.getElementById("startBeacon").className = "btn btn-success"
document.getElementById("startBeacon").disabled = false
document.getElementById("stopBeacon").disabled = true
document.getElementById("beaconInterval").disabled = false
}
// RMS
document.getElementById("rms_level").setAttribute("aria-valuenow", arg.rms_level)
document.getElementById("rms_level").setAttribute("style", "width:" + arg.rms_level + "%;")
// CHANNEL STATE
if (arg.channel_state == 'RECEIVING_SIGNALLING') {
document.getElementById("signalling_state").className = "btn btn-success";
document.getElementById("data_state").className = "btn btn-secondary";
} else if (arg.channel_state == 'SENDING_SIGNALLING') {
document.getElementById("signalling_state").className = "btn btn-danger";
document.getElementById("data_state").className = "btn btn-secondary";
} else if (arg.channel_state == 'RECEIVING_DATA') {
document.getElementById("signalling_state").className = "btn btn-secondary";
document.getElementById("data_state").className = "btn btn-success";
} else if (arg.channel_state == 'SENDING_DATA') {
document.getElementById("signalling_state").className = "btn btn-secondary";
document.getElementById("data_state").className = "btn btn-danger";
} else {
document.getElementById("signalling_state").className = "btn btn-secondary"
document.getElementById("busy_state").className = "btn btn-secondary"
}
// SET FREQUENCY
document.getElementById("frequency").innerHTML = arg.frequency
// SET MODE
document.getElementById("mode").innerHTML = arg.mode
// SET BANDWITH
document.getElementById("bandwith").innerHTML = arg.bandwith
// SET BYTES PER MINUTE
if (typeof(arg.arq_bytes_per_minute) == 'undefined') {
var arq_bytes_per_minute = 0
} else {
var arq_bytes_per_minute = arg.arq_bytes_per_minute
}
document.getElementById("bytes_per_min").innerHTML = arq_bytes_per_minute
// SET TOTAL BYTES
if (typeof(arg.total_bytes) == 'undefined') {
var total_bytes = 0
} else {
var total_bytes = arg.total_bytes
}
document.getElementById("total_bytes").innerHTML = total_bytes
document.getElementById("transmission_progress").setAttribute("aria-valuenow", arg.arq_transmission_percent)
document.getElementById("transmission_progress").setAttribute("style", "width:" + arg.arq_transmission_percent + "%;")
// UPDATE HEARD STATIONS
var tbl = document.getElementById("heardstations");
document.getElementById("heardstations").innerHTML = ''
if (typeof(arg.stations) == 'undefined') {
var heardStationsLength = 0
} else {
var heardStationsLength = arg.stations.length
}
for (i = 0; i < heardStationsLength; i++) {
// first we update the PING window
if (arg.stations[i]['DXCALLSIGN'] == document.getElementById("dxCall").value) {
var dxGrid = arg.stations[i]['DXGRID']
var myGrid = document.getElementById("myGrid").value
try {
var dist = parseInt(distance(myGrid, dxGrid)) + ' km';
document.getElementById("pingDistance").innerHTML = dist
document.getElementById("dataModalPingDistance").innerHTML = dist
} catch {
document.getElementById("pingDistance").innerHTML = '---'
document.getElementById("dataModalPingDistance").innerHTML = '---'
}
document.getElementById("pingDB").innerHTML = arg.stations[i]['SNR']
document.getElementById("dataModalPingDB").innerHTML = arg.stations[i]['SNR']
}
// now we update the heard stations list
var row = document.createElement("tr");
//https://stackoverflow.com/q/51421470
//https://stackoverflow.com/a/847196
timestampRaw = arg.stations[i]['TIMESTAMP']
var date = new Date(timestampRaw * 1000);
var hours = date.getHours();
var minutes = "0" + date.getMinutes();
var seconds = "0" + date.getSeconds();
var datetime = hours + ':' + minutes.substr(-2) + ':' + seconds.substr(-2);
var timestamp = document.createElement("td");
var timestampText = document.createElement('span');
timestampText.innerText = datetime
timestamp.appendChild(timestampText);
var frequency = document.createElement("td");
var frequencyText = document.createElement('span');
frequencyText.innerText = arg.stations[i]['FREQUENCY']
frequency.appendChild(frequencyText);
var dxCall = document.createElement("td");
var dxCallText = document.createElement('span');
dxCallText.innerText = arg.stations[i]['DXCALLSIGN']
dxCall.appendChild(dxCallText);
var dxGrid = document.createElement("td");
var dxGridText = document.createElement('span');
dxGridText.innerText = arg.stations[i]['DXGRID']
dxGrid.appendChild(dxGridText);
var gridDistance = document.createElement("td");
var gridDistanceText = document.createElement('span');
try {
gridDistanceText.innerText = parseInt(distance(document.getElementById("myGrid").value, arg.stations[i]['DXGRID'])) + ' km';
} catch {
gridDistanceText.innerText = '---'
}
gridDistance.appendChild(gridDistanceText);
var dataType = document.createElement("td");
var dataTypeText = document.createElement('span');
if(arg.stations[i]['DATATYPE'] == 'DATA-CHANNEL'){
dataTypeText.innerText = 'DATA-C'
dataType.appendChild(dataTypeText);
}
if (dataTypeText.innerText == 'CQ CQ CQ') {
row.classList.add("table-success");
}
if (dataTypeText.innerText == 'DATA-C') {
row.classList.add("table-warning");
}
if (dataTypeText.innerText == 'BEACON') {
row.classList.add("table-light");
}
if (dataTypeText.innerText == 'PING') {
row.classList.add("table-info");
}
if (dataTypeText.innerText == 'PING-ACK') {
row.classList.add("table-primary");
}
var snr = document.createElement("td");
var snrText = document.createElement('span');
snrText.innerText = arg.stations[i]['SNR']
snr.appendChild(snrText);
var offset = document.createElement("td");
var offsetText = document.createElement('span');
offsetText.innerText = arg.stations[i]['OFFSET']
offset.appendChild(offsetText);
row.appendChild(timestamp);
row.appendChild(frequency);
row.appendChild(dxCall);
row.appendChild(dxGrid);
row.appendChild(gridDistance);
row.appendChild(dataType);
row.appendChild(snr);
row.appendChild(offset);
tbl.appendChild(row);
}
// DISPLAY INFO TOASTS
if (typeof(arg.info) == 'undefined') {
var infoLength = 0
} else {
var infoLength = arg.info.length
}
for (i = 0; i < infoLength; i++) {
// SENDING CQ TOAST
if (arg.info[i] == "CQ;SENDING"){
var toastCQsending = document.getElementById('toastCQsending')
var toast = bootstrap.Toast.getOrCreateInstance(toastCQsending) // Returns a Bootstrap toast instance
toast.show()
}
// RECEIVING CQ TOAST
if (arg.info[i] == "CQ;RECEIVING"){
var toastCQreceiving = document.getElementById('toastCQreceiving')
var toast = bootstrap.Toast.getOrCreateInstance(toastCQreceiving) // Returns a Bootstrap toast instance
toast.show()
}
// RECEIVING BEACON TOAST
if (arg.info[i] == "BEACON;RECEIVING"){
var toastBEACONreceiving = document.getElementById('toastBEACONreceiving')
var toast = bootstrap.Toast.getOrCreateInstance(toastBEACONreceiving) // Returns a Bootstrap toast instance
toast.show()
}
// SENDING PING TOAST
if (arg.info[i] == "PING;SENDING"){
var toastPINGsending = document.getElementById('toastPINGsending')
var toast = bootstrap.Toast.getOrCreateInstance(toastPINGsending) // Returns a Bootstrap toast instance
toast.show()
}
// RECEIVING PING TOAST
if (arg.info[i] == "PING;RECEIVING"){
var toastPINGreceiving = document.getElementById('toastPINGreceiving')
var toast = bootstrap.Toast.getOrCreateInstance(toastPINGreceiving) // Returns a Bootstrap toast instance
toast.show()
}
// RECEIVING PING ACK TOAST
if (arg.info[i] == "PING;RECEIVEDACK"){
var toastPINGreceivedACK = document.getElementById('toastPINGreceivedACK')
var toast = bootstrap.Toast.getOrCreateInstance(toastPINGreceivedACK) // Returns a Bootstrap toast instance
toast.show()
}
// DATACHANNEL OPENING TOAST
if (arg.info[i] == "DATACHANNEL;OPENING"){
var toastDATACHANNELopening = document.getElementById('toastDATACHANNELopening')
var toast = bootstrap.Toast.getOrCreateInstance(toastDATACHANNELopening) // Returns a Bootstrap toast instance
toast.show()
}
// DATACHANNEL OPEN TOAST
if (arg.info[i] == "DATACHANNEL;OPEN"){
var toastDATACHANNELopen = document.getElementById('toastDATACHANNELopen')
var toast = bootstrap.Toast.getOrCreateInstance(toastDATACHANNELopen) // Returns a Bootstrap toast instance
toast.show()
}
// DATACHANNEL RECEIVEDOPENER TOAST
if (arg.info[i] == "DATACHANNEL;RECEIVEDOPENER"){
var toastDATACHANNELreceivedopener = document.getElementById('toastDATACHANNELreceivedopener')
var toast = bootstrap.Toast.getOrCreateInstance(toastDATACHANNELreceivedopener) // Returns a Bootstrap toast instance
toast.show()
}
// DATACHANNEL FAILED TOAST
if (arg.info[i] == "DATACHANNEL;FAILED"){
var toastDATACHANNELfailed = document.getElementById('toastDATACHANNELfailed')
var toast = bootstrap.Toast.getOrCreateInstance(toastDATACHANNELfailed) // Returns a Bootstrap toast instance
toast.show()
}
// ARQ RECEIVING TOAST
if (arg.info[i] == "ARQ;RECEIVING"){
var toastARQreceiving = document.getElementById('toastARQreceiving')
var toast = bootstrap.Toast.getOrCreateInstance(toastARQreceiving) // Returns a Bootstrap toast instance
toast.show()
}
// ARQ RECEIVING SUCCESS TOAST
if (arg.info[i] == "ARQ;RECEIVING;SUCCESS"){
var toastARQreceivingsuccess = document.getElementById('toastARQreceivingsuccess')
var toast = bootstrap.Toast.getOrCreateInstance(toastARQreceivingsuccess) // Returns a Bootstrap toast instance
toast.show()
}
// ARQ RECEIVING FAILED TOAST
if (arg.info[i] == "ARQ;RECEIVING;FAILED"){
var toastARQreceivingfailed = document.getElementById('toastARQreceivingfailed')
var toast = bootstrap.Toast.getOrCreateInstance(toastARQreceivingfailed) // Returns a Bootstrap toast instance
toast.show()
}
// ARQ TRANSMITTING TOAST
if (arg.info[i] == "ARQ;TRANSMITTING"){
var toastARQtransmitting = document.getElementById('toastARQtransmitting')
var toast = bootstrap.Toast.getOrCreateInstance(toastARQtransmitting) // Returns a Bootstrap toast instance
toast.show()
}
// ARQ TRANSMITTING SUCCESS TOAST
if (arg.info[i] == "ARQ;TRANSMITTING;SUCCESS"){
var toastARQtransmittingsuccess = document.getElementById('toastARQtransmittingsuccess')
var toast = bootstrap.Toast.getOrCreateInstance(toastARQtransmittingsuccess) // Returns a Bootstrap toast instance
toast.show()
}
// ARQ TRANSMITTING FAILED TOAST
if (arg.info[i] == "ARQ;TRANSMITTING;FAILED"){
var toastARQtransmittingfailed = document.getElementById('toastARQtransmittingfailed')
var toast = bootstrap.Toast.getOrCreateInstance(toastARQtransmittingfailed) // Returns a Bootstrap toast instance
toast.show()
}
}
});
ipcRenderer.on('action-update-daemon-state', (event, arg) => {
/*
// deactivetd RAM und CPU view so we dont get errors. We need to find a new place for this feature
// RAM
document.getElementById("progressbar_ram").setAttribute("aria-valuenow", arg.ram_usage)
document.getElementById("progressbar_ram").setAttribute("style", "width:" + arg.ram_usage + "%;")
document.getElementById("progressbar_ram_value").innerHTML = arg.ram_usage + "%"
// CPU
document.getElementById("progressbar_cpu").setAttribute("aria-valuenow", arg.cpu_usage)
document.getElementById("progressbar_cpu").setAttribute("style", "width:" + arg.cpu_usage + "%;")
document.getElementById("progressbar_cpu_value").innerHTML = arg.cpu_usage + "%"
*/
document.getElementById("ram_load").innerHTML = "RAM " + arg.ram_usage + "%"
document.getElementById("cpu_load").innerHTML = "CPU " + arg.cpu_usage + "%"
// OPERATING SYSTEM
document.getElementById("operating_system").innerHTML = "OS " + os.type()
// PYTHON VERSION
document.getElementById("python_version").innerHTML = "Python " + arg.python_version
document.getElementById("python_version").className = "btn btn-sm btn-success";
// HAMLIB VERSION
document.getElementById("hamlib_version").innerHTML = "Hamlib " + arg.hamlib_version
document.getElementById("hamlib_version").className = "btn btn-sm btn-success";
// NODE VERSION
document.getElementById("node_version").innerHTML = "Node " + process.version
document.getElementById("node_version").className = "btn btn-sm btn-success";
// UPDATE AUDIO INPUT
if (arg.tnc_running_state == "stopped") {
if (document.getElementById("audio_input_selectbox").length != arg.input_devices.length) {
document.getElementById("audio_input_selectbox").innerHTML = ""
for (i = 0; i < arg.input_devices.length; i++) {
var option = document.createElement("option");
option.text = arg.input_devices[i]['NAME'];
option.value = arg.input_devices[i]['ID'];
document.getElementById("audio_input_selectbox").add(option);
}
}
}
// UPDATE AUDIO OUTPUT
if (arg.tnc_running_state == "stopped") {
if (document.getElementById("audio_output_selectbox").length != arg.output_devices.length) {
document.getElementById("audio_output_selectbox").innerHTML = ""
for (i = 0; i < arg.output_devices.length; i++) {
var option = document.createElement("option");
option.text = arg.output_devices[i]['NAME'];
option.value = arg.output_devices[i]['ID'];
document.getElementById("audio_output_selectbox").add(option);
}
}
}
// UPDATE SERIAL DEVICES
if (arg.tnc_running_state == "stopped") {
if (document.getElementById("hamlib_deviceport").length != arg.serial_devices.length) {
document.getElementById("hamlib_deviceport").innerHTML = ""
for (i = 0; i < arg.serial_devices.length; i++) {
var option = document.createElement("option");
option.text = arg.serial_devices[i]['DESCRIPTION'];
option.value = arg.serial_devices[i]['PORT'];
document.getElementById("hamlib_deviceport").add(option);
}
}
if (document.getElementById("hamlib_deviceport_advanced").length != arg.serial_devices.length) {
document.getElementById("hamlib_deviceport_advanced").innerHTML = ""
for (i = 0; i < arg.serial_devices.length; i++) {
var option = document.createElement("option");
option.text = arg.serial_devices[i]['DESCRIPTION'];
option.value = arg.serial_devices[i]['PORT'];
document.getElementById("hamlib_deviceport_advanced").add(option);
}
}
}
if (arg.tnc_running_state == "stopped") {
if (document.getElementById("hamlib_ptt_port_advanced").length != arg.serial_devices.length) {
document.getElementById("hamlib_ptt_port_advanced").innerHTML = ""
for (i = 0; i < arg.serial_devices.length; i++) {
var option = document.createElement("option");
option.text = arg.serial_devices[i]['DESCRIPTION'];
option.value = arg.serial_devices[i]['PORT'];
document.getElementById("hamlib_ptt_port_advanced").add(option);
}
}
}
// TNC RUNNING STATE
//document.getElementById("tnc_running_state").innerHTML = arg.tnc_running_state;
if (arg.tnc_running_state == "running") {
document.getElementById('hamlib_deviceid').disabled = true
document.getElementById('hamlib_deviceport').disabled = true
document.getElementById('advancedHamlibSettingsButton').disabled = true
document.getElementById('testHamlib').disabled = true
document.getElementById('hamlib_ptt_protocol').disabled = true
document.getElementById('audio_input_selectbox').disabled = true
document.getElementById('audio_output_selectbox').disabled = true
document.getElementById('stopTNC').disabled = false
document.getElementById('startTNC').disabled = true
//document.getElementById('myCall').disabled = false
document.getElementById('dxCall').disabled = false
//document.getElementById('saveMyCall').disabled = false
//document.getElementById('myGrid').disabled = false
//document.getElementById('saveMyGrid').disabled = false
document.getElementById("hamlib_serialspeed").disabled = true
//document.getElementById("startTransmission").disabled = false
document.getElementById("openDataModule").disabled = false
// collapse settings screen
var collapseFirstRow = new bootstrap.Collapse(document.getElementById('collapseFirstRow'), {toggle: false})
collapseFirstRow.hide()
var collapseSecondRow = new bootstrap.Collapse(document.getElementById('collapseSecondRow'), {toggle: false})
collapseSecondRow.hide()
var collapseThirdRow = new bootstrap.Collapse(document.getElementById('collapseThirdRow'), {toggle: false})
collapseThirdRow.show()
var collapseFourthRow = new bootstrap.Collapse(document.getElementById('collapseFourthRow'), {toggle: false})
collapseFourthRow.show()
} else {
document.getElementById('hamlib_deviceid').disabled = false
document.getElementById('hamlib_deviceport').disabled = false
document.getElementById('advancedHamlibSettingsButton').disabled = false
document.getElementById('testHamlib').disabled = false
document.getElementById('hamlib_ptt_protocol').disabled = false
document.getElementById('audio_input_selectbox').disabled = false
document.getElementById('audio_output_selectbox').disabled = false
document.getElementById('stopTNC').disabled = true
document.getElementById('startTNC').disabled = false
//document.getElementById('myCall').disabled = true
document.getElementById('dxCall').disabled = true
//document.getElementById('saveMyCall').disabled = true
//document.getElementById('myGrid').disabled = true
//document.getElementById('saveMyGrid').disabled = true
document.getElementById("hamlib_serialspeed").disabled = false
//document.getElementById("startTransmission").disabled = true
document.getElementById("openDataModule").disabled = true
// collapse settings screen
var collapseFirstRow = new bootstrap.Collapse(document.getElementById('collapseFirstRow'), {toggle: false})
collapseFirstRow.show()
var collapseSecondRow = new bootstrap.Collapse(document.getElementById('collapseSecondRow'), {toggle: false})
collapseSecondRow.show()
var collapseThirdRow = new bootstrap.Collapse(document.getElementById('collapseThirdRow'), {toggle: false})
collapseThirdRow.hide()
var collapseFourthRow = new bootstrap.Collapse(document.getElementById('collapseFourthRow'), {toggle: false})
collapseFourthRow.hide()
}
});
// ACTION UPDATE HAMLIB TEST
ipcRenderer.on('action-update-hamlib-test', (event, arg) => {
console.log(arg.hamlib_result)
if (arg.hamlib_result == 'SUCCESS') {
document.getElementById("testHamlib").className = "btn btn-sm btn-success";
document.getElementById("testHamlibAdvanced").className = "btn btn-sm btn-success";
}
if (arg.hamlib_result == 'NOSUCCESS') {
document.getElementById("testHamlib").className = "btn btn-sm btn-warning";
document.getElementById("testHamlibAdvanced").className = "btn btn-sm btn-warning";
}
if (arg.hamlib_result == 'FAILED') {
document.getElementById("testHamlib").className = "btn btn-sm btn-danger";
document.getElementById("testHamlibAdvanced").className = "btn btn-sm btn-danger";
}
});
ipcRenderer.on('action-update-daemon-connection', (event, arg) => {
if (arg.daemon_connection == 'open') {
document.getElementById("daemon_connection_state").className = "btn btn-success";
document.getElementById("blurdiv").style.webkitFilter = "blur(0px)";
}
if (arg.daemon_connection == 'opening') {
document.getElementById("daemon_connection_state").className = "btn btn-warning";
document.getElementById("blurdiv").style.webkitFilter = "blur(10px)";
}
if (arg.daemon_connection == 'closed') {
document.getElementById("daemon_connection_state").className = "btn btn-danger";
document.getElementById("blurdiv").style.webkitFilter = "blur(10px)";
}
});
ipcRenderer.on('action-update-rx-buffer', (event, arg) => {
var data = arg.data["DATA"]
var tbl = document.getElementById("rx-data");
document.getElementById("rx-data").innerHTML = ''
for (i = 0; i < arg.data.length; i++) {
// first we update the PING window
if (arg.data[i]['DXCALLSIGN'] == document.getElementById("dxCall").value) {
/*
// if we are sending data without doing a ping before, we don't have a grid locator available. This could be a future feature for the TNC!
if(arg.data[i]['DXGRID'] != ''){
document.getElementById("pingDistance").innerHTML = arg.stations[i]['DXGRID']
}
*/
document.getElementById("pingDB").innerHTML = arg.stations[i]['SNR']
}
// now we update the received files list
var row = document.createElement("tr");
//https://stackoverflow.com/q/51421470
//https://stackoverflow.com/a/847196
timestampRaw = arg.data[i]['TIMESTAMP']
var date = new Date(timestampRaw * 1000);
var hours = date.getHours();
var minutes = "0" + date.getMinutes();
var seconds = "0" + date.getSeconds();
var datetime = hours + ':' + minutes.substr(-2) + ':' + seconds.substr(-2);
var timestamp = document.createElement("td");
var timestampText = document.createElement('span');
timestampText.innerText = datetime
timestamp.appendChild(timestampText);
var dxCall = document.createElement("td");
var dxCallText = document.createElement('span');
dxCallText.innerText = arg.data[i]['DXCALLSIGN']
dxCall.appendChild(dxCallText);
/*
var dxGrid = document.createElement("td");
var dxGridText = document.createElement('span');
dxGridText.innerText = arg.data[i]['DXGRID']
dxGrid.appendChild(dxGridText);
*/
var fileName = document.createElement("td");
var fileNameText = document.createElement('span');
var fileNameString = arg.data[i]['RXDATA'][0]['filename']
fileNameText.innerText = fileNameString
fileName.appendChild(fileNameText);
row.appendChild(timestamp);
row.appendChild(dxCall);
// row.appendChild(dxGrid);
row.appendChild(fileName);
tbl.appendChild(row);
// https://stackoverflow.com/a/26227660
var appDataFolder = process.env.HOME
console.log("appDataFolder:" + appDataFolder)
var applicationFolder = path.join(appDataFolder, "FreeDATA");
console.log(applicationFolder)
var receivedFilesFolder = path.join(applicationFolder, "receivedFiles");
console.log("receivedFilesFolder: " + receivedFilesFolder)
// Creates receivedFiles folder if not exists
// https://stackoverflow.com/a/13544465
fs.mkdir(receivedFilesFolder, {
recursive: true
}, function(err) {
console.log(err);
});
// write file to rxdata folder
var base64String = arg.data[i]['RXDATA'][0]['data']
// remove header from base64 String
// https://www.codeblocq.com/2016/04/Convert-a-base64-string-to-a-file-in-Node/
var base64Data = base64String.split(';base64,').pop()
//write data to file
var receivedFile = path.join(receivedFilesFolder, fileNameString);
console.log(receivedFile)
require("fs").writeFile(receivedFile, base64Data, 'base64', function(err) {
console.log(err);
});
}
});
ipcRenderer.on('run-tnc-command', (event, arg) => {
if (arg.command == 'saveMyCall') {
sock.saveMyCall(arg.callsign)
}
if (arg.command == 'saveMyGrid') {
sock.saveMyGrid(arg.grid)
}
if (arg.command == 'ping') {
sock.sendPing(arg.dxcallsign)
}
if (arg.command == 'sendFile') {
sock.sendFile(arg.dxcallsign, arg.mode, arg.frames, arg.filename, arg.filetype, arg.data, arg.checksum)
}
if (arg.command == 'sendMessage') {
sock.sendMessage(arg.dxcallsign, arg.mode, arg.frames, arg.data, arg.checksum)
}
if (arg.command == 'stopTransmission') {
sock.stopTransmission()
}
if (arg.command == 'delRxMsgBuffer') {
sock.delRxMsgBuffer()
}
});