You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
nodebb/public/src/ajaxify.js

341 lines
9.0 KiB
JavaScript

"use strict";
var ajaxify = ajaxify || {};
(function () {
/*global app, templates, utils, socket, translator, config, RELATIVE_PATH*/
12 years ago
var location = document.location || window.location,
rootUrl = location.protocol + '//' + (location.hostname || location.host) + (location.port ? ':' + location.port : ''),
templatesConfig = null,
availableTemplates = null,
apiXHR = null,
PRELOADER_RATE_LIMIT = 10000;
window.onpopstate = function (event) {
if (event !== null && event.state && event.state.url !== undefined && !ajaxify.initialLoad) {
11 years ago
ajaxify.go(event.state.url, function() {
$(window).trigger('action:popstate', {url: event.state.url});
}, true);
}
};
ajaxify.currentPage = null;
ajaxify.initialLoad = false;
ajaxify.preloader = {};
11 years ago
function onAjaxError(err, url) {
11 years ago
var data = err.data, textStatus = err.textStatus;
11 years ago
$('#content, #footer').removeClass('ajaxifying');
11 years ago
if (data) {
if (data.status === 404) {
return ajaxify.go('404');
} else if (data.status === 403) {
app.alertError('[[global:please_log_in]]');
app.previousUrl = url;
return ajaxify.go('login');
} else if (data.status === 302) {
return ajaxify.go(data.responseJSON.slice(1));
11 years ago
}
11 years ago
} else if (textStatus !== "abort") {
app.alertError(data.responseJSON.error);
}
}
ajaxify.go = function (url, callback, quiet) {
// "quiet": If set to true, will not call pushState
app.enterRoom('global');
$(window).off('scroll');
$(window).trigger('action:ajaxify.start', {url: url});
11 years ago
if ($('#content').hasClass('ajaxifying') && apiXHR) {
apiXHR.abort();
}
// Remove trailing slash
url = url.replace(/\/$/, "");
url = ajaxify.removeRelativePath(url);
var tpl_url = ajaxify.getTemplateMapping(url);
var hash = '';
if(ajaxify.initialLoad) {
hash = window.location.hash ? window.location.hash : '';
}
11 years ago
if (ajaxify.isTemplateAvailable(tpl_url) && !!!templatesConfig.force_refresh[tpl_url]) {
11 years ago
ajaxify.currentPage = url;
if (window.history && window.history.pushState) {
window.history[!quiet ? 'pushState' : 'replaceState']({
url: url + hash
}, url, RELATIVE_PATH + '/' + url + hash);
}
translator.load(config.defaultLang, tpl_url);
11 years ago
$('#footer, #content').removeClass('hide').addClass('ajaxifying');
var animationDuration = parseFloat($('#content').css('transition-duration')) || 0.2,
startTime = (new Date()).getTime();
ajaxify.variables.flush();
ajaxify.loadData(url, function(err, data) {
11 years ago
if (err) {
return onAjaxError(err, url);
}
12 years ago
11 years ago
$(window).trigger('action:ajaxify.loadingTemplates', {});
templates.parse(tpl_url, data, function(template) {
translator.translate(template, function(translatedTemplate) {
setTimeout(function() {
$('#content').html(translatedTemplate);
ajaxify.widgets.render(tpl_url, url, function() {
$(window).trigger('action:ajaxify.end', {url: url});
});
11 years ago
ajaxify.variables.parse();
$(window).trigger('action:ajaxify.contentLoaded', {url: url});
ajaxify.loadScript(tpl_url);
if (typeof callback === 'function') {
callback();
}
app.processPage();
$('#content, #footer').removeClass('ajaxifying');
ajaxify.initialLoad = false;
app.refreshTitle(url);
}, animationDuration * 1000 - ((new Date()).getTime() - startTime))
11 years ago
11 years ago
});
});
});
return true;
}
return false;
};
ajaxify.removeRelativePath = function(url) {
if (url.indexOf(RELATIVE_PATH.slice(1)) === 0) {
url = url.slice(RELATIVE_PATH.length);
}
return url;
};
ajaxify.refresh = function() {
ajaxify.go(ajaxify.currentPage);
};
ajaxify.loadScript = function(tpl_url, callback) {
require(['forum/' + tpl_url], function(script) {
if (script && script.init) {
script.init();
}
if (callback) {
callback();
}
});
};
ajaxify.isTemplateAvailable = function(tpl) {
return $.inArray(tpl + '.tpl', availableTemplates) !== -1;
};
ajaxify.getTemplateMapping = function(url) {
var tpl_url = ajaxify.getCustomTemplateMapping(url.split('?')[0]);
if (tpl_url === false && !templates[url]) {
11 years ago
tpl_url = url.split('/');
11 years ago
while(tpl_url.length) {
if (ajaxify.isTemplateAvailable(tpl_url.join('/'))) {
tpl_url = tpl_url.join('/');
break;
}
11 years ago
tpl_url.pop();
}
if (!tpl_url.length) {
tpl_url = url.split('/')[0].split('?')[0];
}
} else if (templates[url]) {
tpl_url = url;
}
return tpl_url;
};
ajaxify.getCustomTemplateMapping = function(tpl) {
11 years ago
if (templatesConfig && templatesConfig.custom_mapping && tpl !== undefined) {
11 years ago
for (var pattern in templatesConfig.custom_mapping) {
if (tpl.match(pattern)) {
11 years ago
return (templatesConfig.custom_mapping[pattern]);
}
}
}
return false;
};
ajaxify.loadData = function(url, callback) {
url = ajaxify.removeRelativePath(url);
$(window).trigger('action:ajaxify.loadingData', {url: url});
11 years ago
if (ajaxify.preloader && ajaxify.preloader[url] && !ajaxify.preloader[url].loading) {
callback(null, ajaxify.preloader[url].data);
11 years ago
ajaxify.preloader = {};
return;
11 years ago
}
var location = document.location || window.location,
api_url = (url === '' || url === '/') ? 'home' : url,
11 years ago
tpl_url = ajaxify.getCustomTemplateMapping(url.split('?')[0]);
if (!tpl_url) {
11 years ago
tpl_url = ajaxify.getTemplateMapping(url);
}
apiXHR = $.ajax({
url: RELATIVE_PATH + '/api/' + api_url,
cache: false,
success: function(data) {
if (!data) {
ajaxify.go('404');
return;
}
data.relative_path = RELATIVE_PATH;
11 years ago
if (callback) {
callback(null, data);
}
},
error: function(data, textStatus) {
11 years ago
callback({
data: data,
textStatus: textStatus
});
}
});
};
ajaxify.loadTemplate = function(template, callback) {
if (templates.cache[template]) {
callback(templates.cache[template]);
} else {
$.ajax({
url: RELATIVE_PATH + '/templates/' + template + '.tpl' + (config['cache-buster'] ? '?v=' + config['cache-buster'] : ''),
type: 'GET',
success: function(data) {
callback(data.toString());
},
error: function(error) {
throw new Error("Unable to load template: " + template + " (" + error.statusText + ")");
}
});
}
};
$('document').ready(function () {
if (!window.history || !window.history.pushState) {
return; // no ajaxification for old browsers
}
11 years ago
function hrefEmpty(href) {
11 years ago
return href === undefined || href === '' || href === 'javascript:;' || href === window.location.href + "#" || href.slice(0, 1) === "#";
11 years ago
}
// Enhancing all anchors to ajaxify...
$(document.body).on('click', 'a', function (e) {
11 years ago
if (hrefEmpty(this.href) || this.target !== '' || this.protocol === 'javascript:' || $(this).attr('data-ajaxify') === 'false') {
12 years ago
return;
}
12 years ago
if(!window.location.pathname.match(/\/(403|404)$/g)) {
app.previousUrl = window.location.href;
}
11 years ago
if ((!e.ctrlKey && !e.shiftKey && !e.metaKey) && e.which === 1) {
if (this.host === window.location.host) {
// Internal link
var url = this.href.replace(rootUrl + '/', '');
11 years ago
if(window.location.pathname === this.pathname && this.hash) {
if (this.hash !== window.location.hash) {
window.location.hash = this.hash;
}
ajaxify.loadScript(ajaxify.getTemplateMapping(url));
e.preventDefault();
11 years ago
} else {
if (ajaxify.go(url)) {
e.preventDefault();
}
}
} else if (window.location.pathname !== '/outgoing') {
// External Link
11 years ago
if (config.openOutgoingLinksInNewTab) {
11 years ago
window.open(this.href, '_blank');
e.preventDefault();
11 years ago
} else if (config.useOutgoingLinksPage) {
ajaxify.go('outgoing?url=' + encodeURIComponent(this.href));
e.preventDefault();
12 years ago
}
}
}
});
11 years ago
$(document.body).on('mouseover', 'a', function (e) {
if (hrefEmpty(this.href) || this.target !== '' || this.protocol === 'javascript:' || $(this).attr('data-ajaxify') === 'false') {
return;
}
if (this.host === window.location.host) {
// Internal link
var url = this.href.replace(rootUrl + '/', ''),
currentTime = (new Date()).getTime();
if (!ajaxify.preloader[url] || (!ajaxify.preloader[url].loading && currentTime - ajaxify.preloader[url].lastFetched > PRELOADER_RATE_LIMIT)) {
ajaxify.preloader[url] = {
loading: true
};
ajaxify.loadData(url, function(err, data) {
11 years ago
ajaxify.preloader[url] = err ? null : {
url: url,
data: data,
lastFetched: currentTime,
loading: false
};
});
}
11 years ago
}
});
templates.registerLoader(ajaxify.loadTemplate);
$.getJSON(RELATIVE_PATH + '/api/get_templates_listing', function (data) {
templatesConfig = data.templatesConfig;
availableTemplates = data.availableTemplates;
app.load();
});
});
}());