From 4bb72f94809d25f8d3bd971729a28319156a049f Mon Sep 17 00:00:00 2001 From: Scott Lahteine Date: Fri, 6 Feb 2015 23:46:16 -0800 Subject: [PATCH] Highlight the edited line - Add a span to the edited text line to provide a highlight - Scroll and highlight for switch checkboxes also - Clean up initialization - More API documentation - Smarter handling of asynchronous file loading during init --- Marlin/configurator/css/configurator.css | 22 +- Marlin/configurator/index.html | 9 +- Marlin/configurator/js/configurator.js | 262 +++++++++++++---------- 3 files changed, 179 insertions(+), 114 deletions(-) diff --git a/Marlin/configurator/css/configurator.css b/Marlin/configurator/css/configurator.css index 9236ec442..2572c1c8b 100644 --- a/Marlin/configurator/css/configurator.css +++ b/Marlin/configurator/css/configurator.css @@ -2,13 +2,19 @@ /* Styles for Marlin Configurator */ body { margin: 0; padding: 0; background: #56A; color: #FFC; font-family: sans-serif; } + #main { max-width: 1000px; margin: 0 auto; } #main { padding: 0 4%; width: 92%; } #main { font-family: monospace; } + +#message { width: 80%; margin: 0 auto 0.25em; color: #FF0; text-align: center; } +#message p { padding: 2px 0; } +#message p.error, #message p.message { color: #F00; background: #FF4; font-weight: bold; border-radius: 0.8em; } +#message p.message { color: #080; background: #CFC; } + .info { color: #AAF; } .info span { color: #FFF; } .info span span { color: #000; font-weight: bold; } -p { width: 80%; color: #FF0; } #help strong { color: #0DD; } img { display: none; } label, input, select, textarea { display: block; float: left; margin: 1px 0; } @@ -24,8 +30,6 @@ input:disabled { color: #BBB; } .clear { clear: both; } h1, h2, h3, h4, h5, h6 { clear: both; } h2 { margin: 0; padding: 1em 0 0; } -#serial_stepper { padding-top: 0.75em; display: block; float: left; } -#SERIAL_PORT { display: none; } ul.tabs { display: inline; list-style: none; } ul.tabs li { display: inline; } @@ -56,3 +60,15 @@ ul.tabs li a:active { fieldset { display: none; border: 1px solid #AAA; border-radius: 1em; } fieldset legend { display: none; } + +.hilightable span { + display: block; + float: left; + width: 100%; + height: 1.3em; + background: rgba(225,255,0,1); + margin: 0 -100% -1em 0; + } + +#serial_stepper { padding-top: 0.75em; display: block; float: left; } +#SERIAL_PORT { display: none; } diff --git a/Marlin/configurator/index.html b/Marlin/configurator/index.html index 2cfbc0aff..e52c163f5 100644 --- a/Marlin/configurator/index.html +++ b/Marlin/configurator/index.html @@ -14,7 +14,10 @@

Marlin Configurator 0.1a

-

Enter values in the form, get a Marlin configuration. Will include a drop-down of known configurations.

+ +
+

Enter values in the form, get a Marlin configuration.
Will include a drop-down of known configurations.

+
@@ -64,9 +67,9 @@

Marlin/Configuration.h

-

+        

         

Marlin/Configuration_adv.h

-

+        

 
         
diff --git a/Marlin/configurator/js/configurator.js b/Marlin/configurator/js/configurator.js index 571809053..25f8a7ed9 100644 --- a/Marlin/configurator/js/configurator.js +++ b/Marlin/configurator/js/configurator.js @@ -79,7 +79,7 @@ var configuratorApp = (function(){ self = this; // a 'this' for use when 'this' is something else // Set up the form - this.setupConfigForm(); + this.initConfigForm(); // Make tabs for the fieldsets var $fset = $('#config_form fieldset'); @@ -105,78 +105,87 @@ var configuratorApp = (function(){ $('
',{class:'clear'}).appendTo('#tabs'); $tabs.find('a:first').trigger('click'); - // Make a droppable file uploader + // Make a droppable file uploader, if possible var $uploader = $('#file-upload'); var fileUploader = new BinaryFileUploader({ element: $uploader[0], onFileLoad: function(file) { self.handleFileLoad(file, $uploader); } }); - - if (!fileUploader.hasFileUploaderSupport()) alert('Your browser doesn\'t support the file reading API'); - - // Read boards.h - boards_list = {}; - - var errFunc = function(jqXHR, textStatus, errorThrown) { - alert('Failed to load '+this.url+'. Try the file field.'); - }; - - $.ajax({ - url: marlin_config+'/'+boards_file, - type: 'GET', - async: false, - cache: false, - success: function(txt) { - // Get all the boards and save them into an object - self.initBoardsFromText(txt); - has_boards = true; - }, - error: errFunc - }); - - // Read Configuration.h - $.ajax({ - url: marlin_config+'/'+config_file, - type: 'GET', - async: false, - cache: false, - success: function(txt) { - // File contents into the textarea - $config.text(txt); - // Get thermistor info too - self.initThermistorsFromText(txt); - has_config = true; - }, - error: errFunc - }); - - // Read Configuration.h - $.ajax({ - url: marlin_config+'/'+config_adv_file, - type: 'GET', - async: false, - cache: false, - success: function(txt) { - // File contents into the textarea - $config_adv.text(txt); - has_config_adv = true; - self.refreshConfigForm(); - }, - error: errFunc + if (!fileUploader.hasFileUploaderSupport()) + this.setMessage("Your browser doesn't support the file reading API.", 'error'); + + // Read boards.h, Configuration.h, Configuration_adv.h + var ajax_count = 0, success_count = 0; + var loaded_items = {}; + var config_files = [boards_file, config_file, config_adv_file]; + $.each(config_files, function(i,fname){ + self.log("Loading " + fname + "...", 3); + $.ajax({ + url: marlin_config+'/'+fname, + type: 'GET', + async: true, + cache: false, + success: function(txt) { + self.log("Loaded " + fname + "...", 3); + loaded_items[fname] = function(){ self.fileLoaded(fname, txt); }; + success_count++; + }, + complete: function() { + ajax_count++; + if (ajax_count >= 3) { + $.each(config_files, function(i,fname){ if (typeof loaded_items[fname] != 'undefined') loaded_items[fname](); }); + self.refreshConfigForm(); + if (success_count < ajax_count) + self.setMessage('Unable to load configurations. Use the upload field instead.', 'error'); + } + } + }); }); + }, + setMessage: function(msg,type) { + if (msg) { + if (typeof type == 'undefined') type = 'message'; + var $err = $('

'+msg+'

'), err = $err[0]; + $('#message').prepend($err); + var baseColor = $err.css('color').replace(/rgba?\(([^),]+,[^),]+,[^),]+).*/, 'rgba($1,'); + var d = new Date(); + err.startTime = d.getTime(); + err.pulser = setInterval(function(){ + d = new Date(); + var pulse_time = (d.getTime() - err.startTime); + $err.css({color:baseColor+(0.5+Math.sin(pulse_time/200)*0.4)+')'}); + if (pulse_time > 5000) { + clearInterval(err.pulser); + $err.remove(); + } + }, 50); + } + else { + $('#message p.error, #message p.warning').each(function() { + if (typeof this.pulser != 'undefined' && this.pulser) + clearInterval(this.pulser); + $(this).remove(); + }); + } }, + /** + * Init the boards array from a boards.h file + */ initBoardsFromText: function(txt) { boards_list = {}; var r, findDef = new RegExp('[ \\t]*#define[ \\t]+(BOARD_[\\w_]+)[ \\t]+(\\d+)[ \\t]*(//[ \\t]*)?(.+)?', 'gm'); while((r = findDef.exec(txt)) !== null) { boards_list[r[1]] = r[2].prePad(3, '  ') + " — " + r[4].replace(/\).*/, ')'); } - this.log("Loaded boards", 0); - this.log(boards_list, 0); + this.log("Loaded boards", 3); this.log(boards_list, 3); + has_boards = true; }, + /** + * Init the thermistors array from the Configuration.h file + */ initThermistorsFromText: function(txt) { // Get all the thermistors and save them into an object var r, s, findDef = new RegExp('(//.*\n)+\\s+(#define[ \\t]+TEMP_SENSOR_0)', 'g'); @@ -187,48 +196,67 @@ var configuratorApp = (function(){ } }, - handleFileLoad: function(file, $uploader) { - file += ''; + /** + * Handle a file being dropped on the file field + */ + handleFileLoad: function(txt, $uploader) { + txt += ''; var filename = $uploader.val().replace(/.*[\/\\](.*)$/, '$1'); switch(filename) { case boards_file: - this.initBoardsFromText(file); - has_boards = true; + case config_file: + case config_adv_file: + this.fileLoaded(filename, txt); + break; + default: + this.log("Can't parse "+filename, 1); + break; + } + }, + + fileLoaded: function(filename, txt) { + this.log("fileLoaded:"+filename,4); + switch(filename) { + case boards_file: + this.initBoardsFromText(txt); $('#MOTHERBOARD').html('').addOptions(boards_list); if (has_config) this.initField('MOTHERBOARD'); + this.setMessage(boards_file+' loaded successfully.'); break; case config_file: if (has_boards) { - $config.text(file); - has_config = true; - total_config_lines = file.replace(/[^\n]+/g, '').length; - this.initThermistorsFromText(file); + $config.text(txt); + total_config_lines = txt.split(/\r?\n|\r/).length; + this.initThermistorsFromText(txt); this.purgeDefineInfo(false); this.refreshConfigForm(); + this.setMessage(config_file+' loaded successfully.'); + has_config = true; } else { - alert("Upload a " + boards_file + " file first!"); + this.setMessage("Upload a " + boards_file + " file first!", 'error'); } break; case config_adv_file: if (has_config) { - $config_adv.text(file); - has_config_adv = true; - total_config_adv_lines = file.replace(/[^\n]+/g, '').length; + $config_adv.text(txt); + total_config_adv_lines = txt.split(/\r?\n|\r/).length; this.purgeDefineInfo(true); this.refreshConfigForm(); + this.setMessage(config_adv_file+' loaded successfully.'); + has_config_adv = true; } else { - alert("Upload a " + config_file + " file first!"); + this.setMessage("Upload a " + config_file + " file first!", 'error'); } break; - default: - this.log("Can't parse "+filename, 1); - break; } }, - setupConfigForm: function() { + /** + * Add enhancements to the form + */ + initConfigForm: function() { // Modify form fields and make the form responsive. // As values change on the form, we could update the // contents of text areas containing the configs, for @@ -258,14 +286,16 @@ var configuratorApp = (function(){ ); }); + // Add options to the popup menus $('#SERIAL_PORT').addOptions([0,1,2,3,4,5,6,7]); $('#BAUDRATE').addOptions([2400,9600,19200,38400,57600,115200,250000]); $('#EXTRUDERS').addOptions([1,2,3,4]); $('#POWER_SUPPLY').addOptions({'1':'ATX','2':'Xbox 360'}); + // Replace the Serial popup menu with a stepper control $('#serial_stepper').jstepper({ min: 0, - max: 7, + max: 3, val: $('#SERIAL_PORT').val(), arrowWidth: '18px', arrowHeight: '15px', @@ -276,7 +306,6 @@ var configuratorApp = (function(){ textStyle: {width:'1.5em',fontSize:'120%',textAlign:'center'}, onChange: function(v) { $('#SERIAL_PORT').val(v).trigger('change'); } }); - }, refreshConfigForm: function() { @@ -322,30 +351,36 @@ var configuratorApp = (function(){ this.initField('MAX_REDUNDANT_TEMP_SENSOR_DIFF'); this.initField('TEMP_RESIDENCY_TIME'); + }, - // prettyPrint(); + setTextAndHighlight: function($field, txt, name) { + var $elm = $('#'+name), elm = $elm[0], inf = elm.defineInfo; + if (inf == null) return; }, /** - * initField - make a field responsive and get info - * about its configuration file define + * Make a field responsive and initialize its defineInfo */ initField: function(name, adv) { - this.log("initField:"+name,3); + this.log("initField:"+name,4); var $elm = $('#'+name), elm = $elm[0]; - if (elm.defineInfo === undefined) { + if (elm.defineInfo == null) { elm.defineInfo = this.getDefineInfo(name, adv); $elm.on($elm.attr('type') == 'text' ? 'input' : 'change', this.handleChange); } this.setFieldFromDefine(name); }, - handleChange: function(e) { - self.updateDefineFromField(e.target.id); - }, + /** + * Handle any value field being changed + */ + handleChange: function() { self.updateDefineFromField(this.id); }, - handleSwitch: function(e) { - var $elm = $(e.target), $prev = $elm.prev(); + /** + * Handle a switch checkbox being changed + */ + handleSwitch: function() { + var $elm = $(this), $prev = $elm.prev(); var on = $elm.prop('checked') || false; $prev.attr('disabled', !on); self.setDefineEnabled($prev[0].id, on); @@ -357,6 +392,7 @@ var configuratorApp = (function(){ defineValue: function(name) { this.log('defineValue:'+name,4); var $elm = $('#'+name), elm = $elm[0], inf = elm.defineInfo; + if (inf == null) return 'n/a'; var result = inf.regex.exec($(inf.field).text()); this.log(result,2); @@ -370,12 +406,13 @@ var configuratorApp = (function(){ defineIsEnabled: function(name) { this.log('defineIsEnabled:'+name,4); var $elm = $('#'+name), elm = $elm[0], inf = elm.defineInfo; + if (inf == null) return false; var result = inf.regex.exec($(inf.field).text()); this.log(result,2); var on = result !== null ? result[1].trim() != '//' : true; - this.log(name + ' = ' + on, 4); + this.log(name + ' = ' + on, 2); return on; }, @@ -385,23 +422,15 @@ var configuratorApp = (function(){ */ setDefineEnabled: function(name, val) { this.log('setDefineEnabled:'+name,4); - - var $elm = $('#'+name), elm = $elm[0], inf = elm.defineInfo; - var $c = $(inf.field), txt = $c.text(); + var $elm = $('#'+name), inf = $elm[0].defineInfo; + if (inf == null) return; var slash = val ? '' : '//'; var newline = inf.line - .replace(/^([ \t]*)(\/\/)([ \t]*)/, '$1$3') // remove slashes - .replace(inf.pre+inf.define, inf.pre+slash+inf.define); // add them back - - txt = txt.replace(inf.line, newline); - inf.line = newline; - this.log(newline, 2); - - $c.text(txt); + .replace(/^([ \t]*)(\/\/)([ \t]*)/, '$1$3') // remove slashes + .replace(inf.pre+inf.define, inf.pre+slash+inf.define); // add them back - // Scroll to reveal the define - this.scrollToDefine(name); + this.setDefineLine(name, newline); }, /** @@ -409,11 +438,8 @@ var configuratorApp = (function(){ */ updateDefineFromField: function(name) { this.log('updateDefineFromField:'+name,4); - var $elm = $('#'+name), elm = $elm[0], inf = elm.defineInfo; - var $c = $(inf.field), txt = $c.text(); - - // var result = inf.repl.exec(txt); - // this.log(result, 2); + var $elm = $('#'+name), inf = $elm[0].defineInfo; + if (inf == null) return; var isCheck = $elm.attr('type') == 'checkbox', val = isCheck ? $elm.prop('checked') : $elm.val(); @@ -446,18 +472,36 @@ var configuratorApp = (function(){ break; } - txt = txt.replace(inf.line, newline); + this.setDefineLine(name, newline); + }, + + /** + * Set the define's line in the text to a new line, + * then update, highlight, and scroll to the line + */ + setDefineLine: function(name, newline) { + var $elm = $('#'+name), elm = $elm[0], inf = elm.defineInfo; + var $c = $(inf.field), txt = $c.text(); + + var hilite_token = '[HIGHLIGHTER-TOKEN]'; + + txt = txt.replace(inf.line, hilite_token + newline); inf.line = newline; + this.log(newline, 2); - $c.text(txt); + // Convert txt into HTML before storing + var html = $('
').text(txt).html().replace(hilite_token, ''); + + // Set the final text including the highlighter + $c.html(html); // Scroll to reveal the define this.scrollToDefine(name); }, /** - * Scroll the field to show a define + * Scroll a pre box to reveal a #define */ scrollToDefine: function(name, always) { this.log('scrollToDefine:'+name,4); @@ -474,7 +518,6 @@ var configuratorApp = (function(){ if (always == true || Math.abs($c.prop('scrollTop') - textScrollY) > halfHeight) $c.animate({ scrollTop: textScrollY < 0 ? 0 : textScrollY }); - }, /** @@ -587,6 +630,9 @@ var configuratorApp = (function(){ return null; }, + /** + * Count the number of lines before a match, return -1 on fail + */ getLineInText: function(line, txt) { var pos = txt.indexOf(line); return (pos < 0) ? pos : txt.substr(0, pos).replace(/[^\n]+/g, '').length;