Solving simple problems with client-side web applications

Whether we like it or not, JavaScript has been exponentially growing in popularity. While the benefits can be obvious, there as some side-effects that are usually overlooked. As more and more features are being pushed to modern web browsers (making it compelling to build web applications that work in almost every device), on the other hand feature fragmentation is getting worst every day. We can point fingers at all the different browser vendors and their constant push of new updates. To make matters worse, the same browser might have different versions for Desktop and Mobile devices (Chrome != Chrome for Android != Android Browser). Some of this problems could be avoided if everyone always ran the latest versions (and Google Chrome started off with great ideas to make that possible) but the reality is very different. Take a look at can I use to have a rough idea of feature reach and disparity. At a higher level, consider how many people still use Windows 7 (and uses IE)... Does everyone upgrade to the latest iPhone or Samsung Galaxy as soon as they come out? Most Android phones aren't even compatible with the latest Android version. If fragmentation is gigantic in the most used mobile OS, imagine how big that is problem if you account for all available web browsers. To be fair fragmentation is an issue if your audience uses a diverse variety of browsers and you are using brand new browser features. Both problems are easily identifiable, fixing them might not be so easy. Anyway, just be aware of this issues when you're working on web applications. Now that we are done the warnings, let's jump into a practical example.

Imagine a family member just asked your help to solve simple problem. Analyze each row of a given csv file (comma-separated values. Looks like a spreadsheet separated by commas and line breaks, very simple way to structure data), to verify if the element on the last column (let's call it the 'pivot'): is smaller than all elements in the first six columns and (if previous condition is meet) add a new column at the end of the row, containing the difference between the 'pivot' and the value presented on the sixth column. Quite simple, right? Anybody could do it by hand, but we are developers so let's make the machine do the hard work for us.

For this particular case we could develop a Desktop or Mobile application, but let's try to create a Web Application. Plus this problem is so simple, that server processing won't be required at all, let's do anything in client-side JavaScript. In other words, our friend's web browser will do all the work, we just provide the set of instructions it needs to run.

This problem is quite simple to solve, even so let's break it down to a list of straight forward requirements:

  • the user needs to upload a csv
  • instead of a file upload dialog, drag-drop is preferred
  • figure out a way to read the contents of the file provided by the user
  • analyze the csv file
  • return the resulting csv file to the user (user must download it)
  • it must run on the latest Google Chrome (for Desktop), don't worry about any other browser
  • all code must run on the client browser (no file uploads to our web server)

Due to the simplicity of this problem, let's jump straight into code. Just check the comments for more details. Let's start with the HTML:

<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta name="description" content="Simple Match prem analysis">
<meta name="author" content="John Louros">
<title>Basic csv analysis</title>

<style type="text/css">
  body { font: normal 16px/20px "Helvetica Neue", Helvetica, sans-serif; background: rgb(237, 237, 236); margin: 0; margin-top: 40px; padding: 0; }
  section, header, footer { display: block; }
  header, article > * { margin: 20px; }
  h1 { padding-top: 10px; }
  [contenteditable]:hover:not(:focus) { outline: 1px dotted #ccc; }

  #wrapper { width: 600px; margin: 0 auto; background: #fff url('') repeat-x center bottom; border-radius: 10px; border-top: 1px solid #fff; padding-bottom: 76px; }
  #holder { border: 10px dashed #ccc; width: 300px; min-height: 300px; margin: 20px auto; }
  #holder.hover { border: 10px dashed #0c0; }
  #saveOutputFileBtn { margin: 0 auto; display: block; }
  .errorMessage { color: #c00 }
  .fail { background: #c00; padding: 2px; color: #fff; }
  .hidden { display: none !important; }
</style>


<script src="http://code.jquery.com/jquery-2.2.4.min.js"></script>
<!-- source code for jQuery CSV can be found here https://github.com/evanplaice/jquery-csv -->
<script src="https://cdn.rawgit.com/evanplaice/jquery-csv/c99aa27290e103cc4cf76136c86f60d4985dd0d6/src/jquery.csv.min.js"></script>
<script src="js/main.js"></script>
</head>
<body>

<section id="wrapper">
  <header>
  <h1>Drag and drop, automatic upload</h1>
  </header>
  <article>
  <div id="holder"></div>

  <p id="filereader">
    <span class="errorMessage">File API &amp; FileReader API not supported</span>
  </p>

  <p id="statusMessage"></p>    
  </article>
  <article>
  <button id="saveOutputFileBtn">save result</button>
  </article>
</section>

</body>
</html> 

Finally the JavaScript:

$(document).ready(function () {

  var processedData;
  var outputFileName;

  // initial state (right after the page loads)
  $("#statusMessage").text("Drag an csv file from your desktop on to the drop zone above to kick-off the analysis.");
  $("#saveOutputFileBtn").attr("disabled", "disabled");

  // check if this browser supports 'FileReader'. Required to read csv contents
  if((typeof FileReader != 'undefined') === false) {
    $("#filereader").addClass("fail");
  } else {
    $("#filereader").addClass("hidden");
  }

  // check if this browser support drag & drop and wire it up
  if('draggable' in document.createElement('span')) {
    var holder = document.getElementById('holder');
    holder.ondragover = function () { this.className = 'hover'; return false; };
    holder.ondragend = function () { this.className = ''; return false; };
    holder.ondrop = function (e) {
      this.className = '';
      e.preventDefault();
      readfiles(e.dataTransfer.files);
    }
  }

  // read uploaded files
  function readfiles(files) {

    if (files.length === 0) {
      return;
    }

    // set output file name
    outputFileName = files[0].name.replace(".", "_result.");

    var reader = new FileReader();

    reader.onloadend = function (evt) {
      if (evt.target.readyState == FileReader.DONE) { // DONE == 2
        var csvData = evt.target.result;

        try {
          runCsvAnalysis(csvData);
        } catch(err) {
          $("#statusMessage").text("Unsupported file or unexcepted error processing input. Try another file.");  
        }
      }
    };

    reader.readAsText(files[0]);

    $("#saveOutputFileBtn").text("save result '" + outputFileName + "'");
    $("#statusMessage").text("File upload successful!");
  }

  function runCsvAnalysis(csvContent) {
    // parse csv content to array 
    var dataArr = $.csv.toArrays(csvContent);

    // create a copy of the array
    var output = dataArr.slice();

    // check if the first row container header or values
    var firstRow = isNaN(parseInt(dataArr[0][0])) ? 1 : 0;

    if (firstRow > 0) {
      // add new column name
      output[0][12] = "Result";

    }

    // crappy algorithm, since it does not run any validation and 
    // expects the values in a particular order. For demo proposes only.
    for (firstRow, len = dataArr.length; idx < len; idx++) {
      var isMatch = false;
      var result = " ";
      var row = dataArr[idx];
      var pivot = parseInt(row[11]);

      if (pivot < parseInt(row[0])
        && pivot < parseInt(row[1])
        && pivot < parseInt(row[2])
        && pivot < parseInt(row[3])
        && pivot < parseInt(row[4])
        && pivot < parseInt(row[5])
      ) {
        isMatch = true;
        result = row[5] - pivot;
      }

      output[idx][12] = result;
    }

    // save output to processedData
    processedData = output;

    $("#statusMessage").text("Analysis complete, press the save button below to download the result.");
    $("#saveOutputFileBtn").removeAttr("disabled");
  }

  $("#saveOutputFileBtn").bind("click", function () {
    // prepare results to be downloaded
    var csvContent = "data:text/csv;charset=utf-8,";
    processedData.forEach(function (infoArray, index) {

      dataString = infoArray.join(",");
      csvContent += index < processedData.length ? dataString + "\n" : dataString;

    });

    // create a anchor element to automatically download the results file
    var encodedUri = encodeURI(csvContent);
    var link = document.createElement("a");
    link.setAttribute("href", encodedUri);
    link.setAttribute("download", outputFileName);

    link.click(); // This will trigger file download
  });

}); 

To exercise the code use the following test csv:

Column A,Column B,Column C,Column D,Column E,Column F,Column G,Column H,Column I,Column J,Column K,Column L
49,66,92,86,80,56,100,48,65,46,58,55
96,75,35,93,92,21,35,81,80,30,66,43
92,75,79,54,97,9,85,87,41,75,55,82
10,48,80,45,32,34,72,15,97,0,44,23
8,67,14,34,85,16,33,73,22,64,79,46
67,70,89,66,91,65,33,96,59,61,64,1
25,92,27,6,30,78,33,10,89,91,10,92
42,63,44,53,59,33,61,41,82,66,100,79
45,49,23,24,74,94,38,53,6,31,76,17 

Quite simple right? Keep in mind that this code was only tested in Chrome. This is a very basic example but you get the idea. Hopefully this will serve as inspiration for someone. Please send any questions or comments that you might have.

web services