-
Dec 11, 2014
deWordify Updated
I just updated deWordify, a tool to convert Microsoft Word files to HTML. Now, you can optionally remove any unnecessary non-breaking spaces or line breaks formatted as paragraphs that are sometimes in Word files.
I made deWordify to help me format the Microsoft Word writing submissions that end up on the online literary magazine I maintain. I like that process improvement doesn’t really feel like work, but it’s probably some of the most important work.
-
Jul 17, 2014
Machine Learning
I started taking a Machine Learning course through Coursera. Finished the first programming assignment last night, which was to implement gradient descent for linear regression with one variable in Octave (or MATLAB). I’m not sure I’ll ever get to use ML at work, but it’s interesting as heck.
-
Jul 16, 2014
DataTables TableTools Extension with AngularJS/RequireJS
We use a flexible jQuery plug-in called DataTables at my job to handle generating large display tables. One often-requested feature was the ability to download the table as a CSV file. DataTables has an extension called TableTools that I used to implement this feature. TableTools lets you copy the table to the clipboard, export to CSV, XLS, and PDF, and also gives a clean, large view suitable for printing. Getting TableTools to work w/ AngularJS was a bit challenging.
// Define as an AMD module if possible if ( typeof define === 'function' && define.amd ) { define( ['jquery', 'datatables'], factory ); } else if ( typeof exports === 'object' ) { // Node/CommonJS factory( require('jquery'), require('datatables') ); } else if ( jQuery && !jQuery.fn.dataTable.TableTools ) { // Otherwise simply initialise as normal, stopping multiple evaluation factory( jQuery, jQuery.fn.dataTable ); }
^ This logic at the very end of
dataTables.tableTools.js
was conflicting with RequireJS. Specifically,require.js
was properly loading the DataTables file (jquery.dataTables.js
), but the conditional block here was trying to loaddataTables.js
. A 404 error was being thrown. I only needed the very last branch of the if statement:factory( jQuery, jQuery.fn.dataTable);
You can check if TableTools (or another extension) is being properly loaded by using the DataTables Debug Bookmarklet. The debug bookmarklet shows you what extensions are correctly installed:
download it from here. Once TableTools is installed, you should be able to use it with code like this:
# Set up the TableTools DataTables extension for CSV exporting _createTableTools = (myDataTable) -> tt = new $.fn.DataTable.TableTools( myDataTable, sSwfPath: "/path/to/TableTools/swf/copy_csv_xls.swf" aButtons: [ "csv" ] ) $(tt.fnContainer()).insertBefore('div.dataTables_wrapper');
That’ll place a CSV download button right above an existing DataTable. I call this function after I create a table, passing in the DataTable instance as a parameter. Note that
/path/to/TableTools/swf/copy_csv_xls.swf
is from the site root, not the machine’s filesystem root. Note also thatdiv.dataTables_wrapper
is created by default when the DataTable is made–you don’t have to create that element yourself. You can also create the TableTools object as a property of the DataTable. Something like this should work:table = elem.find('table').dataTable( sDom: 'T<"clear">lfrtip' tableTools: sSwfPath: '/path/to/TableTools/swf/copy_csv_xls.swf' # Other table properties # ... )
Repositioning the buttons anywhere besides on the DataTable is…well…annoying. I think you have to first initialize the TableTools DOM element on the DataTable, and then remove it from its parent node and append it to another node. Even if you get the DOM element in the correct place, I found there were a few CSS gotchas. I won’t write about them here, as the changes a coworker and I made were specific to our layout. I had to change some of the default styles created within the
ZeroClipboard_TableTools.Client.prototype
section ofdataTables.tableTools.js
. If the buttons appear but aren’t working, ensure that the path to the SWF file is correct and that the actual SWF element in the DOM is in the correct space. When I moved the buttons from their default position, I found the SWF element was being absolutely positioned, which in this particular layout made it sit invisibly in the top left-hand corner of the browser window. When you’re done though, you’ll have export functionality sitting right alongside the DataTable. -
Jun 17, 2014
Weird ellipsis in AngularJS…
Today at work I came across a strange issue.
I had written a function that would take some raw numeric data, then format it to be displayed. For instance, ‘1.234567’ might become ‘$1.23’.
In Angular I would normally just do plain old data binding like this:
$scope.adjustedValue = '$1.23'
<div></div>
That was working fine for most cases, but some of our values required special formatting:
$scope.adjustedValue = '<span class="prefix">$</span>1.23'
When I used
ng-bind-html
to let me bind a variable containing HTML, however, an ellipsis was being appended to the end of my number.<div ng-bind-html="adjustedValue"></div>
Instead of ‘$1.23’ as I expected, I was seeing ‘$1.23…’.
I thought there was maybe some CSS being applied that was inserting that ellipsis, but when I disabled CSS the rogue ‘…’ still appeared.
Binding with double curly braces behaved as expected, but using
ng-bind-html
did not. Weird.According to the docs for
ngBindHtml
: “By default, the innerHTML-ed content will be sanitized using the $sanitize service.”I tried bypassing the
$sanitize
step to see if that would solve this ellipsis problem (I also had to inject$sce
into my controller):$scope.adjustedValue = '<span class="prefix">$</span>1.23' $scope.bypassSanitize = -> $sce.trustAsHtml($scope.adjustedValue)
<div ng-bind-html="bypassSanitize()"></div>
That resolved the problem; the output no longer contained an ellipsis. Always kind of unsettling when you’re not exactly sure why something works…
The
$sanitize
method will remove any unclean HTML from your bound variable, so forgo it with that caveat. Indeed, when I set$scope.adjustedValue = <script type="text/javascript">alert('hello');</script>
it was clear that arbitrary HTML was being bound. In this particular case our backend is scrubbing the possible output values, so I was able to move ahead using this solution. -
Jun 12, 2014
122
Spent ~3 hours today writing Jasmine unit tests for a new AngularJS controller I made yesterday.
Recent Posts
- Vibecoding’s allure: the inventor and the fiend
- Philadelphia Data Explorer
- Plant identification
- Let’s Flip an Unfair Coin
- Flight Focus
- Generative AI Accelerates Our Exploration of the Search Space
- AI Assistant Use-Case: Performance Feedback
- Poor Man's Computer Use with Execute Arbitrary AppleScript MCP Server
- Please don’t disable paste
- Blogging via Email