mirror of
https://github.com/DJ2LS/FreeDATA
synced 2024-05-14 08:04:33 +00:00
build action
This commit is contained in:
parent
bbb3ab0360
commit
91deefb905
8 changed files with 7121 additions and 0 deletions
13
config.json
Normal file
13
config.json
Normal file
|
@ -0,0 +1,13 @@
|
|||
{
|
||||
"tnc_host": "192.168.178.163",
|
||||
"tnc_port": "3000",
|
||||
"daemon_host": "192.168.178.163",
|
||||
"daemon_port": "30001",
|
||||
"mycall": "DJ2LS",
|
||||
"mygrid": "JN48ea",
|
||||
"deviceid": "2028",
|
||||
"deviceport": "/dev/ttyUSB0",
|
||||
"serialspeed": "9600",
|
||||
"ptt": "RTS",
|
||||
"spectrum": "scatter"
|
||||
}
|
161
daemon.js
Normal file
161
daemon.js
Normal file
|
@ -0,0 +1,161 @@
|
|||
var net = require('net');
|
||||
var config = require('./config.json');
|
||||
|
||||
|
||||
var daemon = new net.Socket();
|
||||
var msg = ''; // Current message, per connection.
|
||||
|
||||
const {
|
||||
ipcRenderer
|
||||
} = require('electron');
|
||||
|
||||
|
||||
setTimeout(connectDAEMON, 500)
|
||||
|
||||
function connectDAEMON() {
|
||||
|
||||
console.log('connecting to DAEMON...')
|
||||
|
||||
//clear message buffer after reconnecting or inital connection
|
||||
msg = '';
|
||||
daemon.connect(config.daemon_port, config.daemon_host)
|
||||
//client.setTimeout(5000);
|
||||
}
|
||||
|
||||
daemon.on('connect', function(data) {
|
||||
console.log('DAEMON connection established')
|
||||
})
|
||||
|
||||
daemon.on('error', function(data) {
|
||||
console.log('DAEMON connection error');
|
||||
setTimeout(connectDAEMON, 2000)
|
||||
});
|
||||
|
||||
/*
|
||||
client.on('close', function(data) {
|
||||
console.log(' TNC connection closed');
|
||||
setTimeout(connectTNC, 2000)
|
||||
});
|
||||
*/
|
||||
|
||||
daemon.on('end', function(data) {
|
||||
console.log('DAEMON connection ended');
|
||||
setTimeout(connectDAEMON, 2000)
|
||||
});
|
||||
|
||||
|
||||
//exports.writeCommand = function(command){
|
||||
writeDaemonCommand = function(command) {
|
||||
|
||||
|
||||
// we use the writingCommand function to update our TCPIP state because we are calling this function a lot
|
||||
// if socket openend, we are able to run commands
|
||||
if (daemon.readyState == 'open') {
|
||||
//uiMain.setDAEMONconnection('open')
|
||||
daemon.write(command + '\n');
|
||||
}
|
||||
|
||||
if (daemon.readyState == 'closed') {
|
||||
//uiMain.setDAEMONconnection('closed')
|
||||
}
|
||||
|
||||
if (daemon.readyState == 'opening') {
|
||||
//uiMain.setDAEMONconnection('opening')
|
||||
}
|
||||
|
||||
|
||||
let Data = {
|
||||
daemon_connection: daemon.readyState,
|
||||
};
|
||||
ipcRenderer.send('request-update-daemon-connection', Data);
|
||||
}
|
||||
|
||||
// "https://stackoverflow.com/questions/9070700/nodejs-net-createserver-large-amount-of-data-coming-in"
|
||||
|
||||
|
||||
daemon.on('data', function(data) {
|
||||
|
||||
data = data.toString('utf8'); /* convert data to string */
|
||||
msg += data.toString('utf8'); /*append data to buffer so we can stick long data together */
|
||||
|
||||
|
||||
|
||||
/* check if we reached an EOF, if true, clear buffer and parse JSON data */
|
||||
if (data.endsWith('}')) {
|
||||
/*console.log(msg)*/
|
||||
try {
|
||||
/*console.log(msg)*/
|
||||
data = JSON.parse(msg)
|
||||
} catch (e) {
|
||||
console.log(e); /* "SyntaxError */
|
||||
}
|
||||
msg = '';
|
||||
/*console.log("EOF detected!")*/
|
||||
|
||||
if (data['COMMAND'] == 'DAEMON_STATE') {
|
||||
let Data = {
|
||||
input_devices: data['INPUT_DEVICES'],
|
||||
output_devices: data['OUTPUT_DEVICES'],
|
||||
tnc_running_state: data['DAEMON_STATE'][0]['STATUS'],
|
||||
ram_usage: data['RAM'],
|
||||
cpu_usage: data['CPU'],
|
||||
version: data['VERSION'],
|
||||
};
|
||||
ipcRenderer.send('request-update-daemon-state', Data);
|
||||
|
||||
}
|
||||
|
||||
////// check if EOF ...
|
||||
}
|
||||
|
||||
|
||||
|
||||
});
|
||||
|
||||
function hexToBytes(hex) {
|
||||
for (var bytes = [], c = 0; c < hex.length; c += 2)
|
||||
bytes.push(parseInt(hex.substr(c, 2), 16));
|
||||
return bytes;
|
||||
}
|
||||
|
||||
exports.getDaemonState = function() {
|
||||
//function getDaemonState(){
|
||||
command = '{"type" : "GET", "command": "DAEMON_STATE"}'
|
||||
writeDaemonCommand(command)
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
// START TNC
|
||||
// ` `== multi line string
|
||||
|
||||
|
||||
exports.startTNC = function(rx_audio, tx_audio, deviceid, deviceport, ptt, serialspeed) {
|
||||
var json_command = JSON.stringify({
|
||||
type: 'SET',
|
||||
command: 'STARTTNC',
|
||||
parameter: [{
|
||||
rx_audio: rx_audio,
|
||||
tx_audio: tx_audio,
|
||||
deviceid: deviceid,
|
||||
deviceport: deviceport,
|
||||
ptt: ptt,
|
||||
serialspeed: serialspeed
|
||||
|
||||
}]
|
||||
})
|
||||
|
||||
|
||||
|
||||
//console.log(json_command)
|
||||
writeDaemonCommand(json_command)
|
||||
|
||||
|
||||
}
|
||||
|
||||
// STOP TNC
|
||||
exports.stopTNC = function() {
|
||||
command = '{"type" : "SET", "command": "STOPTNC" , "parameter": "---" }'
|
||||
writeDaemonCommand(command)
|
||||
}
|
124
main.js
Normal file
124
main.js
Normal file
|
@ -0,0 +1,124 @@
|
|||
const {app,BrowserWindow,ipcMain} = require('electron')
|
||||
const path = require('path')
|
||||
var config = require('./config.json');
|
||||
|
||||
let win = null;
|
||||
let data = null;
|
||||
|
||||
function createWindow() {
|
||||
win = new BrowserWindow({
|
||||
width: 1220,
|
||||
height: 920,
|
||||
webPreferences: {
|
||||
//preload: path.join(__dirname, 'preload-main.js'),
|
||||
preload: require.resolve('./preload-main.js'),
|
||||
nodeIntegration: true,
|
||||
contextIsolation: false,
|
||||
enableRemoteModule: false, //https://stackoverflow.com/questions/53390798/opening-new-window-electron/53393655 https://github.com/electron/remote
|
||||
}
|
||||
})
|
||||
//open dev tools
|
||||
win.webContents.openDevTools({
|
||||
mode: 'undocked',
|
||||
activate: true,
|
||||
})
|
||||
win.loadFile('src/index.html')
|
||||
/*
|
||||
data = new BrowserWindow({
|
||||
height: 900,
|
||||
width: 600,
|
||||
parent: win,
|
||||
webPreferences: {
|
||||
preload: require.resolve('./preload-data.js'),
|
||||
nodeIntegration: true,
|
||||
|
||||
}
|
||||
})
|
||||
//open dev tools
|
||||
data.webContents.openDevTools({
|
||||
mode: 'undocked',
|
||||
activate: true,
|
||||
})
|
||||
data.loadFile('src/data-module.html')
|
||||
data.hide()
|
||||
*/
|
||||
|
||||
// Emitted when the window is closed.
|
||||
win.on('closed', function() {
|
||||
// Dereference the window object, usually you would store windows
|
||||
// in an array if your app supports multi windows, this is the time
|
||||
// when you should delete the corresponding element.
|
||||
win = null;
|
||||
data = null;
|
||||
})
|
||||
|
||||
/*
|
||||
data.on('closed', function () {
|
||||
// Dereference the window object, usually you would store windows
|
||||
// in an array if your app supports multi windows, this is the time
|
||||
// when you should delete the corresponding element.
|
||||
})
|
||||
*/
|
||||
|
||||
|
||||
// https://stackoverflow.com/questions/44258831/only-hide-the-window-when-closing-it-electron
|
||||
/*
|
||||
data.on('close', function(evt) {
|
||||
evt.preventDefault();
|
||||
data.hide()
|
||||
});
|
||||
*/
|
||||
}
|
||||
|
||||
|
||||
app.whenReady().then(() => {
|
||||
createWindow()
|
||||
|
||||
app.on('activate', () => {
|
||||
if (BrowserWindow.getAllWindows().length === 0) {
|
||||
createWindow()
|
||||
}
|
||||
})
|
||||
})
|
||||
|
||||
app.on('window-all-closed', () => {
|
||||
|
||||
if (process.platform !== 'darwin') {
|
||||
app.quit()
|
||||
}
|
||||
})
|
||||
|
||||
|
||||
// IPC HANDLER
|
||||
/*
|
||||
ipcMain.on('show-data-window', (event, arg) => {
|
||||
data.show()
|
||||
});
|
||||
*/
|
||||
ipcMain.on('request-update-tnc-state', (event, arg) => {
|
||||
win.webContents.send('action-update-tnc-state', arg);
|
||||
//data.webContents.send('action-update-tnc-state', arg);
|
||||
});
|
||||
|
||||
/*
|
||||
ipcMain.on('request-update-data-state', (event, arg) => {
|
||||
//win.webContents.send('action-update-data-state', arg);
|
||||
//data.webContents.send('action-update-data-state', arg);
|
||||
});
|
||||
|
||||
ipcMain.on('request-update-heard-stations', (event, arg) => {
|
||||
win.webContents.send('action-update-heard-stations', arg);
|
||||
});
|
||||
*/
|
||||
ipcMain.on('request-update-daemon-state', (event, arg) => {
|
||||
win.webContents.send('action-update-daemon-state', arg);
|
||||
});
|
||||
|
||||
ipcMain.on('request-update-daemon-connection', (event, arg) => {
|
||||
win.webContents.send('action-update-daemon-connection', arg);
|
||||
});
|
||||
|
||||
ipcMain.on('run-tnc-command', (event, arg) => {
|
||||
win.webContents.send('run-tnc-command', arg);
|
||||
});
|
||||
|
5755
package-lock.json
generated
Normal file
5755
package-lock.json
generated
Normal file
File diff suppressed because it is too large
Load diff
54
package.json
Normal file
54
package.json
Normal file
|
@ -0,0 +1,54 @@
|
|||
{
|
||||
"name": "codec2-FreeDATA",
|
||||
"version": "0.0.1",
|
||||
"description": "codec2-FreeDATA ",
|
||||
"main": "main.js",
|
||||
"build": {
|
||||
"appId": "com.DJ2LS.codec2-FreeDATA",
|
||||
"productName": "codec2-FreeDATA",
|
||||
"mac": {
|
||||
"category": "public.app-category.utilities"
|
||||
},
|
||||
"dmg": {
|
||||
"icon": false
|
||||
},
|
||||
"linux": {
|
||||
"target": [
|
||||
"AppImage"
|
||||
],
|
||||
"category": "Science"
|
||||
}
|
||||
},
|
||||
"scripts": {
|
||||
"start": "electron .",
|
||||
"test": "echo \"Error: no test specified\" && exit 1",
|
||||
"postinstall": "electron-builder install-app-deps",
|
||||
"build": "electron-builder --mac --linux",
|
||||
"release": "electron-builder --mac --linux --publish always"
|
||||
},
|
||||
"repository": {
|
||||
"type": "git",
|
||||
"url": "git+https://github.com/DJ2LS/codec2-FreeDATA.git"
|
||||
},
|
||||
"keywords": [
|
||||
"TNC",
|
||||
"GUI",
|
||||
"FreeDV",
|
||||
"codec2"
|
||||
],
|
||||
"author": "DJ2LS",
|
||||
"license": "LGPL-2.1",
|
||||
"bugs": {
|
||||
"url": "https://github.com/DJ2LS/codec2-FreeDATA/issues"
|
||||
},
|
||||
"homepage": "https://github.com/DJ2LS/codec2-FreeDATA#readme",
|
||||
"dependencies": {
|
||||
"bootstrap": "^5.1.0",
|
||||
"chart.js": "^3.5.0",
|
||||
"qth-locator": "^2.1.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
"electron": "13.1.8",
|
||||
"electron-builder": "^22.11.7"
|
||||
}
|
||||
}
|
104
preload-data.js
Normal file
104
preload-data.js
Normal file
|
@ -0,0 +1,104 @@
|
|||
const {
|
||||
ipcRenderer
|
||||
} = require('electron');
|
||||
var config = require('./config.json');
|
||||
|
||||
window.addEventListener('DOMContentLoaded', () => {
|
||||
const replaceText = (selector, text) => {
|
||||
const element = document.getElementById(selector)
|
||||
if (element) element.innerText = text
|
||||
}
|
||||
|
||||
// sendPing button clicked
|
||||
document.getElementById("sendPing").addEventListener("click", () => {
|
||||
dxcallsign = document.getElementById("dxCall").value
|
||||
//sock.sendPing(dxcallsign)
|
||||
|
||||
let Data = {
|
||||
command: "ping",
|
||||
dxcallsign: document.getElementById("dxCall").value
|
||||
};
|
||||
ipcRenderer.send('run-tnc-command', Data);
|
||||
})
|
||||
|
||||
|
||||
|
||||
|
||||
document.getElementById("startTransmission").addEventListener("click", () => {
|
||||
let Data = {
|
||||
command: "sendFile",
|
||||
dxcallsign: document.getElementById("dxCall").value,
|
||||
mode: document.getElementById("datamode").value,
|
||||
frames: document.getElementById("framesperburst").value,
|
||||
filetype: 'txt',
|
||||
filename: 'testfile',
|
||||
data: '1234567',
|
||||
checksum: '123123123',
|
||||
};
|
||||
ipcRenderer.send('run-tnc-command', Data);
|
||||
|
||||
|
||||
})
|
||||
|
||||
})
|
||||
|
||||
|
||||
|
||||
|
||||
ipcRenderer.on('action-update-tnc-state', (event, arg) => {
|
||||
|
||||
// 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";
|
||||
} else if (arg.busy_state == 'IDLE') {
|
||||
document.getElementById("busy_state").className = "btn btn-success";
|
||||
} else {
|
||||
document.getElementById("busy_state").className = "btn btn-secondary"
|
||||
}
|
||||
|
||||
// ARQ STATE
|
||||
if (arg.arq_state == 'DATA') {
|
||||
document.getElementById("arq_state").className = "btn btn-warning";
|
||||
} else if (arg.arq_state == 'IDLE') {
|
||||
document.getElementById("arq_state").className = "btn btn-secondary";
|
||||
} else {
|
||||
document.getElementById("arq_state").className = "btn btn-secondary"
|
||||
}
|
||||
|
||||
// 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"
|
||||
|
||||
}
|
||||
|
||||
});
|
||||
|
||||
|
||||
ipcRenderer.on('action-update-data-state', (event, arg) => {
|
||||
|
||||
});
|
666
preload-main.js
Normal file
666
preload-main.js
Normal file
|
@ -0,0 +1,666 @@
|
|||
const sock = require('./sock.js')
|
||||
const daemon = require('./daemon.js')
|
||||
const configPath = './config.json'
|
||||
const config = require(configPath);
|
||||
const {
|
||||
ipcRenderer
|
||||
} = require('electron');
|
||||
const fs = require('fs');
|
||||
|
||||
const { locatorToLatLng, distance, bearingDistance, latLngToLocator } = require('qth-locator');
|
||||
|
||||
|
||||
// START INTERVALL COMMAND EXECUTION FOR STATES
|
||||
setInterval(daemon.getDaemonState, 1000)
|
||||
setInterval(sock.getTncState, 250)
|
||||
//setInterval(sock.getDataState, 500)
|
||||
//setInterval(sock.getHeardStations, 1000)
|
||||
|
||||
// UPDATE FFT DEMO
|
||||
|
||||
updateFFT = function(fft) {
|
||||
var fft = Array.from({
|
||||
length: 2048
|
||||
}, () => Math.floor(Math.random() * 10));
|
||||
spectrum.addData(fft);
|
||||
}
|
||||
setInterval(updateFFT, 250)
|
||||
|
||||
|
||||
|
||||
// WINDOW LISTENER
|
||||
window.addEventListener('DOMContentLoaded', () => {
|
||||
// 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').value = config.ptt
|
||||
|
||||
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 = '350px';
|
||||
} 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';
|
||||
}
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
// Create spectrum object on canvas with ID "waterfall"
|
||||
global.spectrum = new Spectrum(
|
||||
"waterfall", {
|
||||
spectrumPercent: 20
|
||||
});
|
||||
|
||||
|
||||
// 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
|
||||
},
|
||||
y: {
|
||||
display: false
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
// 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 = '350px';
|
||||
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 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));
|
||||
|
||||
});
|
||||
|
||||
document.getElementById("tnc_port").addEventListener("change", () => {
|
||||
config.tnc_port = document.getElementById("tnc_port").value
|
||||
config.daemon_port = 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
|
||||
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", () => {
|
||||
dxcallsign = document.getElementById("dxCall").value
|
||||
sock.sendPing(dxcallsign)
|
||||
});
|
||||
|
||||
// sendCQ button clicked
|
||||
document.getElementById("sendCQ").addEventListener("click", () => {
|
||||
sock.sendCQ()
|
||||
});
|
||||
|
||||
// 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 ptt = document.getElementById("hamlib_ptt").value
|
||||
|
||||
|
||||
config.deviceid = deviceid
|
||||
config.deviceport = deviceport
|
||||
config.serialspeed = serialspeed
|
||||
config.ptt = ptt
|
||||
fs.writeFileSync(configPath, JSON.stringify(config, null, 2));
|
||||
|
||||
|
||||
daemon.startTNC(rx_audio, tx_audio, deviceid, deviceport, ptt, serialspeed)
|
||||
setTimeout(function() {
|
||||
sock.saveMyCall(config.mycall);
|
||||
}, 5000);
|
||||
setTimeout(function() {
|
||||
sock.saveMyGrid(config.mygrid);
|
||||
}, 6000);
|
||||
})
|
||||
|
||||
// stopTNC button clicked
|
||||
document.getElementById("stopTNC").addEventListener("click", () => {
|
||||
daemon.stopTNC()
|
||||
})
|
||||
|
||||
// openDataModule button clicked
|
||||
document.getElementById("openDataModule").addEventListener("click", () => {
|
||||
if(document.getElementById("mySidebar").style.width == "40%"){
|
||||
document.getElementById("mySidebar").style.width = "0px";
|
||||
} else {
|
||||
document.getElementById("mySidebar").style.width = "40%";
|
||||
}
|
||||
})
|
||||
|
||||
|
||||
|
||||
})
|
||||
|
||||
|
||||
|
||||
|
||||
ipcRenderer.on('action-update-tnc-state', (event, arg) => {
|
||||
|
||||
|
||||
// TOE TIME OF EXECUTION --> How many time needs a command to be executed until data arrives
|
||||
if (typeof(arg.toe) == 'undefined'){
|
||||
var toe = 0
|
||||
} else {
|
||||
var toe = arg.toe
|
||||
}
|
||||
document.getElementById("toe").innerHTML = toe + ' ms'
|
||||
|
||||
|
||||
|
||||
// SCATTER DIAGRAM PLOTTING
|
||||
//global.myChart.destroy();
|
||||
|
||||
//console.log(arg.scatter.length)
|
||||
|
||||
var data = arg.scatter
|
||||
var newdata = {
|
||||
datasets: [{
|
||||
label: 'constellation diagram',
|
||||
data: data,
|
||||
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: {
|
||||
animation: false,
|
||||
legend: {
|
||||
display: false,
|
||||
tooltips: {
|
||||
enabled: false,
|
||||
},
|
||||
},
|
||||
scales: {
|
||||
display: false,
|
||||
grid: {
|
||||
display: false
|
||||
},
|
||||
x: {
|
||||
type: 'linear',
|
||||
position: 'bottom',
|
||||
display: false
|
||||
},
|
||||
y: {
|
||||
display: false
|
||||
}
|
||||
}
|
||||
},
|
||||
});
|
||||
}
|
||||
|
||||
// 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";
|
||||
} else if (arg.busy_state == 'IDLE') {
|
||||
document.getElementById("busy_state").className = "btn btn-success";
|
||||
} else {
|
||||
document.getElementById("busy_state").className = "btn btn-secondary"
|
||||
}
|
||||
|
||||
// ARQ STATE
|
||||
if (arg.arq_state == 'DATA') {
|
||||
document.getElementById("arq_state").className = "btn btn-warning";
|
||||
} else if (arg.arq_state == 'IDLE') {
|
||||
document.getElementById("arq_state").className = "btn btn-secondary";
|
||||
} else {
|
||||
document.getElementById("arq_state").className = "btn btn-secondary"
|
||||
}
|
||||
|
||||
// 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
|
||||
|
||||
|
||||
|
||||
// UPDATE HEARD STATIONS
|
||||
//console.log(arg.stations)
|
||||
//console.log(arg.stations[0]['DXGRID'])
|
||||
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
|
||||
console.log(document.getElementById("dxCall").value)
|
||||
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
|
||||
|
||||
} catch {
|
||||
document.getElementById("pingDistance").innerHTML = '---'
|
||||
}
|
||||
document.getElementById("pingDB").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 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');
|
||||
dataTypeText.innerText = arg.stations[i]['DATATYPE']
|
||||
dataType.appendChild(dataTypeText);
|
||||
|
||||
|
||||
|
||||
|
||||
|
||||
if(dataTypeText.innerText == 'CQ CQ CQ'){
|
||||
row.classList.add("table-success");
|
||||
}
|
||||
|
||||
if(dataTypeText.innerText == 'DATA-CHANNEL'){
|
||||
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);
|
||||
|
||||
row.appendChild(timestamp);
|
||||
row.appendChild(dxCall);
|
||||
row.appendChild(dxGrid);
|
||||
row.appendChild(gridDistance);
|
||||
row.appendChild(dataType);
|
||||
row.appendChild(snr);
|
||||
|
||||
tbl.appendChild(row);
|
||||
}
|
||||
|
||||
|
||||
});
|
||||
|
||||
|
||||
|
||||
|
||||
ipcRenderer.on('action-update-daemon-state', (event, arg) => {
|
||||
|
||||
|
||||
// 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 + "%"
|
||||
|
||||
|
||||
// UPDATE AUDIO INPUT
|
||||
|
||||
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 (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);
|
||||
}
|
||||
}
|
||||
|
||||
// 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('hamlib_ptt').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
|
||||
|
||||
} else {
|
||||
document.getElementById('hamlib_deviceid').disabled = false
|
||||
document.getElementById('hamlib_deviceport').disabled = false
|
||||
document.getElementById('hamlib_ptt').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
|
||||
|
||||
}
|
||||
|
||||
});
|
||||
|
||||
|
||||
ipcRenderer.on('action-update-daemon-connection', (event, arg) => {
|
||||
|
||||
if (arg.daemon_connection == 'open') {
|
||||
document.getElementById("daemon_connection_state").className = "btn btn-success";
|
||||
}
|
||||
if (arg.daemon_connection == 'opening') {
|
||||
document.getElementById("daemon_connection_state").className = "btn btn-warning";
|
||||
}
|
||||
if (arg.daemon_connection == 'closed') {
|
||||
document.getElementById("daemon_connection_state").className = "btn btn-danger";
|
||||
}
|
||||
|
||||
});
|
||||
|
||||
|
||||
|
||||
/*
|
||||
ipcRenderer.on('action-update-heard-stations', (event, arg) => {
|
||||
//console.log(arg.stations)
|
||||
//console.log(arg.stations[0]['DXGRID'])
|
||||
|
||||
var tbl = document.getElementById("heardstations");
|
||||
document.getElementById("heardstations").innerHTML = ''
|
||||
|
||||
for (i = 0; i < arg.stations.length; i++) {
|
||||
|
||||
|
||||
// first we update the PING window
|
||||
console.log(document.getElementById("dxCall").value)
|
||||
if (arg.stations[i]['DXCALLSIGN'] == document.getElementById("dxCall").value) {
|
||||
document.getElementById("pingDistance").innerHTML = arg.stations[i]['DXGRID']
|
||||
document.getElementById("pingDB").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 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 dataType = document.createElement("td");
|
||||
var dataTypeText = document.createElement('span');
|
||||
dataTypeText.innerText = arg.stations[i]['DATATYPE']
|
||||
dataType.appendChild(dataTypeText);
|
||||
|
||||
|
||||
|
||||
row.appendChild(timestamp);
|
||||
row.appendChild(dxCall);
|
||||
row.appendChild(dxGrid);
|
||||
row.appendChild(dataType);
|
||||
|
||||
tbl.appendChild(row);
|
||||
}
|
||||
|
||||
});
|
||||
*/
|
||||
|
||||
|
||||
|
||||
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)
|
||||
}
|
||||
});
|
244
sock.js
Normal file
244
sock.js
Normal file
|
@ -0,0 +1,244 @@
|
|||
var net = require('net');
|
||||
var config = require('./config.json');
|
||||
const {
|
||||
ipcRenderer
|
||||
} = require('electron');
|
||||
|
||||
var client = new net.Socket();
|
||||
var msg = ''; // Current message, per connection.
|
||||
|
||||
setTimeout(connectTNC, 3000)
|
||||
|
||||
function connectTNC() {
|
||||
//exports.connectTNC = function(){
|
||||
console.log('connecting to TNC...')
|
||||
|
||||
//clear message buffer after reconnecting or inital connection
|
||||
msg = '';
|
||||
client.connect(config.tnc_port, config.tnc_host)
|
||||
}
|
||||
|
||||
client.on('connect', function(data) {
|
||||
console.log('TNC connection established')
|
||||
})
|
||||
|
||||
client.on('error', function(data) {
|
||||
console.log('TNC connection error');
|
||||
|
||||
|
||||
let Data = {
|
||||
busy_state: "-",
|
||||
arq_state: "-",
|
||||
channel_state: "-",
|
||||
frequency: "-",
|
||||
mode: "-",
|
||||
bandwith: "-",
|
||||
rms_level: 0
|
||||
|
||||
};
|
||||
ipcRenderer.send('request-update-tnc-state', Data);
|
||||
|
||||
setTimeout(connectTNC, 2000)
|
||||
// setTimeout( function() { exports.connectTNC(tnc_host, tnc_port); }, 2000 );
|
||||
|
||||
});
|
||||
|
||||
/*
|
||||
client.on('close', function(data) {
|
||||
console.log(' TNC connection closed');
|
||||
setTimeout(connectTNC, 2000)
|
||||
});
|
||||
*/
|
||||
|
||||
client.on('end', function(data) {
|
||||
console.log('TNC connection ended');
|
||||
//setTimeout(connectTNC, 2000)
|
||||
setTimeout(connectTNC, 0)
|
||||
|
||||
// setTimeout( function() { exports.connectTNC(tnc_host, tnc_port); }, 2000 );
|
||||
|
||||
});
|
||||
|
||||
|
||||
//exports.writeTncCommand = function(command){
|
||||
writeTncCommand = function(command) {
|
||||
|
||||
console.log(command)
|
||||
// we use the writingCommand function to update our TCPIP state because we are calling this function a lot
|
||||
// if socket openend, we are able to run commands
|
||||
if (client.readyState == 'open') {
|
||||
//uiMain.setTNCconnection('open')
|
||||
client.write(command + '\n');
|
||||
}
|
||||
|
||||
if (client.readyState == 'closed') {
|
||||
//uiMain.setTNCconnection('closed')
|
||||
console.log("CLOSED!!!!!")
|
||||
}
|
||||
|
||||
if (client.readyState == 'opening') {
|
||||
//uiMain.setTNCconnection('opening')
|
||||
console.log("OPENING!!!!!")
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
client.on('data', function(data) {
|
||||
|
||||
/*
|
||||
stackoverflow.com questions 9070700 nodejs-net-createserver-large-amount-of-data-coming-in
|
||||
*/
|
||||
|
||||
data = data.toString('utf8'); // convert data to string
|
||||
msg += data.toString('utf8'); // append data to buffer so we can stick long data together
|
||||
//console.log(data)
|
||||
// check if we reached an EOF, if true, clear buffer and parse JSON data
|
||||
if (data.endsWith('"EOF": "EOF"}')) {
|
||||
//console.log(msg)
|
||||
try {
|
||||
//console.log(msg)
|
||||
data = JSON.parse(msg)
|
||||
} catch (e) {
|
||||
console.log(e); /* "SyntaxError*/
|
||||
}
|
||||
msg = '';
|
||||
/* console.log("EOF detected!") */
|
||||
|
||||
|
||||
if (data['COMMAND'] == 'TNC_STATE') {
|
||||
|
||||
let Data = {
|
||||
toe: Date.now() - data['TIMESTAMP'], // time of execution
|
||||
ptt_state: data['PTT_STATE'],
|
||||
busy_state: data['TNC_STATE'],
|
||||
arq_state: data['ARQ_STATE'],
|
||||
channel_state: data['CHANNEL_STATE'],
|
||||
frequency: data['FREQUENCY'],
|
||||
mode: data['MODE'],
|
||||
bandwith: data['BANDWITH'],
|
||||
rms_level: (data['AUDIO_RMS'] / 1000) * 100,
|
||||
scatter: data['SCATTER'],
|
||||
rx_buffer_length: data['RX_BUFFER_LENGTH'],
|
||||
tx_n_max_retries: data['TX_N_MAX_RETRIES'],
|
||||
arq_tx_n_frames_per_burst: data['ARQ_TX_N_FRAMES_PER_BURST'],
|
||||
arq_tx_n_bursts: data['ARQ_TX_N_BURSTS'],
|
||||
arq_tx_n_current_arq_frame: data['ARQ_TX_N_CURRENT_ARQ_FRAME'],
|
||||
arq_tx_n_total_arq_frames: data['ARQ_TX_N_TOTAL_ARQ_FRAMES'],
|
||||
arq_rx_frame_n_bursts: data['ARQ_RX_FRAME_N_BURSTS'],
|
||||
arq_rx_n_current_arq_frame: data['ARQ_RX_N_CURRENT_ARQ_FRAME'],
|
||||
arq_n_arq_frames_per_data_frame: data['ARQ_N_ARQ_FRAMES_PER_DATA_FRAME'],
|
||||
stations: data['STATIONS'],
|
||||
};
|
||||
console.log(Data)
|
||||
ipcRenderer.send('request-update-tnc-state', Data);
|
||||
}
|
||||
/*
|
||||
if (data['COMMAND'] == 'DATA_STATE') {
|
||||
let Data = {
|
||||
rx_buffer_length: data['RX_BUFFER_LENGTH'],
|
||||
tx_n_max_retries: data['TX_N_MAX_RETRIES'],
|
||||
arq_tx_n_frames_per_burst: data['ARQ_TX_N_FRAMES_PER_BURST'],
|
||||
arq_tx_n_bursts: data['ARQ_TX_N_BURSTS'],
|
||||
arq_tx_n_current_arq_frame: data['ARQ_TX_N_CURRENT_ARQ_FRAME'],
|
||||
arq_tx_n_total_arq_frames: data['ARQ_TX_N_TOTAL_ARQ_FRAMES'],
|
||||
arq_rx_frame_n_bursts: data['ARQ_RX_FRAME_N_BURSTS'],
|
||||
arq_rx_n_current_arq_frame: data['ARQ_RX_N_CURRENT_ARQ_FRAME'],
|
||||
arq_n_arq_frames_per_data_frame: data['ARQ_N_ARQ_FRAMES_PER_DATA_FRAME'],
|
||||
};
|
||||
console.log(Data)
|
||||
ipcRenderer.send('request-update-data-state', Data);
|
||||
}
|
||||
*/
|
||||
/*
|
||||
if (data['COMMAND'] == 'HEARD_STATIONS') {
|
||||
//console.log(data['STATIONS'])
|
||||
let Data = {
|
||||
stations: data['STATIONS'],
|
||||
};
|
||||
//console.log(Data)
|
||||
ipcRenderer.send('request-update-heard-stations', Data);
|
||||
}
|
||||
*/
|
||||
|
||||
/*
|
||||
if (data['COMMAND'] == 'SCATTER') {
|
||||
console.log(data['SCATTER'])
|
||||
let Data = {
|
||||
stations: data['STATIONS'],
|
||||
};
|
||||
//console.log(Data)
|
||||
//ipcRenderer.send('request-update-heard-stations', Data);
|
||||
|
||||
|
||||
}
|
||||
*/
|
||||
|
||||
// check if EOF ...
|
||||
}
|
||||
|
||||
|
||||
|
||||
});
|
||||
|
||||
function hexToBytes(hex) {
|
||||
for (var bytes = [], c = 0; c < hex.length; c += 2)
|
||||
bytes.push(parseInt(hex.substr(c, 2), 16));
|
||||
return bytes;
|
||||
}
|
||||
|
||||
|
||||
//Save myCall
|
||||
exports.saveMyCall = function(callsign) {
|
||||
command = '{"type" : "SET", "command": "MYCALLSIGN" , "parameter": "' + callsign + '", "timestamp" : '+Date.now()+'}'
|
||||
writeTncCommand(command)
|
||||
}
|
||||
|
||||
// Save myGrid
|
||||
exports.saveMyGrid = function(grid) {
|
||||
command = '{"type" : "SET", "command": "MYGRID" , "parameter": "' + grid + '", "timestamp" : '+Date.now()+'}'
|
||||
writeTncCommand(command)
|
||||
}
|
||||
|
||||
//Get TNC State
|
||||
exports.getTncState = function() {
|
||||
command = '{"type" : "GET", "command" : "TNC_STATE", "timestamp" : '+Date.now()+'}';
|
||||
writeTncCommand(command)
|
||||
}
|
||||
|
||||
//Get DATA State
|
||||
exports.getDataState = function() {
|
||||
command = '{"type" : "GET", "command" : "DATA_STATE", "timestamp" : '+Date.now()+'}';
|
||||
//writeTncCommand(command)
|
||||
}
|
||||
|
||||
//Get Heard Stations
|
||||
exports.getHeardStations = function() {
|
||||
command = '{"type" : "GET", "command" : "HEARD_STATIONS", "timestamp" : '+Date.now()+'}';
|
||||
writeTncCommand(command)
|
||||
}
|
||||
|
||||
|
||||
// Send Ping
|
||||
exports.sendPing = function(dxcallsign) {
|
||||
command = '{"type" : "PING", "command" : "PING", "dxcallsign" : "' + dxcallsign + '", "timestamp" : '+Date.now()+'}'
|
||||
writeTncCommand(command)
|
||||
}
|
||||
|
||||
// Send CQ
|
||||
exports.sendCQ = function() {
|
||||
command = '{"type" : "CQ", "command" : "CQCQCQ", "timestamp" : '+Date.now()+'}'
|
||||
writeTncCommand(command)
|
||||
}
|
||||
|
||||
// Send File
|
||||
exports.sendFile = function(dxcallsign, mode, frames, filename, filetype, data, checksum) {
|
||||
command = '{"type" : "ARQ", "command" : "sendFile", "dxcallsign" : "'+dxcallsign+'", "mode" : "'+mode+'", "n_frames" : "'+frames+'", "filename" : "'+filename+'", "filetype" : "'+filetype+'", "data" : "'+data+'", "checksum" : "'+checksum+'", "timestamp" : '+Date.now()+'}'
|
||||
writeTncCommand(command)
|
||||
}
|
||||
|
||||
// Send Message
|
||||
exports.sendMessage = function(dxcallsign, mode, frames, data, checksum) {
|
||||
command = '{"type" : "ARQ", "command" : "sendMessage", "dxcallsign" : " '+dxcallsign+' ", "mode" : " '+mode+' ", "n_frames" : " '+frames+' ", "data" : " '+data+' ", "checksum" : " '+checksum+' ", "timestamp" : '+Date.now()+'}'
|
||||
writeTncCommand(command)
|
||||
}
|
Loading…
Reference in a new issue