mercurial.js
513 lines
| 13.8 KiB
| application/javascript
|
JavascriptLexer
Benoit Allard
|
r14046 | // mercurial.js - JavaScript utility functions | ||
// | ||||
// Rendering of branch DAGs on the client side | ||||
// Display of elapsed time | ||||
Steven Brown
|
r14571 | // Show or hide diffstat | ||
Benoit Allard
|
r14046 | // | ||
// Copyright 2008 Dirkjan Ochtman <dirkjan AT ochtman DOT nl> | ||||
// Copyright 2006 Alexander Schremmer <alex AT alexanderweb DOT de> | ||||
// | ||||
// derived from code written by Scott James Remnant <scott@ubuntu.com> | ||||
// Copyright 2005 Canonical Ltd. | ||||
// | ||||
// This software may be used and distributed according to the terms | ||||
// of the GNU General Public License, incorporated herein by reference. | ||||
var colors = [ | ||||
[ 1.0, 0.0, 0.0 ], | ||||
[ 1.0, 1.0, 0.0 ], | ||||
[ 0.0, 1.0, 0.0 ], | ||||
[ 0.0, 1.0, 1.0 ], | ||||
[ 0.0, 0.0, 1.0 ], | ||||
[ 1.0, 0.0, 1.0 ] | ||||
]; | ||||
function Graph() { | ||||
Kevin Bullock
|
r19530 | |||
Benoit Allard
|
r14046 | this.canvas = document.getElementById('graph'); | ||
Matt Mackall
|
r17694 | if (window.G_vmlCanvasManager) this.canvas = window.G_vmlCanvasManager.initElement(this.canvas); | ||
Benoit Allard
|
r14046 | this.ctx = this.canvas.getContext('2d'); | ||
this.ctx.strokeStyle = 'rgb(0, 0, 0)'; | ||||
this.ctx.fillStyle = 'rgb(0, 0, 0)'; | ||||
this.bg = [0, 4]; | ||||
this.cell = [2, 0]; | ||||
this.columns = 0; | ||||
Kevin Bullock
|
r19530 | |||
r35255 | } | |||
Graph.prototype = { | ||||
reset: function() { | ||||
Alexander Plavin
|
r19780 | this.bg = [0, 4]; | ||
this.cell = [2, 0]; | ||||
this.columns = 0; | ||||
document.getElementById('nodebgs').innerHTML = ''; | ||||
r35255 | }, | |||
Alexander Plavin
|
r19780 | |||
r35255 | scale: function(height) { | |||
Benoit Allard
|
r14046 | this.bg_height = height; | ||
this.box_size = Math.floor(this.bg_height / 1.2); | ||||
this.cell_height = this.box_size; | ||||
r35255 | }, | |||
Kevin Bullock
|
r19530 | |||
r35255 | setColor: function(color, bg, fg) { | |||
Kevin Bullock
|
r19530 | |||
Benoit Allard
|
r14046 | // Set the colour. | ||
// | ||||
Patrick Mezard
|
r16138 | // If color is a string, expect an hexadecimal RGB | ||
// value and apply it unchanged. If color is a number, | ||||
// pick a distinct colour based on an internal wheel; | ||||
// the bg parameter provides the value that should be | ||||
// assigned to the 'zero' colours and the fg parameter | ||||
// provides the multiplier that should be applied to | ||||
// the foreground colours. | ||||
Constantine Linnick
|
r16130 | var s; | ||
r35158 | if(typeof color === "string") { | |||
Patrick Mezard
|
r16138 | s = "#" + color; | ||
r35158 | } else { //typeof color === "number" | |||
Constantine Linnick
|
r16130 | color %= colors.length; | ||
var red = (colors[color][0] * fg) || bg; | ||||
var green = (colors[color][1] * fg) || bg; | ||||
var blue = (colors[color][2] * fg) || bg; | ||||
red = Math.round(red * 255); | ||||
green = Math.round(green * 255); | ||||
blue = Math.round(blue * 255); | ||||
s = 'rgb(' + red + ', ' + green + ', ' + blue + ')'; | ||||
} | ||||
Benoit Allard
|
r14046 | this.ctx.strokeStyle = s; | ||
this.ctx.fillStyle = s; | ||||
return s; | ||||
Kevin Bullock
|
r19530 | |||
r35255 | }, | |||
Benoit Allard
|
r14046 | |||
r35255 | edge: function(x0, y0, x1, y1, color, width) { | |||
Kevin Bullock
|
r19530 | |||
Patrick Mezard
|
r16137 | this.setColor(color, 0.0, 0.65); | ||
Patrick Mezard
|
r16138 | if(width >= 0) | ||
this.ctx.lineWidth = width; | ||||
Patrick Mezard
|
r16137 | this.ctx.beginPath(); | ||
this.ctx.moveTo(x0, y0); | ||||
this.ctx.lineTo(x1, y1); | ||||
this.ctx.stroke(); | ||||
Kevin Bullock
|
r19530 | |||
r35255 | }, | |||
Patrick Mezard
|
r16137 | |||
r35256 | vertex: function(x, y, radius, color, parity, cur) { | |||
this.ctx.beginPath(); | ||||
this.setColor(color, 0.25, 0.75); | ||||
this.ctx.arc(x, y, radius, 0, Math.PI * 2, true); | ||||
this.ctx.fill(); | ||||
var left = (this.bg_height - this.box_size) + (this.columns + 1) * this.box_size; | ||||
var item = document.querySelector('[data-node="' + cur.node + '"]'); | ||||
if (item) { | ||||
item.style.paddingLeft = left + 'px'; | ||||
} | ||||
return ['', '']; | ||||
}, | ||||
r35255 | render: function(data) { | |||
Kevin Bullock
|
r19530 | |||
Benoit Allard
|
r14046 | var backgrounds = ''; | ||
var nodedata = ''; | ||||
r35408 | var i, j, cur, line, start, end, color, x, y, x0, y0, x1, y1, column, radius; | |||
Kevin Bullock
|
r19530 | |||
r35408 | var cols = 0; | |||
for (i = 0; i < data.length; i++) { | ||||
cur = data[i]; | ||||
for (j = 0; j < cur.edges.length; j++) { | ||||
line = cur.edges[j]; | ||||
cols = Math.max(cols, line[0], line[1]); | ||||
} | ||||
} | ||||
this.canvas.width = (cols + 1) * this.bg_height; | ||||
this.canvas.height = (data.length + 1) * this.bg_height - 27; | ||||
for (i = 0; i < data.length; i++) { | ||||
Kevin Bullock
|
r19530 | |||
Benoit Allard
|
r14046 | var parity = i % 2; | ||
this.cell[1] += this.bg_height; | ||||
this.bg[1] += this.bg_height; | ||||
Kevin Bullock
|
r19530 | |||
r35408 | cur = data[i]; | |||
Benoit Allard
|
r14046 | var fold = false; | ||
Kevin Bullock
|
r19530 | |||
Patrick Mezard
|
r16138 | var prevWidth = this.ctx.lineWidth; | ||
r35408 | for (j = 0; j < cur.edges.length; j++) { | |||
Kevin Bullock
|
r19530 | |||
r35218 | line = cur.edges[j]; | |||
Benoit Allard
|
r14046 | start = line[0]; | ||
end = line[1]; | ||||
color = line[2]; | ||||
Patrick Mezard
|
r16138 | var width = line[3]; | ||
if(width < 0) | ||||
width = prevWidth; | ||||
var branchcolor = line[4]; | ||||
if(branchcolor) | ||||
color = branchcolor; | ||||
Kevin Bullock
|
r19530 | |||
Benoit Allard
|
r14046 | if (end > this.columns || start > this.columns) { | ||
this.columns += 1; | ||||
} | ||||
Kevin Bullock
|
r19530 | |||
r35158 | if (start === this.columns && start > end) { | |||
r35033 | fold = true; | |||
Benoit Allard
|
r14046 | } | ||
Kevin Bullock
|
r19530 | |||
Benoit Allard
|
r14046 | x0 = this.cell[0] + this.box_size * start + this.box_size / 2; | ||
y0 = this.bg[1] - this.bg_height / 2; | ||||
x1 = this.cell[0] + this.box_size * end + this.box_size / 2; | ||||
y1 = this.bg[1] + this.bg_height / 2; | ||||
Kevin Bullock
|
r19530 | |||
Patrick Mezard
|
r16138 | this.edge(x0, y0, x1, y1, color, width); | ||
Kevin Bullock
|
r19530 | |||
Benoit Allard
|
r14046 | } | ||
Patrick Mezard
|
r16138 | this.ctx.lineWidth = prevWidth; | ||
Kevin Bullock
|
r19530 | |||
Benoit Allard
|
r14046 | // Draw the revision node in the right column | ||
Kevin Bullock
|
r19530 | |||
r35218 | column = cur.vertex[0]; | |||
color = cur.vertex[1]; | ||||
Kevin Bullock
|
r19530 | |||
Benoit Allard
|
r14046 | radius = this.box_size / 8; | ||
x = this.cell[0] + this.box_size * column + this.box_size / 2; | ||||
y = this.bg[1] - this.bg_height / 2; | ||||
r35161 | var add = this.vertex(x, y, radius, color, parity, cur); | |||
Benoit Allard
|
r14046 | backgrounds += add[0]; | ||
nodedata += add[1]; | ||||
Kevin Bullock
|
r19530 | |||
Benoit Allard
|
r14046 | if (fold) this.columns -= 1; | ||
Kevin Bullock
|
r19530 | |||
Benoit Allard
|
r14046 | } | ||
Kevin Bullock
|
r19530 | |||
Benoit Allard
|
r14046 | document.getElementById('nodebgs').innerHTML += backgrounds; | ||
document.getElementById('graphnodes').innerHTML += nodedata; | ||||
Kevin Bullock
|
r19530 | |||
r35255 | } | |||
Benoit Allard
|
r14046 | |||
r35255 | }; | |||
Benoit Allard
|
r14046 | |||
Alexander Plavin
|
r19858 | function process_dates(parentSelector){ | ||
Benoit Allard
|
r14046 | |||
// derived from code from mercurial/templatefilter.py | ||||
var scales = { | ||||
'year': 365 * 24 * 60 * 60, | ||||
'month': 30 * 24 * 60 * 60, | ||||
'week': 7 * 24 * 60 * 60, | ||||
'day': 24 * 60 * 60, | ||||
'hour': 60 * 60, | ||||
'minute': 60, | ||||
'second': 1 | ||||
}; | ||||
function format(count, string){ | ||||
var ret = count + ' ' + string; | ||||
if (count > 1){ | ||||
ret = ret + 's'; | ||||
} | ||||
Benoit Allard
|
r14881 | return ret; | ||
} | ||||
function shortdate(date){ | ||||
var ret = date.getFullYear() + '-'; | ||||
// getMonth() gives a 0-11 result | ||||
var month = date.getMonth() + 1; | ||||
if (month <= 9){ | ||||
ret += '0' + month; | ||||
} else { | ||||
ret += month; | ||||
} | ||||
ret += '-'; | ||||
var day = date.getDate(); | ||||
if (day <= 9){ | ||||
ret += '0' + day; | ||||
} else { | ||||
ret += day; | ||||
} | ||||
Benoit Allard
|
r14046 | return ret; | ||
} | ||||
Benoit Allard
|
r14881 | function age(datestr){ | ||
var now = new Date(); | ||||
var once = new Date(datestr); | ||||
Benoit Allard
|
r14046 | if (isNaN(once.getTime())){ | ||
// parsing error | ||||
return datestr; | ||||
} | ||||
var delta = Math.floor((now.getTime() - once.getTime()) / 1000); | ||||
Alexander Plavin
|
r19834 | var future = false; | ||
Benoit Allard
|
r14046 | if (delta < 0){ | ||
Alexander Plavin
|
r19834 | future = true; | ||
Benoit Allard
|
r14046 | delta = -delta; | ||
if (delta > (30 * scales.year)){ | ||||
return "in the distant future"; | ||||
} | ||||
} | ||||
if (delta > (2 * scales.year)){ | ||||
Benoit Allard
|
r14881 | return shortdate(once); | ||
Benoit Allard
|
r14046 | } | ||
r35033 | for (var unit in scales){ | |||
r35159 | if (!scales.hasOwnProperty(unit)) { continue; } | |||
Benoit Allard
|
r14046 | var s = scales[unit]; | ||
var n = Math.floor(delta / s); | ||||
r35158 | if ((n >= 2) || (s === 1)){ | |||
Benoit Allard
|
r14046 | if (future){ | ||
return format(n, unit) + ' from now'; | ||||
} else { | ||||
return format(n, unit) + ' ago'; | ||||
} | ||||
} | ||||
} | ||||
} | ||||
Alexander Plavin
|
r19858 | var nodes = document.querySelectorAll((parentSelector || '') + ' .age'); | ||
Alexander Plavin
|
r19834 | var dateclass = new RegExp('\\bdate\\b'); | ||
for (var i=0; i<nodes.length; ++i){ | ||||
var node = nodes[i]; | ||||
var classes = node.className; | ||||
Alexander Plavin
|
r19857 | var agevalue = age(node.textContent); | ||
if (dateclass.test(classes)){ | ||||
// We want both: date + (age) | ||||
node.textContent += ' ('+agevalue+')'; | ||||
} else { | ||||
node.title = node.textContent; | ||||
node.textContent = agevalue; | ||||
Benoit Allard
|
r14046 | } | ||
} | ||||
Alexander Plavin
|
r19834 | } | ||
Steven Brown
|
r14571 | |||
Alexander Plavin
|
r19428 | function toggleDiffstat() { | ||
var curdetails = document.getElementById('diffstatdetails').style.display; | ||||
r35158 | var curexpand = curdetails === 'none' ? 'inline' : 'none'; | |||
Alexander Plavin
|
r19428 | document.getElementById('diffstatdetails').style.display = curexpand; | ||
document.getElementById('diffstatexpand').style.display = curdetails; | ||||
Steven Brown
|
r14571 | } | ||
Alexander Plavin
|
r19430 | |||
function toggleLinewrap() { | ||||
function getLinewrap() { | ||||
var nodes = document.getElementsByClassName('sourcelines'); | ||||
// if there are no such nodes, error is thrown here | ||||
return nodes[0].classList.contains('wrap'); | ||||
} | ||||
function setLinewrap(enable) { | ||||
var nodes = document.getElementsByClassName('sourcelines'); | ||||
r35033 | var i; | |||
for (i = 0; i < nodes.length; i++) { | ||||
Alexander Plavin
|
r19430 | if (enable) { | ||
nodes[i].classList.add('wrap'); | ||||
} else { | ||||
nodes[i].classList.remove('wrap'); | ||||
} | ||||
} | ||||
var links = document.getElementsByClassName('linewraplink'); | ||||
r35033 | for (i = 0; i < links.length; i++) { | |||
Alexander Plavin
|
r19430 | links[i].innerHTML = enable ? 'on' : 'off'; | ||
} | ||||
} | ||||
setLinewrap(!getLinewrap()); | ||||
} | ||||
Alexander Plavin
|
r19739 | |||
function format(str, replacements) { | ||||
return str.replace(/%(\w+)%/g, function(match, p1) { | ||||
return String(replacements[p1]); | ||||
}); | ||||
} | ||||
Alexander Plavin
|
r19740 | |||
function makeRequest(url, method, onstart, onsuccess, onerror, oncomplete) { | ||||
r35161 | var xhr = new XMLHttpRequest(); | |||
r35160 | xhr.onreadystatechange = function() { | |||
if (xhr.readyState === 4) { | ||||
Alexander Plavin
|
r19740 | try { | ||
r35160 | if (xhr.status === 200) { | |||
onsuccess(xhr.responseText); | ||||
Alexander Plavin
|
r19740 | } else { | ||
throw 'server error'; | ||||
} | ||||
} catch (e) { | ||||
onerror(e); | ||||
} finally { | ||||
oncomplete(); | ||||
} | ||||
} | ||||
}; | ||||
r35160 | xhr.open(method, url); | |||
xhr.overrideMimeType("text/xhtml; charset=" + document.characterSet.toLowerCase()); | ||||
xhr.send(); | ||||
Alexander Plavin
|
r19740 | onstart(); | ||
r35160 | return xhr; | |||
Alexander Plavin
|
r19740 | } | ||
Alexander Plavin
|
r19741 | |||
Alexander Plavin
|
r19742 | function removeByClassName(className) { | ||
var nodes = document.getElementsByClassName(className); | ||||
while (nodes.length) { | ||||
nodes[0].parentNode.removeChild(nodes[0]); | ||||
} | ||||
} | ||||
Alexander Plavin
|
r19741 | function docFromHTML(html) { | ||
var doc = document.implementation.createHTMLDocument(''); | ||||
doc.documentElement.innerHTML = html; | ||||
return doc; | ||||
} | ||||
Alexander Plavin
|
r19743 | |||
function appendFormatHTML(element, formatStr, replacements) { | ||||
element.insertAdjacentHTML('beforeend', format(formatStr, replacements)); | ||||
} | ||||
Alexander Plavin
|
r19746 | |||
r35217 | function adoptChildren(from, to) { | |||
var nodes = from.children; | ||||
var curClass = 'c' + Date.now(); | ||||
while (nodes.length) { | ||||
var node = nodes[0]; | ||||
node = document.adoptNode(node); | ||||
node.classList.add(curClass); | ||||
to.appendChild(node); | ||||
} | ||||
process_dates('.' + curClass); | ||||
} | ||||
Alexander Plavin
|
r19746 | function ajaxScrollInit(urlFormat, | ||
Alexander Plavin
|
r19781 | nextPageVar, | ||
nextPageVarGet, | ||||
Alexander Plavin
|
r19746 | containerSelector, | ||
Alexander Plavin
|
r19782 | messageFormat, | ||
mode) { | ||||
r35161 | var updateInitiated = false; | |||
var container = document.querySelector(containerSelector); | ||||
Alexander Plavin
|
r19746 | |||
function scrollHandler() { | ||||
if (updateInitiated) { | ||||
return; | ||||
} | ||||
var scrollHeight = document.documentElement.scrollHeight; | ||||
var clientHeight = document.documentElement.clientHeight; | ||||
r35033 | var scrollTop = document.body.scrollTop || document.documentElement.scrollTop; | |||
Alexander Plavin
|
r19746 | |||
if (scrollHeight - (scrollTop + clientHeight) < 50) { | ||||
updateInitiated = true; | ||||
Alexander Plavin
|
r19756 | removeByClassName('scroll-loading-error'); | ||
Alexander Plavin
|
r19760 | container.lastElementChild.classList.add('scroll-separator'); | ||
Alexander Plavin
|
r19746 | |||
Alexander Plavin
|
r19781 | if (!nextPageVar) { | ||
Alexander Plavin
|
r19754 | var message = { | ||
r26832 | 'class': 'scroll-loading-info', | |||
Alexander Plavin
|
r19754 | text: 'No more entries' | ||
}; | ||||
appendFormatHTML(container, messageFormat, message); | ||||
Alexander Plavin
|
r19746 | return; | ||
} | ||||
makeRequest( | ||||
Alexander Plavin
|
r19781 | format(urlFormat, {next: nextPageVar}), | ||
Alexander Plavin
|
r19746 | 'GET', | ||
function onstart() { | ||||
Alexander Plavin
|
r19755 | var message = { | ||
r26832 | 'class': 'scroll-loading', | |||
Alexander Plavin
|
r19755 | text: 'Loading...' | ||
}; | ||||
appendFormatHTML(container, messageFormat, message); | ||||
Alexander Plavin
|
r19746 | }, | ||
function onsuccess(htmlText) { | ||||
r35217 | var doc = docFromHTML(htmlText); | |||
r35158 | if (mode === 'graph') { | |||
r35161 | var graph = window.graph; | |||
Alexander Plavin
|
r19782 | var dataStr = htmlText.match(/^\s*var data = (.*);$/m)[1]; | ||
Takumi IINO
|
r19957 | var data = JSON.parse(dataStr); | ||
Alexander Plavin
|
r19907 | if (data.length < nextPageVar) { | ||
nextPageVar = undefined; | ||||
} | ||||
Alexander Plavin
|
r19782 | graph.reset(); | ||
r35217 | adoptChildren(doc.querySelector('#graphnodes'), container.querySelector('#graphnodes')); | |||
Alexander Plavin
|
r19782 | graph.render(data); | ||
} else { | ||||
r35217 | adoptChildren(doc.querySelector(containerSelector), container); | |||
Alexander Plavin
|
r19746 | } | ||
Alexander Plavin
|
r19907 | |||
nextPageVar = nextPageVarGet(htmlText, nextPageVar); | ||||
Alexander Plavin
|
r19746 | }, | ||
function onerror(errorText) { | ||||
Alexander Plavin
|
r19756 | var message = { | ||
r26832 | 'class': 'scroll-loading-error', | |||
Alexander Plavin
|
r19756 | text: 'Error: ' + errorText | ||
}; | ||||
appendFormatHTML(container, messageFormat, message); | ||||
Alexander Plavin
|
r19746 | }, | ||
function oncomplete() { | ||||
Alexander Plavin
|
r19755 | removeByClassName('scroll-loading'); | ||
Alexander Plavin
|
r19746 | updateInitiated = false; | ||
scrollHandler(); | ||||
} | ||||
); | ||||
} | ||||
} | ||||
window.addEventListener('scroll', scrollHandler); | ||||
window.addEventListener('resize', scrollHandler); | ||||
scrollHandler(); | ||||
} | ||||
Gregory Szorc
|
r30765 | |||
Gregory Szorc
|
r34392 | function renderDiffOptsForm() { | ||
// We use URLSearchParams for query string manipulation. Old browsers don't | ||||
// support this API. | ||||
if (!("URLSearchParams" in window)) { | ||||
return; | ||||
} | ||||
var form = document.getElementById("diffopts-form"); | ||||
var KEYS = [ | ||||
"ignorews", | ||||
"ignorewsamount", | ||||
"ignorewseol", | ||||
"ignoreblanklines", | ||||
]; | ||||
r35162 | var urlParams = new window.URLSearchParams(window.location.search); | |||
Gregory Szorc
|
r34392 | |||
function updateAndRefresh(e) { | ||||
var checkbox = e.target; | ||||
var name = checkbox.id.substr(0, checkbox.id.indexOf("-")); | ||||
urlParams.set(name, checkbox.checked ? "1" : "0"); | ||||
window.location.search = urlParams.toString(); | ||||
} | ||||
r35158 | var allChecked = form.getAttribute("data-ignorews") === "1"; | |||
Gregory Szorc
|
r34392 | |||
for (var i = 0; i < KEYS.length; i++) { | ||||
var key = KEYS[i]; | ||||
var checkbox = document.getElementById(key + "-checkbox"); | ||||
if (!checkbox) { | ||||
continue; | ||||
} | ||||
r35161 | var currentValue = form.getAttribute("data-" + key); | |||
r35158 | checkbox.checked = currentValue !== "0"; | |||
Gregory Szorc
|
r34392 | |||
// ignorews implies ignorewsamount and ignorewseol. | ||||
r35158 | if (allChecked && (key === "ignorewsamount" || key === "ignorewseol")) { | |||
Gregory Szorc
|
r34392 | checkbox.checked = true; | ||
checkbox.disabled = true; | ||||
} | ||||
checkbox.addEventListener("change", updateAndRefresh, false); | ||||
} | ||||
form.style.display = 'block'; | ||||
} | ||||
Gregory Szorc
|
r30765 | document.addEventListener('DOMContentLoaded', function() { | ||
process_dates(); | ||||
}, false); | ||||