1
0
Fork 0
mirror of https://github.com/iNavFlight/inav-configurator.git synced 2025-07-15 20:35:19 +03:00

Add progress dialogs for flash save and erase

This commit is contained in:
Nicholas Sherlock 2015-02-13 21:05:36 +13:00
parent 0f22b92e35
commit 49741b45a8
7 changed files with 357 additions and 63 deletions

View file

@ -845,13 +845,42 @@
"message": "Automatically loaded previous log file: <strong>$1</strong>" "message": "Automatically loaded previous log file: <strong>$1</strong>"
}, },
"dataflashNote": {
"message": "Blackbox flight logs can be stored on the onboard dataflash chip if your flight controller supports one."
},
"dataflashButtonSaveFile": { "dataflashButtonSaveFile": {
"message": "Save flash to file..." "message": "Save flash to file..."
}, },
"dataflashButtonErase": { "dataflashButtonErase": {
"message": "Erase flash" "message": "Erase flash"
}, },
"dataflashConfirmEraseTitle": {
"message": "Confirm dataflash erase"
},
"dataflashConfirmEraseNote": {
"message": "This will erase any Blackbox logs or other data contained in the dataflash which will take about 20 seconds, are you sure?"
},
"dataflashSavingTitle": {
"message": "Saving dataflash to file"
},
"dataflashSavingNote": {
"message": "Saving could take several minutes, please wait."
},
"dataflashSavingNoteAfter": {
"message": "Save completed! Press \"Ok\" to continue."
},
"dataflashButtonSaveCancel": {
"message": "Cancel"
},
"dataflashButtonSaveDismiss": {
"message": "Ok"
},
"dataflashButtonEraseConfirm": {
"message": "Yes, erase dataflash"
},
"dataflashButtonEraseCancel": {
"message": "Cancel"
},
"firmwareFlasherReleaseSummaryHead": { "firmwareFlasherReleaseSummaryHead": {
"message": "Release info" "message": "Release info"
}, },

View file

@ -151,6 +151,8 @@ var MISC = {
}; };
var DATAFLASH = { var DATAFLASH = {
ready: false,
sectors: 0, sectors: 0,
totalSize: 0 totalSize: 0,
usedSize: 0
}; };

View file

@ -670,14 +670,16 @@ var MSP = {
console.log('Led strip config saved'); console.log('Led strip config saved');
break; break;
case MSP_codes.MSP_DATAFLASH_SUMMARY: case MSP_codes.MSP_DATAFLASH_SUMMARY:
DATAFLASH.sectors = data.getUint32(0, 1); DATAFLASH.ready = (data.getUint8(0) & 1) != 0;
DATAFLASH.totalSize = data.getUint32(4, 1); DATAFLASH.sectors = data.getUint32(1, 1);
DATAFLASH.totalSize = data.getUint32(5, 1);
DATAFLASH.usedSize = data.getUint32(9, 1);
break; break;
case MSP_codes.MSP_DATAFLASH_READ: case MSP_codes.MSP_DATAFLASH_READ:
// No-op, let callback handle it // No-op, let callback handle it
break; break;
case MSP_codes.MSP_DATAFLASH_ERASE: case MSP_codes.MSP_DATAFLASH_ERASE:
console.log("Data flash erased"); console.log("Data flash erase begun...");
break; break;
case MSP_codes.MSP_SET_MODE_RANGE: case MSP_codes.MSP_SET_MODE_RANGE:
console.log('Mode range saved'); console.log('Mode range saved');

View file

@ -353,3 +353,14 @@ input[type="number"]::-webkit-inner-spin-button {
text-align: center; text-align: center;
font-weight: bold; font-weight: bold;
} }
dialog {
background-color: white;
padding: 1em;
height: auto;
margin: auto auto;
position: absolute;
width: 50%;
border-radius: 5px;
border: 1px solid silver;
}

View file

