// Lets see if CW is selected const ModeSelected = document.getElementById('mode'); $('#winkey_buttons').hide(); if (location.protocol == 'http:') { // Do something if the page is being served over SSL $('#winkey').hide(); // Hide the CW buttons } if (ModeSelected.value == 'CW') { // Show the CW buttons $('#winkey').show(); } else { // Hide the CW buttons $('#winkey').hide(); } ModeSelected.addEventListener('change', (event) => { if (event.target.value == 'CW') { // Show the CW buttons $('#winkey').show(); } else { // Hide the CW buttons $('#winkey').hide(); } }); $('#winkeycwspeed').change(function (event) { // Get the value from the input let speed = parseInt($('#winkeycwspeed').val(), 10); // Convert to hexadecimal and pad if necessary let hexspeed = speed.toString(16).padStart(2, '0'); // Create the command let command = `02 ${hexspeed}`; // Send the command as hex bytes sendHexToSerial(command); }); let function1Name, function1Macro, function2Name, function2Macro, function3Name, function3Macro, function4Name, function4Macro, function5Name, function5Macro; getMacros(); document.addEventListener('keydown', function(event) { if (event.key === 'F1') { event.preventDefault(); morsekey_func1(); } if (event.key === 'F2') { event.preventDefault(); morsekey_func2(); } if (event.key === 'F3') { event.preventDefault(); morsekey_func3(); } if (event.key === 'F4') { event.preventDefault(); morsekey_func4(); } if (event.key === 'F5') { event.preventDefault(); morsekey_func5(); } }); let sendText = document.getElementById("sendText"); let sendButton = document.getElementById("sendButton"); let receiveText = document.getElementById("receiveText"); let connectButton = document.getElementById("connectButton"); let statusBar = document.getElementById("statusBar"); //Couple the elements to the Events connectButton.addEventListener("click", clickConnect); sendButton.addEventListener("click", clickSend); //When the connectButton is pressed async function clickConnect() { if (port) { //if already connected, disconnect disconnect(); $('#winkey_buttons').hide(); } else { //otherwise connect await connect(); $('#winkey_buttons').show(); } } //Define outputstream, inputstream and port so they can be used throughout the sketch var outputStream, inputStream, port; navigator.serial.addEventListener('connect', e => { statusBar.innerText = `Connected to ${e.port}`; connectButton.innerText = "Disconnect" }); navigator.serial.addEventListener('disconnect', e => { statusBar.innerText = `Disconnected`; connectButton.innerText = "Connect" }); let debug = 0; let speed = 24; let minSpeed = 20; let maxSpeed = 40; //Connect to the serial async function connect() { //Optional filter to only see relevant boards const filter = { usbVendorId: 0x2341 // Arduino SA }; //Try to connect to the Serial port try { port = await navigator.serial.requestPort(/*{ filters: [filter] }*/); // Continue connecting to |port|. // - Wait for the port to open. await port.open({ baudRate: 1200 }); await port.setSignals({ dataTerminalReady: true }); statusBar.innerText = "Connected"; connectButton.innerText = "Disconnect" let decoder = new TextDecoderStream(); inputDone = port.readable.pipeTo(decoder.writable); inputStream = decoder.readable; // Keyer init sendHexToSerial("00 02"); await delay(300); // Wait for 300ms sendHexToSerial("02 00"); await delay(300); // Wait for 300ms sendHexToSerial("02 14"); // init 20 wpm $('#winkey_buttons').show(); reader = inputStream.getReader(); readLoop(); } catch (e) { //If the pipeTo error appears; clarify the problem by giving suggestions. if (e == "TypeError: Cannot read property 'pipeTo' of undefined") { e += "\n Use Google Chrome and enable-experimental-web-platform-features" } connectButton.innerText = "Connect" statusBar.innerText = e; } } function stop_cw_sending() { sendHexToSerial("0A"); $("#send_carrier").attr("hidden", false); $("#stop_carrier").attr("hidden", true); } function send_carrier() { sendHexToSerial("0B 01"); $("#send_carrier").attr("hidden", true); $("#stop_carrier").attr("hidden", false); } function stop_carrier() { sendHexToSerial("0B 00"); $("#send_carrier").attr("hidden", false); $("#stop_carrier").attr("hidden", true); } function delay(ms) { return new Promise(resolve => setTimeout(resolve, ms)); } // Helper function to convert a hex string to a Uint8Array function hexStringToUint8Array(hexString) { // Remove any spaces or non-hex characters hexString = hexString.replace(/[^0-9a-f]/gi, ''); // Ensure the string has an even length if (hexString.length % 2 !== 0) { console.warn('Hex string has an odd length, padding with a leading zero.'); hexString = '0' + hexString; } const byteArray = new Uint8Array(hexString.length / 2); for (let i = 0; i < hexString.length; i += 2) { byteArray[i / 2] = parseInt(hexString.substr(i, 2), 16); } return byteArray; } async function sendHexToSerial(hexString) { if (port && port.writable) { // Convert the hex string to a Uint8Array const byteArray = hexStringToUint8Array(hexString); // Create a writer from the writable stream const writer = port.writable.getWriter(); try { // Write the byte array to the serial port await writer.write(byteArray); } catch (error) { console.error('Error writing to serial port:', error); } finally { // Release the lock on the writer writer.releaseLock(); } } else { console.error('Port is not available or writable.'); } } //Write to the Serial port async function writeToStream(line) { const outputStream = port.writable.getWriter(); // Convert the text to a Uint8Array const encoder = new TextEncoder(); const buffer = encoder.encode(line.toUpperCase()); // Write the Uint8Array to the serial port await outputStream.write(buffer); // Release the stream lock outputStream.releaseLock(); } //Disconnect from the Serial port async function disconnect() { sendHexToSerial("00 03"); if (reader) { await reader.cancel(); await inputDone.catch(() => { }); reader = null; inputDone = null; } if (outputStream) { await outputStream.getWriter().close(); await outputDone; outputStream = null; outputDone = null; } statusBar.innerText = "Disconnected"; connectButton.innerText = "Connect" //Close the port. await port.close(); port = null; } //When the send button is pressed function clickSend() { writeToStream(sendText.value).then(function() { // writeToStream("\r"); //and clear the input field, so it's clear it has been sent $('#sendText').val(''); }); } function morsekey_func1() { console.log("F1: " + UpdateMacros(function1Macro)); writeToStream(UpdateMacros(function1Macro)); //and clear the input field, so it's clear it has been sent sendText.value = ""; } function morsekey_func2() { console.log("F2: " + UpdateMacros(function2Macro)); writeToStream(UpdateMacros(function2Macro)); //and clear the input field, so it's clear it has been sent sendText.value = ""; } function morsekey_func3() { console.log("F3: " + UpdateMacros(function3Macro)); writeToStream(UpdateMacros(function3Macro)); //and clear the input field, so it's clear it has been sent sendText.value = ""; } function morsekey_func4() { console.log("F4: " + UpdateMacros(function4Macro)); writeToStream(UpdateMacros(function4Macro)); //and clear the input field, so it's clear it has been sent sendText.value = ""; } function morsekey_func5() { console.log("F5: " + UpdateMacros(function5Macro)); writeToStream(UpdateMacros(function5Macro)); //and clear the input field, so it's clear it has been sent sendText.value = ""; } //Read the incoming data async function readLoop() { while (true) { const { value, done } = await reader.read(); if (done === true){ break; } console.log(value); //When recieved something add it to the big textarea receiveText.value += value; //Scroll to the bottom of the text field receiveText.scrollTop = receiveText.scrollHeight; } } function UpdateMacros(macrotext) { // Get the values from the form set to uppercase let CALL = document.getElementById("callsign").value.toUpperCase(); let RSTS = document.getElementById("rst_sent").value; let newString; newString = macrotext.replace(/\[MYCALL\]/g, my_call); newString = newString.replace(/\[CALL\]/g, CALL); newString = newString.replace(/\[RSTS\]/g, RSTS); console.log(newString); return newString; } // Call url and store the returned json data as variables function getMacros() { fetch(base_url + 'index.php/qso/cwmacros_json') .then(response => response.json()) .then(data => { function1Name = data.function1_name; function1Macro = data.function1_macro; function2Name = data.function2_name; function2Macro = data.function2_macro; function3Name = data.function3_name; function3Macro = data.function3_macro; function4Name = data.function4_name; function4Macro = data.function4_macro; function5Name = data.function5_name; function5Macro = data.function5_macro; const morsekey_func1_Button = document.getElementById('morsekey_func1'); morsekey_func1_Button.textContent = 'F1 (' + function1Name + ')'; const morsekey_func2_Button = document.getElementById('morsekey_func2'); morsekey_func2_Button.textContent = 'F2 (' + function2Name + ')'; const morsekey_func3_Button = document.getElementById('morsekey_func3'); morsekey_func3_Button.textContent = 'F3 (' + function3Name + ')'; const morsekey_func4_Button = document.getElementById('morsekey_func4'); morsekey_func4_Button.textContent = 'F4 (' + function4Name + ')'; const morsekey_func5_Button = document.getElementById('morsekey_func5'); morsekey_func5_Button.textContent = 'F5 (' + function5Name + ')'; }); } $('#winkey_settings').click(function (event) { $.ajax({ url: base_url + 'index.php/qso/winkeysettings', type: 'post', success: function (html) { BootstrapDialog.show({ title: 'Winkey Macros', size: BootstrapDialog.SIZE_NORMAL, cssClass: 'options', nl2br: false, message: html, onshown: function(dialog) { }, buttons: [{ label: 'Save', cssClass: 'btn-primary btn-sm', id: 'saveButton', action: function (dialogItself) { winkey_macro_save(); dialogItself.close(); } }, { label: lang_admin_close, cssClass: 'btn-sm', id: 'closeButton', action: function (dialogItself) { $('#optionButton').prop("disabled", false); dialogItself.close(); } }], onhide: function(dialogRef){ $('#optionButton').prop("disabled", false); }, }); } }); }); function winkey_macro_save() { $.ajax({ url: base_url + 'index.php/qso/cwmacrosave', type: 'post', data: { function1_name: $('#function1_name').val(), function1_macro: $('#function1_macro').val(), function2_name: $('#function2_name').val(), function2_macro: $('#function2_macro').val(), function3_name: $('#function3_name').val(), function3_macro: $('#function3_macro').val(), function4_name: $('#function4_name').val(), function4_macro: $('#function4_macro').val(), function5_name: $('#function5_name').val(), function5_macro: $('#function5_macro').val(), }, success: function (html) { BootstrapDialog.alert({ title: 'INFO', message: 'Macros were saved.', type: BootstrapDialog.TYPE_INFO, closable: false, draggable: false, callback: function (result) { getMacros(); } }); } }); }