Last active
June 11, 2016 21:17
-
-
Save tortuetorche/ffce4d2f3fe216040123 to your computer and use it in GitHub Desktop.
Legacy browsers compatibility (like IE 8 and 9, Android 2.3, Safari 5...) for Turbolinks.replace() ⚠️ Alpha version ⚠️
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
//= require jquery | |
//= require jquery_ujs | |
//= require_tree . | |
//= require turbolinks | |
//= require jquery.turbolinks.replace.shim |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// jQuery Turbolinks Replace Shim | |
// Experimental IE 8 and 9 compatibility for the `replace()` method of Turbolinks 3 | |
// TODO: Port `scroll`?, `cacheRequest`? and `showProgressBar`? options for the `replace()` method | |
// `executeScriptTags()`, `onNodeRemoved()`? methods// WIP: CSRF protection | |
// See: https://github.com/rails/turbolinks/issues/526 | |
(function($) { | |
if (typeof Turbolinks !== 'undefined' && Turbolinks) { | |
if (!Turbolinks.supported) { | |
var CSRFToken = { | |
get: function(doc) { | |
if (!doc) { | |
doc = document; | |
} | |
var tag = $(doc).find('meta[name="csrf-token"]'); | |
if (!tag || tag.length === 0) { | |
tag = $(doc).filter('meta[name="csrf-token"]'); | |
} | |
var param = $(doc).find('meta[name="csrf-param"]'); | |
if (!param || param.length === 0) { | |
param = $(doc).filter('meta[name="csrf-param"]'); | |
} | |
return { | |
node: tag, | |
token: tag.attr('content'), | |
param: param.attr('content') | |
}; | |
}, | |
update: function(latest) { | |
var current = this.get(); | |
// console.log('Update CSRF Token!');//debug | |
if ((current.token) && (latest) && current.token !== latest) { | |
current.node.attr('content', latest); | |
} | |
}, | |
// Making sure that all forms have actual up-to-date token(cached forms contain old one) | |
// TODO: Some Server-side frameworks like Laravel need this. | |
// If your application doesn't use `jquery.turbolinks` library, | |
// You should add this on every `page:load` event. | |
// Even if Turbolinks is supported by the browser. | |
refreshForms: function() { | |
// console.log('Refresh Forms tokens!');//debug | |
var current = this.get(); | |
$('form input[name="' + current.param + '"]').val(current.token); | |
}, | |
// Make sure that every Ajax request sends the CSRF token | |
// TODO: If your application doesn't use jQuery UJS but it send jQuery AJAX requests. | |
// You should add this on every `page:load` event. | |
// Even if Turbolinks is supported by the browser. | |
setupAjaxHeader: function() { | |
$.ajaxPrefilter(function(options, originalOptions, xhr) { | |
if (!options.crossDomain) { | |
var current = CSRFToken.get(); | |
if (current.token) { | |
// console.log('Refresh X-CSRF-Token' + current.token);//debug | |
xhr.setRequestHeader('X-CSRF-Token', current.token); | |
} | |
} | |
}); | |
} | |
}; | |
Turbolinks.replaceContent = function(content, options) { | |
if (typeof content === 'object' && typeof options === 'object') { | |
// We can't add a `body` tag, it seems buggy with IE 8 and 9 | |
var body = $('<div>' + content.body + '</div>'); | |
var currentBody = $('body'); | |
if (content.title) { | |
document.title = content.title; | |
} | |
if (options.change) { | |
var nodesToChange = $('[data-turbolinks-temporary]', 'body').toArray(); | |
var matchingNodes = []; | |
var keys = $.makeArray(options.change); | |
$.each(keys, function(i, key) { | |
$.merge(matchingNodes, $('[id^="' + key + ':"], #' + key, 'body').toArray()); | |
}); | |
$.merge(nodesToChange, matchingNodes); | |
var changedNodes = []; | |
$.each(nodesToChange, function(i, existingNode) { | |
var nodeId = existingNode.getAttribute('id'); | |
if (!nodeId) { | |
throw new Error('Turbolinks partial replace: turbolinks elements must have an id.'); | |
} | |
var targetNode = $('#' + nodeId, body); | |
if (targetNode.length) { | |
targetNode = targetNode.get(0).cloneNode(true); | |
$(existingNode).replaceWith(targetNode); | |
changedNodes.push(targetNode); | |
} | |
}); | |
$(changedNodes).trigger('page:load'); | |
} else if (options.flush) { | |
var targetNode = body.html(); | |
currentBody.empty().append(targetNode); | |
currentBody = $('body'); | |
if (content.csrfToken) { | |
CSRFToken.update(content.csrfToken); | |
CSRFToken.setupAjaxHeader(); | |
} | |
currentBody.trigger('page:load'); | |
} else { | |
var nodesToKeep = $('[data-turbolinks-permanent]', 'body').toArray(); | |
if (options.keep) { | |
var matchingNodes = []; | |
var keys = $.makeArray(options.keep); | |
$.each(keys, function(i, key) { | |
$.merge(matchingNodes, $('[id^="' + key + ':"], #' + key, 'body').toArray()); | |
}); | |
$.merge(nodesToKeep, matchingNodes); | |
} | |
$.each(nodesToKeep, function(i, existingNode) { | |
var nodeId = existingNode.getAttribute('id'); | |
if (!nodeId) { | |
throw new Error('Turbolinks partial replace: turbolinks elements must have an id.'); | |
} | |
var targetNode = $('#' + nodeId, body); | |
if (targetNode.length) { | |
$(targetNode).replaceWith(existingNode.cloneNode(true)); | |
} | |
}); | |
currentBody.empty().append(body.html()); | |
currentBody = $('body'); | |
if (content.csrfToken) { | |
CSRFToken.update(content.csrfToken); | |
CSRFToken.setupAjaxHeader(); | |
} | |
currentBody.trigger('page:load'); | |
} | |
CSRFToken.refreshForms(); | |
return true;// TODO: Reflect the Turbolinks.defaultReplace() return behaviour | |
} else { | |
throw new Error('Invalid arguments'); | |
// Fallback to default Turbolinks.replace() method? | |
// return Turbolinks.defaultReplace(html, options); | |
} | |
}; | |
Turbolinks.defaultReplace = Turbolinks.replace; | |
Turbolinks.replace = function(html, options) { | |
if (typeof options === 'undefined' || options === null) { | |
options = {}; | |
} | |
var $html = $(html); | |
var csrfTokenContent = CSRFToken.get(html).token; | |
var titleContent = null; | |
if (options.title !== false) { | |
var $title = $html.filter('title'); | |
if (options.title) { | |
titleContent = options.title; | |
} else if ($title.length) { | |
titleContent = $('<div></div>').text($title.html()).html(); | |
} else { | |
// For IE 8 | |
var result = html.match(/<title>([^<]+)<\/title>/); | |
if (result) { | |
titleContent = $('<div></div>').text(result[1]).html(); | |
} | |
} | |
} | |
var bodyContent = html.replace(/^(.|[\n\r])*<body[^>]*>((.|[\n\r])*)<\/body>(.|[\n\r])*$/im, '$2'); | |
var content = {title: titleContent, body: bodyContent, csrfToken: csrfTokenContent}; | |
return Turbolinks.replaceContent(content, options); | |
}; | |
} | |
} | |
})(jQuery); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment