SpeedyLoader/renderer.js

514 lines
16 KiB
JavaScript
Raw Normal View History

2018-12-31 05:59:18 -08:00
const serialport = require('serialport')
2021-01-12 03:04:10 -08:00
const usb = require('usb')
2019-01-01 15:47:54 -08:00
const {ipcRenderer} = require("electron")
2019-01-14 14:34:00 -08:00
const {remote} = require('electron')
const { shell } = require('electron')
2018-12-31 05:59:18 -08:00
2019-07-16 23:51:27 -07:00
var basetuneList = [];
2020-10-20 19:05:17 -07:00
var isTeensy = false;
2019-07-16 23:51:27 -07:00
2021-01-12 03:04:10 -08:00
function getTeensyVersion(id)
{
var idString = ""
switch(id) {
case 0x273:
idString = "LC"
break;
case 0x274:
idString = "3.0"
break;
case 0x275:
idString = "3.2"
break;
case 0x276:
idString = "3.5"
break;
case 0x277:
idString = "3.6"
break;
case 0x279:
idString = "4.0"
break;
case 0x280:
idString = "4.1"
break;
}
return idString;
}
2018-12-31 05:59:18 -08:00
function refreshSerialPorts()
{
serialport.list((err, ports) => {
console.log('Serial ports found: ', ports);
2021-01-12 03:04:10 -08:00
2018-12-31 05:59:18 -08:00
if (err) {
document.getElementById('serialDetectError').textContent = err.message
return
} else {
document.getElementById('serialDetectError').textContent = ''
}
if (ports.length === 0) {
document.getElementById('serialDetectError').textContent = 'No ports discovered'
}
select = document.getElementById('portsSelect');
//Clear the current options
for (i = 0; i <= select.options.length; i++)
2018-12-31 05:59:18 -08:00
{
select.remove(0); //Always 0 index (As each time an item is removed, everything shuffles up 1 place)
2018-12-31 05:59:18 -08:00
}
//Load the current serial values
2019-01-01 15:47:54 -08:00
for(var i = 0; i < ports.length; i++)
2018-12-31 05:59:18 -08:00
{
var newOption = document.createElement('option');
newOption.value = ports[i].comName;
newOption.innerHTML = ports[i].comName;
2020-10-20 19:05:17 -07:00
if(ports[i].vendorId == "2341")
{
//Arduino device
if(ports[i].productId == "0010" || ports[i].productId == "0042")
{
//Mega2560
newOption.innerHTML = newOption.innerHTML + " (Arduino Mega)";
2021-01-12 03:04:10 -08:00
newOption.setAttribute("board", "ATMEGA2560");
2020-10-20 19:05:17 -07:00
}
}
2020-10-21 03:01:53 -07:00
else if(ports[i].vendorId == "16c0" || ports[i].vendorId == "16C0")
2020-10-20 19:05:17 -07:00
{
//Teensy
2021-01-12 03:04:10 -08:00
var teensyDevices = usb.getDeviceList().filter( function(d) { return d.deviceDescriptor.idVendor===0x16C0; });
var teensyVersion = getTeensyVersion(teensyDevices[0].deviceDescriptor.bcdDevice);
newOption.innerHTML = newOption.innerHTML + " (Teensy " + teensyVersion + ")";
//Get the short copy of the teensy version
teensyVersion = teensyVersion.replace(".", "");
newOption.setAttribute("board", "TEENSY"+teensyVersion);
2020-10-20 19:05:17 -07:00
}
select.add(newOption);
2020-10-20 19:05:17 -07:00
//console.log(ports[i].serialNumber );
2018-12-31 05:59:18 -08:00
}
2021-01-12 03:04:10 -08:00
//Look for any unintialised Teensy boards (ie boards in HID rather than serial mode)
var uninitialisedTeensyDevices = usb.getDeviceList().filter( function(d) {
return d.deviceDescriptor.idVendor===0x16C0 && d.configDescriptor.interfaces[0][0].bInterfaceClass == 3; //Interface class 3 is HID
});
uninitialisedTeensyDevices.forEach((device, index) => {
console.log("Uninit Teensy found: ", getTeensyVersion(device.deviceDescriptor.bcdDevice))
var newOption = document.createElement('option');
newOption.value = "TeensyHID";
var teensyVersion = getTeensyVersion(device.deviceDescriptor.bcdDevice);
newOption.innerHTML = "Uninitialised Teensy " + teensyVersion;
teensyVersion = teensyVersion.replace(".", "");
newOption.setAttribute("board", "TEENSY"+teensyVersion);
newOption.setAttribute("uninitialised", "true");
select.add(newOption);
})
2020-10-20 19:05:17 -07:00
var button = document.getElementById("btnInstall")
if(ports.length > 0)
{
select.selectedIndex = 0;
button.disabled = false;
}
else { button.disabled = true; }
2018-12-31 05:59:18 -08:00
})
}
2019-01-01 19:47:18 -08:00
function refreshDetails()
2019-01-01 15:47:54 -08:00
{
var selectElement = document.getElementById('versionsSelect');
var version = selectElement.options[selectElement.selectedIndex].value;
var url = "https://api.github.com/repos/noisymime/speeduino/releases/tags/" + version;
document.getElementById('detailsHeading').innerHTML = version;
2019-01-01 15:47:54 -08:00
2019-01-01 19:47:18 -08:00
var request = require('request');
const options = {
url: url,
headers: {
'User-Agent': 'request'
}
};
request.get(options, function (error, response, body) {
if (!error ) {
console.log(body);
var result = JSON.parse(body);
// Continue with your processing here.
textField = document.getElementById('detailsText');
//Need to convert the Markdown that comes from Github to HTML
var myMarked = require('marked');
textField.innerHTML = myMarked(result.body);
}
});
//Finally, make the details section visible
document.getElementById('details').style.display = "inline";
//And jump to it
window.location.href = "#details";
2019-01-01 15:47:54 -08:00
}
function refreshAvailableFirmwares()
{
//Disable the buttons. These are only re-enabled if the retrieve is successful
var DetailsButton = document.getElementById("btnDetails");
var ChoosePortButton = document.getElementById("btnChoosePort");
2019-07-16 23:51:27 -07:00
var basetuneButton = document.getElementById("btnBasetune");
DetailsButton.disabled = true;
ChoosePortButton.disabled = true;
2019-07-16 23:51:27 -07:00
basetuneButton.disabled = true;
2019-01-01 15:47:54 -08:00
var request = require('request');
request.get('http://speeduino.com/fw/versions', {timeout: 10000}, function (error, response, body)
{
select = document.getElementById('versionsSelect');
2019-01-01 15:47:54 -08:00
if (!error && response.statusCode == 200) {
var lines = body.split('\n');
// Continue with your processing here.
2019-01-01 15:47:54 -08:00
for(var i = 0;i < lines.length;i++)
{
var newOption = document.createElement('option');
newOption.value = lines[i];
newOption.innerHTML = lines[i];
select.appendChild(newOption);
}
select.selectedIndex = 0;
2019-07-16 23:51:27 -07:00
refreshBasetunes();
//Re-enable the buttons
DetailsButton.disabled = false;
ChoosePortButton.disabled = false;
2019-07-16 23:51:27 -07:00
basetuneButton.disabled = false;
2019-01-01 15:47:54 -08:00
}
else if(error)
{
console.log("Error retrieving available firmwares");
var newOption = document.createElement('option');
if(error.code === 'ETIMEDOUT')
{
newOption.value = "Connection timed out";
newOption.innerHTML = "Connection timed out";
}
else
{
newOption.value = "Cannot retrieve firmware list";
newOption.innerHTML = "Cannot retrieve firmware list. Check internet connection and restart";
}
select.appendChild(newOption);
}
else if(response.statusCode == 404)
{
}
}
);
2019-01-01 15:47:54 -08:00
}
2019-07-16 23:51:27 -07:00
function refreshBasetunes()
{
//Check whether the base tunes list has been populated yet
if(basetuneList === undefined || basetuneList.length == 0)
{
console.log("No tunes loaded. Retrieving from server");
//Load the json
var url = "https://speeduino.com/fw/basetunes.json";
var request = require('request');
const options = {
url: url,
headers: {
'User-Agent': 'request'
}
};
request.get(options, function (error, response, body) {
if (!error )
{
basetuneList = JSON.parse(body);
refreshBasetunes();
}
});
}
else
{
//JSON list of base tunes has been downloaded
//Get the display list object
var select = document.getElementById('basetunesSelect');
//Get the currently selected version
selectElement = document.getElementById('versionsSelect');
if(selectElement.selectedIndex == -1) { return; } //Check for no value being selected
var selectedFW = selectElement.options[selectElement.selectedIndex].value;
//Clear the current options from the list
while(select.options.length)
{
select.remove(0);
}
for (var tune in basetuneList)
{
//Check whether the current tune was available for the selected firmware
if(parseInt(basetuneList[tune].introduced) <= parseInt(selectedFW))
{
var url = basetuneList[tune].baseURL.replace("$VERSION", selectedFW) + basetuneList[tune].filename;
//console.log("Tune url: " + url);
//console.log("Found a valid tune: " + basetuneList[tune].displayName);
var newOption = document.createElement('option');
newOption.style.background = "#022b3a";
newOption.value = url;
newOption.innerHTML = basetuneList[tune].displayName;
select.appendChild(newOption);
}
}
//Finally update the selected firmware label on the basetunes page
document.getElementById('basetunesSelectedFW').innerHTML = selectedFW;
}
}
2020-10-20 19:05:17 -07:00
function downloadHex(isTeensy)
2018-12-31 05:59:18 -08:00
{
2018-12-31 05:59:18 -08:00
var e = document.getElementById('versionsSelect');
2020-10-20 19:05:17 -07:00
var DLurl;
if(isTeensy)
{
if(e.options[e.selectedIndex].value == 'master') { DLurl = "http://speeduino.com/fw/teensy35/" + e.options[e.selectedIndex].value + ".hex"; }
else { DLurl = "http://speeduino.com/fw/teensy35/" + e.options[e.selectedIndex].value + "-teensy35.hex"; }
2020-10-20 19:05:17 -07:00
console.log("Downloading Teensy 35 firmware: " + DLurl);
}
else
{
DLurl = "http://speeduino.com/fw/bin/" + e.options[e.selectedIndex].value + ".hex";
console.log("Downloading AVR firmware: " + DLurl);
}
2018-12-31 05:59:18 -08:00
//Download the Hex file
ipcRenderer.send("download", {
url: DLurl,
properties: {directory: "downloads"}
});
}
function downloadIni()
{
var e = document.getElementById('versionsSelect');
2019-07-16 23:51:27 -07:00
var DLurl = "https://speeduino.com/fw/" + e.options[e.selectedIndex].value + ".ini";
console.log("Downloading: " + DLurl);
//Download the ini file
2018-12-31 05:59:18 -08:00
ipcRenderer.send("download", {
url: DLurl,
properties: {directory: "downloads"}
});
2018-12-31 05:59:18 -08:00
}
2019-07-16 23:51:27 -07:00
function downloadBasetune()
{
2019-07-16 23:51:27 -07:00
var basetuneSelect = document.getElementById('basetunesSelect');
var version = document.getElementById('versionsSelect');
//var DLurl = "https://github.com/noisymime/speeduino/raw/" + version + "/reference/Base%20Tunes/" + e.options[e.selectedIndex].value;
var DLurl = basetuneSelect.options[basetuneSelect.selectedIndex].value;
2019-07-08 21:56:05 -07:00
console.log("Downloading: " + DLurl);
//Download the ini file
ipcRenderer.send("download", {
url: DLurl,
properties: {directory: "downloads"}
});
2019-07-16 23:51:27 -07:00
}
//Installing the Windows drivers
function installDrivers()
{
ipcRenderer.send("installWinDrivers", {
});
2019-07-08 21:56:05 -07:00
}
2019-01-01 15:47:54 -08:00
function uploadFW()
2018-12-31 05:59:18 -08:00
{
//Jump to the progress section
window.location.href = "#progress";
2019-01-03 23:54:39 -08:00
//Start the spinner
var spinner = document.getElementById('progressSpinner');
//Disable the Re-burn/re-install button
var reinstallButton = document.getElementById("btnReinstall")
reinstallButton.disabled = true;
2019-01-03 23:54:39 -08:00
//Remove any old icons
spinner.classList.remove('fa-pause');
spinner.classList.remove('fa-check');
spinner.classList.remove('fa-times');
spinner.classList.add('fa-spinner');
2020-10-20 19:05:17 -07:00
//Lookup what platform we're using
var portSelect = document.getElementById('portsSelect');
var isTeensy = portSelect.innerHTML.includes("Teensy");
2019-01-14 14:21:56 -08:00
//Hide the terminal section incase it was there from a previous burn attempt
document.getElementById('terminalSection').style.display = "none";
//Same for the ini location link
document.getElementById('iniFileText').style.display = "none";
2019-01-14 14:21:56 -08:00
2019-01-01 19:47:18 -08:00
var statusText = document.getElementById('statusText');
2019-01-04 04:10:03 -08:00
var burnPercentText = document.getElementById('burnPercent');
statusText.innerHTML = "Downloading INI file"
downloadIni();
2019-01-01 19:47:18 -08:00
2019-01-04 04:10:03 -08:00
ipcRenderer.on("download complete", (event, file, state) => {
console.log("Saved file: " + file); // Full file path
var extension = file.substr(file.length - 3);
if(extension == "ini")
{
statusText.innerHTML = "Downloading firmware"
document.getElementById('iniFileText').style.display = "block"
document.getElementById('iniFileLocation').innerHTML = file
2020-10-20 19:05:17 -07:00
downloadHex(isTeensy);
2021-01-12 03:04:10 -08:00
//downloadHex(e.options[e.selectedIndex].getAttribute("board"));
}
else if(extension == "hex")
{
2019-01-14 14:21:56 -08:00
statusText.innerHTML = "Beginning upload..."
//Retrieve the select serial port
var e = document.getElementById('portsSelect');
uploadPort = e.options[e.selectedIndex].value;
console.log("Using port: " + uploadPort);
2020-10-20 23:00:15 -07:00
//Show the sponsor banner
2020-10-21 00:50:51 -07:00
document.getElementById('sponsor').style.height = "7em"
2020-10-20 23:00:15 -07:00
//Begin the upload
2020-10-20 19:05:17 -07:00
if(isTeensy)
{
ipcRenderer.send("uploadFW_teensy35", {
port: uploadPort,
firmwareFile: file
2020-10-20 19:05:17 -07:00
});
}
else
{
ipcRenderer.send("uploadFW", {
port: uploadPort,
firmwareFile: file
});
}
}
console.log();
});
ipcRenderer.on("upload completed", (event, code) => {
2019-01-02 06:05:30 -08:00
statusText.innerHTML = "Upload to arduino completed successfully!";
2019-01-04 04:10:03 -08:00
burnPercentText.innerHTML = "";
//Turn the spinner off
2019-01-03 23:54:39 -08:00
spinner.classList.remove('fa-spinner');
spinner.classList.add('fa-check');
//Re-enable the re-burn button
reinstallButton.disabled = false;
2020-10-20 23:00:15 -07:00
2019-01-01 15:47:54 -08:00
});
2019-01-04 04:10:03 -08:00
ipcRenderer.on("upload percent", (event, percent) => {
2019-01-14 14:21:56 -08:00
statusText.innerHTML = "Uploading firmware to board"
2019-01-04 04:10:03 -08:00
burnPercentText.innerHTML = " (" + percent + "%)";
});
ipcRenderer.on("upload error", (event, code) => {
2019-01-14 14:21:56 -08:00
statusText.innerHTML = "Upload to Speeduino failed";
//Mke the terminal/error section visible
2019-01-03 23:54:39 -08:00
document.getElementById('terminalSection').style.display = "block";
document.getElementById('terminalText').innerHTML = code;
2019-01-03 23:54:39 -08:00
spinner.classList.remove('fa-spinner');
spinner.classList.add('fa-times');
reinstallButton.disabled = false;
});
2018-12-31 05:59:18 -08:00
}
//Opens a native file manager window at the location of the downloaded ini file
function openFileMgr()
{
var location = document.getElementById('iniFileLocation').innerHTML
if (location != "")
{
shell.showItemInFolder(location);
}
}
2019-01-14 14:34:00 -08:00
function quit()
{
let w = remote.getCurrentWindow();
w.close();
}
2019-07-02 00:03:07 -07:00
function checkForUpdates()
{
var url = "https://api.github.com/repos/speeduino/SpeedyLoader/releases/latest";
2019-01-26 18:01:17 -08:00
2019-07-02 00:03:07 -07:00
//document.getElementById('detailsHeading').innerHTML = version;
var request = require('request');
const options = {
url: url,
headers: {
'User-Agent': 'request'
}
};
request.get(options, function (error, response, body) {
if (!error )
{
var result = JSON.parse(body);
latest_version = result.tag_name.substring(1);
console.log("Latest version: " + latest_version);
var semver = require('semver');
if(semver.gt(latest_version, remote.app.getVersion()))
{
//New version has been found
document.getElementById('update_url').setAttribute("href", result.html_url);
document.getElementById('update_text').style.display = "block";
}
}
});
}
window.onload = function () {
2019-01-26 18:01:17 -08:00
//Adds the current version number to the Titlebar
document.getElementById('title').innerHTML = "Speeduino Universal Firmware Loader (v" + remote.app.getVersion() + ")"
2019-07-02 00:03:07 -07:00
refreshAvailableFirmwares();
2019-07-16 23:51:27 -07:00
refreshBasetunes();
2019-07-02 00:03:07 -07:00
refreshSerialPorts();
checkForUpdates();
};