@ -1,12 +1,28 @@
.tab-dataflash { .tab-dataflash .info {
} margin: 0 0 10px 0;
.tab-dataflash .dataflash-info dd{ position: relative;
} }
.tab-dataflash .info .progressLabel {
position: absolute;
width: 100%;
height: 26px;
top: 0;
left: 0;
text-align: center;
line-height: 24px;
color: white;
font-weight: bold;
/* text-shadow: 1px 0px 2px rgba(0, 0, 0, 0.9);*/
}
.tab-dataflash .note { .tab-dataflash .note {
padding: 5px; padding: 5px;
border: 1px dashed silver; border: 1px dashed silver;
margin-bottom: 8px;
} }
.tab-dataflash .properties { .tab-dataflash .properties {
margin-top: 10px; margin-top: 10px;
@ -76,6 +92,88 @@
.tab-dataflash .buttons a:hover { .tab-dataflash .buttons a:hover {
background-color: #dedcdc; background-color: #dedcdc;
} }
.tab-dataflash .buttons .back { .tab-dataflash .buttons a.disabled {
cursor: default;
color: #999;
pointer-events: none;
}
.tab-dataflash .dataflash-progress {
display: none; display: none;
}
.tab-dataflash .dataflash-contents {
margin:9px 16px;
border:1px solid silver;
background-color:#eee;
display:flex;
flex-direction:row;
flex-wrap:nowrap;
justify-content:flex-start;
border-radius:6px;
}
.tab-dataflash .dataflash-contents li {
height:26px;
position:relative;
}
.tab-dataflash .dataflash-contents li div {
position:absolute;
top:26px;
margin-top:4px;
text-align:center;
left: 0;
right: 0;
}
.tab-dataflash .dataflash-used {
background-color:#bcf;
}
.tab-dataflash progress::-webkit-progress-bar {
height:24px;
background-color:#eee;
}
.tab-dataflash progress::-webkit-progress-value {
background-color:#bcf;
}
.tab-dataflash dialog {
width:40em;
}
.tab-dataflash dialog .buttons {
position:static;
margin-top: 2em;
overflow: hidden;
width:auto;
}
.tab-dataflash dialog h3 {
margin-bottom: 0.5em;
}
.dataflash-confirm-erase .dataflash-erase-progress {
height:125px;
display:none;
}
.dataflash-confirm-erase.erasing .dataflash-erase-progress {
display:block;
}
.dataflash-confirm-erase.erasing h3,
.dataflash-confirm-erase.erasing .erase-flash-confirm,
.dataflash-confirm-erase.erasing .dataflash-confirm-erase-note {
display:none;
}
.tab-dataflash progress {
display:block;
width:100%;
margin:1em 0;
}
.dataflash-saving .dataflash-saving-after {
display:none;
}
.dataflash-saving.done .dataflash-saving-before {
display:none;
}
.dataflash-saving.done .dataflash-saving-after {
display:block;
} }

View file

@ -1,14 +1,63 @@
<div class="tab-dataflash"> <div class="tab-dataflash">
<h3>Dataflash</h3> <div class="note" i18n="dataflashNote">
<dl class="dataflash-info"> </div>
<dt>Capacity (bytes)</dt>
<dd class="dataflash-capacity"></dd> <dialog class="dataflash-confirm-erase">
<dt>Capacity (sectors)</dt> <h3 i18n="dataflashConfirmEraseTitle"></h3>
<dd class="dataflash-sectors"></dd> <div class="dataflash-confirm-erase-note" i18n="dataflashConfirmEraseNote">
</dl> </div>
<div class="dataflash-erase-progress">
<div class="data-loading">
<p>Erase in progress, please wait...</p>
</div>
</div>
<div class="buttons">
<a href="#" class="erase-flash-cancel" i18n="dataflashButtonEraseCancel"></a>
<a href="#" class="erase-flash-confirm" i18n="dataflashButtonEraseConfirm"></a>
</div>
</dialog>
<dialog class="dataflash-saving">
<h3 i18n="dataflashSavingTitle"></h3>
<div class="dataflash-saving-before">
<div i18n="dataflashSavingNote">
</div>
<progress value="0" min="0" max="100"></progress>
<div class="buttons">
<a href="#" class="save-flash-cancel" i18n="dataflashButtonSaveCancel"></a>
</div>
</div>
<div class="dataflash-saving-after">
<div i18n="dataflashSavingNoteAfter">
</div>
<div class="buttons">
<a href="#" class="save-flash-dismiss" i18n="dataflashButtonSaveDismiss"></a>
</div>
</div>
</dialog>
<h3>Dataflash contents</h3>
<ul class="dataflash-contents">
<li class="dataflash-used">
<div class="legend">
Used space
</div>
</li>
<li class="dataflash-free">
<div class="legend">
Free space
</div>
</li>
</ul>
<div class="buttons"> <div class="buttons">
<a href="#" class="erase_flash" i18n="dataflashButtonErase"></a> <a href="#" class="erase-flash" i18n="dataflashButtonErase"></a>
<a href="#" class="save_to_file" i18n="dataflashButtonSaveFile"></a> <a href="#" class="save-flash" i18n="dataflashButtonSaveFile"></a>
</div> </div>
</div> </div>

View file

@ -2,39 +2,93 @@
TABS.dataflash = {}; TABS.dataflash = {};
TABS.dataflash.initialize = function (callback) { TABS.dataflash.initialize = function (callback) {
var self = this; var
self = this,
saveCancelled, eraseCancelled;
if (GUI.active_tab != 'dataflash') { if (GUI.active_tab != 'dataflash') {
GUI.active_tab = 'dataflash'; GUI.active_tab = 'dataflash';
googleAnalytics.sendAppView('dataflash'); googleAnalytics.sendAppView('dataflash');
} }
var requested_properties = [], var
requested_properties = [],
samples = 0, samples = 0,
requests = 0, requests = 0,
log_buffer = []; log_buffer = [];
if (CONFIGURATOR.connectionValid) { if (CONFIGURATOR.connectionValid) {
MSP.send_message(MSP_codes.MSP_DATAFLASH_SUMMARY, false, false, function() { MSP.send_message(MSP_codes.MSP_DATAFLASH_SUMMARY, false, false, function() {
$('#content').load("./tabs/dataflash.html", process_html); $('#content').load("./tabs/dataflash.html", function() {
create_html();
});
}); });
} }
function process_html() { function formatFilesize(bytes) {
if (bytes < 1024) {
return bytes + "B";
}
var kilobytes = bytes / 1024;
if (kilobytes < 1024) {
return Math.round(kilobytes) + "kB";
}
var megabytes = kilobytes / 1024;
return megabytes.toFixed(1) + "MB";
}
function update_html() {
if (DATAFLASH.usedSize > 0) {
$(".tab-dataflash .dataflash-used").css({
width: (DATAFLASH.usedSize / DATAFLASH.totalSize * 100) + "%",
display: 'block'
});
$(".tab-dataflash .dataflash-used div").text('Used space ' + formatFilesize(DATAFLASH.usedSize));
} else {
$(".tab-dataflash .dataflash-used").css({
display: 'none'
});
}
if (DATAFLASH.totalSize - DATAFLASH.usedSize > 0) {
$(".tab-dataflash .dataflash-free").css({
width: ((DATAFLASH.totalSize - DATAFLASH.usedSize) / DATAFLASH.totalSize * 100) + "%",
display: 'block'
});
$(".tab-dataflash .dataflash-free div").text('Free space ' + formatFilesize(DATAFLASH.totalSize - DATAFLASH.usedSize));
} else {
$(".tab-dataflash .dataflash-free").css({
display: 'none'
});
}
$(".tab-dataflash a.erase-flash, .tab-dataflash a.save-flash").toggleClass("disabled", DATAFLASH.usedSize == 0);
}
function create_html() {
// translate to user-selected language // translate to user-selected language
localize(); localize();
$(".tab-dataflash .dataflash-capacity").text(DATAFLASH.totalSize);
$(".tab-dataflash .dataflash-sectors").text(DATAFLASH.sectors);
// UI hooks // UI hooks
$('.tab-dataflash a.erase_flash').click(erase_flash); $('.tab-dataflash a.erase-flash').click(ask_to_erase_flash);
$('.tab-dataflash a.save_to_file').click(stream_flash_to_file); $('.tab-dataflash a.erase-flash-confirm').click(flash_erase);
$('.tab-dataflash a.erase-flash-cancel').click(flash_erase_cancel);
$('.tab-dataflash a.save-flash').click(flash_save_begin);
$('.tab-dataflash a.save-flash-cancel').click(flash_save_cancel);
$('.tab-dataflash a.save-flash-dismiss').click(dismiss_saving_dialog);
update_html();
if (callback) callback(); if (callback) callback();
} }
// IO related methods // IO related methods
function zeroPad(value, width) { function zeroPad(value, width) {
value = "" + value; value = "" + value;
@ -46,21 +100,60 @@ TABS.dataflash.initialize = function (callback) {
return value; return value;
} }
function stream_flash_to_file() { function flash_save_cancel() {
saveCancelled = true;
}
function show_saving_dialog() {
$(".dataflash-saving progress").attr("value", 0);
saveCancelled = false;
$(".dataflash-saving").removeClass("done");
$(".dataflash-saving")[0].showModal();
}
function dismiss_saving_dialog() {
$(".dataflash-saving")[0].close();
}
function mark_saving_dialog_done() {
$(".dataflash-saving").addClass("done");
}
function flash_save_begin() {
var
maxBytes = DATAFLASH.usedSize;
if (GUI.connected_to) { if (GUI.connected_to) {
prepare_file(function(fileWriter) { prepare_file(function(fileWriter) {
var var
nextAddress = 0; nextAddress = 0;
show_saving_dialog();
function onChunkRead(chunkAddress, chunkDataView) { function onChunkRead(chunkAddress, chunkDataView) {
// If we didn't get a zero-byte chunk (indicating end-of-file), request more // If we didn't get a zero-byte chunk (indicating end-of-file), request more
if (chunkDataView.byteLength > 0) { if (chunkDataView.byteLength > 0) {
var blob = new Blob([chunkDataView]); nextAddress += chunkDataView.byteLength;
$(".dataflash-saving progress").attr("value", nextAddress / maxBytes * 100);
var
blob = new Blob([chunkDataView]);
fileWriter.write(blob); fileWriter.write(blob);
nextAddress += chunkDataView.byteLength; if (saveCancelled || nextAddress >= maxBytes) {
MSP.dataflashRead(nextAddress, onChunkRead); if (saveCancelled) {
dismiss_saving_dialog();
} else {
mark_saving_dialog_done();
}
} else {
MSP.dataflashRead(nextAddress, onChunkRead);
}
} else {
mark_saving_dialog_done();
} }
} }
@ -88,41 +181,51 @@ TABS.dataflash.initialize = function (callback) {
console.log('Dataflash dump file path: ' + path); console.log('Dataflash dump file path: ' + path);
}); });
prepare_writer(fileEntry, onComplete); fileEntry.createWriter(function (fileWriter) {
}); fileWriter.onerror = function (e) {
} console.error(e);
function prepare_writer(fileEntry, onComplete) { // stop logging if the procedure was/is still running
fileEntry.createWriter(function (fileWriter) { };
fileWriter.onerror = function (e) {
onComplete(fileWriter);
}, function (e) {
// File is not readable or does not exist!
console.error(e); console.error(e);
});
// stop logging if the procedure was/is still running
};
fileWriter.onwriteend = function () {
};
onComplete(fileWriter);
}, function (e) {
// File is not readable or does not exist!
console.error(e);
}); });
} }
function erase_flash() { function ask_to_erase_flash() {
/* var dialog = $("<dialog>lol</dialog>"); eraseCancelled = false;
$(".dataflash-confirm-erase").removeClass('erasing');
$("body").append(dialog);
$(".dataflash-confirm-erase")[0].showModal();
dialog[0].showModal(); }
TODO modal dialog to confirm erase */ function poll_for_erase_completion() {
MSP.send_message(MSP_codes.MSP_DATAFLASH_SUMMARY, false, false, function() {
MSP.send_message(MSP_codes.MSP_DATAFLASH_ERASE, false, false, function(data) { update_html();
if (!eraseCancelled) {
if (DATAFLASH.ready) {
$(".dataflash-confirm-erase")[0].close();
} else {
setTimeout(poll_for_erase_completion, 500);
}
}
}); });
} }
function flash_erase() {
$(".dataflash-confirm-erase").addClass('erasing');
MSP.send_message(MSP_codes.MSP_DATAFLASH_ERASE, false, false, poll_for_erase_completion);
}
function flash_erase_cancel() {
eraseCancelled = true;
$(".dataflash-confirm-erase")[0].close();
}
}; };
TABS.dataflash.cleanup = function (callback) { TABS.dataflash.cleanup = function (callback) {