Browse Source

gitweb.js: use setTimeout rather than setInterval in blame_incremental.js

If there is a possibility that your logic could take longer to execute
than the interval time, it is recommended that you recursively call a
named function using window.setTimeout rather than window.setInterval.

Therefore instead of using setInterval as an alternate way of invoking
handleResponse (because some web browsers call onreadystatechange only
once per each distinct state, and not for each server flush), use
setTimeout and reset it from handleResponse.  As a bonus this allows
us to get rid of timer if it turns out that web browser calls
onreadystatechange on each server flush.

While at it get rid of `xhr' global variable, creating it instead as
local variable in startBlame and passing it as parameter, and of
`pollTimer' global variable, passing it as member of xhr object
(xhr.pollTimer).

Signed-off-by: Jakub Narebski <jnareb@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
maint
Jakub Narebski 14 years ago committed by Junio C Hamano
parent
commit
42ab5d40de
  1. 55
      gitweb/static/js/blame_incremental.js

55
gitweb/static/js/blame_incremental.js

@ -29,7 +29,6 @@ @@ -29,7 +29,6 @@
/* ............................................................ */
/* utility/helper functions (and variables) */

var xhr; // XMLHttpRequest object
var projectUrl; // partial query + separator ('?' or ';')

// 'commits' is an associative map. It maps SHA1s to Commit objects.
@ -431,8 +430,6 @@ var endRe = /^END ?([^ ]*) ?(.*)/; @@ -431,8 +430,6 @@ var endRe = /^END ?([^ ]*) ?(.*)/;
var curCommit = new Commit();
var curGroup = {};

var pollTimer = null;

/**
* Parse output from 'git blame --incremental [...]', received via
* XMLHttpRequest from server (blamedataUrl), and call handleLine
@ -533,26 +530,34 @@ function processData(unprocessed, nextReadPos) { @@ -533,26 +530,34 @@ function processData(unprocessed, nextReadPos) {
* Handle XMLHttpRequest errors
*
* @param {XMLHttpRequest} xhr: XMLHttpRequest object
* @param {Number} [xhr.pollTimer] ID of the timeout to clear
*
* @globals pollTimer, commits
* @globals commits
*/
function handleError(xhr) {
errorInfo('Server error: ' +
xhr.status + ' - ' + (xhr.statusText || 'Error contacting server'));

clearInterval(pollTimer);
if (typeof xhr.pollTimer === "number") {
clearTimeout(xhr.pollTimer);
delete xhr.pollTimer;
}
commits = {}; // free memory
}

/**
* Called after XMLHttpRequest finishes (loads)
*
* @param {XMLHttpRequest} xhr: XMLHttpRequest object (unused)
* @param {XMLHttpRequest} xhr: XMLHttpRequest object
* @param {Number} [xhr.pollTimer] ID of the timeout to clear
*
* @globals pollTimer, commits
* @globals commits
*/
function responseLoaded(xhr) {
clearInterval(pollTimer);
if (typeof xhr.pollTimer === "number") {
clearTimeout(xhr.pollTimer);
delete xhr.pollTimer;
}

fixColorsAndGroups();
writeTimeInterval();
@ -563,9 +568,13 @@ function responseLoaded(xhr) { @@ -563,9 +568,13 @@ function responseLoaded(xhr) {
* handler for XMLHttpRequest onreadystatechange event
* @see startBlame
*
* @globals xhr
* @param {XMLHttpRequest} xhr: XMLHttpRequest object
* @param {Number} xhr.prevDataLength: previous value of xhr.responseText.length
* @param {Number} xhr.nextReadPos: start of unread part of xhr.responseText
* @param {Number} [xhr.pollTimer] ID of the timeout (to reset or cancel)
* @param {Boolean} fromTimer: if handler was called from timer
*/
function handleResponse() {
function handleResponse(xhr, fromTimer) {

/*
* xhr.readyState
@ -614,6 +623,19 @@ function handleResponse() { @@ -614,6 +623,19 @@ function handleResponse() {
// did we finish work?
if (xhr.readyState === 4) {
responseLoaded(xhr);
return;
}

// if we get from timer, we have to restart it
// otherwise onreadystatechange gives us partial response, timer not needed
if (fromTimer) {
setTimeout(function () {
handleResponse(xhr, true);
}, 1000);

} else if (typeof xhr.pollTimer === "number") {
clearTimeout(xhr.pollTimer);
delete xhr.pollTimer;
}
}

@ -629,11 +651,11 @@ function handleResponse() { @@ -629,11 +651,11 @@ function handleResponse() {
* Called from 'blame_incremental' view after loading table with
* file contents, a base for blame view.
*
* @globals xhr, t0, projectUrl, div_progress_bar, totalLines, pollTimer
* @globals t0, projectUrl, div_progress_bar, totalLines
*/
function startBlame(blamedataUrl, bUrl) {

xhr = createRequestObject();
var xhr = createRequestObject();
if (!xhr) {
errorInfo('ERROR: XMLHttpRequest not supported');
return;
@ -652,8 +674,9 @@ function startBlame(blamedataUrl, bUrl) { @@ -652,8 +674,9 @@ function startBlame(blamedataUrl, bUrl) {
xhr.prevDataLength = -1; // used to detect if we have new data
xhr.nextReadPos = 0; // where unread part of response starts

xhr.onreadystatechange = handleResponse;
//xhr.onreadystatechange = function () { handleResponse(xhr); };
xhr.onreadystatechange = function () {
handleResponse(xhr, false);
};

xhr.open('GET', blamedataUrl);
xhr.setRequestHeader('Accept', 'text/plain');
@ -661,7 +684,9 @@ function startBlame(blamedataUrl, bUrl) { @@ -661,7 +684,9 @@ function startBlame(blamedataUrl, bUrl) {

// not all browsers call onreadystatechange event on each server flush
// poll response using timer every second to handle this issue
pollTimer = setInterval(xhr.onreadystatechange, 1000);
xhr.pollTimer = setTimeout(function () {
handleResponse(xhr, true);
}, 1000);
}

/* end of blame_incremental.js */

Loading…
Cancel
Save