followlines.js
242 lines
| 8.9 KiB
| application/javascript
|
JavascriptLexer
Denis Laxalde
|
r31785 | // followlines.js - JavaScript utilities for followlines UI | ||
// | ||||
// Copyright 2017 Logilab SA <contact@logilab.fr> | ||||
// | ||||
// This software may be used and distributed according to the terms of the | ||||
// GNU General Public License version 2 or any later version. | ||||
//** Install event listeners for line block selection and followlines action */ | ||||
document.addEventListener('DOMContentLoaded', function() { | ||||
var sourcelines = document.getElementsByClassName('sourcelines')[0]; | ||||
if (typeof sourcelines === 'undefined') { | ||||
return; | ||||
} | ||||
// URL to complement with "linerange" query parameter | ||||
var targetUri = sourcelines.dataset.logurl; | ||||
if (typeof targetUri === 'undefined')Â { | ||||
return; | ||||
} | ||||
Denis Laxalde
|
r32993 | // Tag of children of "sourcelines" element on which to add "line | ||
// selection" style. | ||||
var selectableTag = sourcelines.dataset.selectabletag; | ||||
if (typeof selectableTag === 'undefined') { | ||||
return; | ||||
} | ||||
Denis Laxalde
|
r32070 | var isHead = parseInt(sourcelines.dataset.ishead || "0"); | ||
Denis Laxalde
|
r31848 | // tooltip to invite on lines selection | ||
var tooltip = document.createElement('div'); | ||||
tooltip.id = 'followlines-tooltip'; | ||||
tooltip.classList.add('hidden'); | ||||
var initTooltipText = 'click to start following lines history from here'; | ||||
tooltip.textContent = initTooltipText; | ||||
sourcelines.appendChild(tooltip); | ||||
Denis Laxalde
|
r31849 | //* position "element" on top-right of cursor */ | ||
function positionTopRight(element, event) { | ||||
var x = (event.clientX + 10) + 'px', | ||||
y = (event.clientY - 20) + 'px'; | ||||
element.style.top = y; | ||||
element.style.left = x; | ||||
} | ||||
Denis Laxalde
|
r31848 | var tooltipTimeoutID; | ||
//* move the "tooltip" with cursor (top-right) and show it after 1s */ | ||||
function moveAndShowTooltip(e) { | ||||
if (typeof tooltipTimeoutID !== 'undefined') { | ||||
// avoid accumulation of timeout callbacks (blinking) | ||||
window.clearTimeout(tooltipTimeoutID); | ||||
} | ||||
tooltip.classList.add('hidden'); | ||||
Denis Laxalde
|
r31849 | positionTopRight(tooltip, e); | ||
Denis Laxalde
|
r31848 | tooltipTimeoutID = window.setTimeout(function() { | ||
tooltip.classList.remove('hidden'); | ||||
}, 1000); | ||||
} | ||||
// on mousemove, show tooltip close to cursor position | ||||
sourcelines.addEventListener('mousemove', moveAndShowTooltip); | ||||
Denis Laxalde
|
r32993 | // retrieve all direct *selectable* children of class="sourcelines" | ||
// element | ||||
var selectableElements = Array.prototype.filter.call( | ||||
Denis Laxalde
|
r31785 | sourcelines.children, | ||
Denis Laxalde
|
r32993 | function(x) { return x.tagName === selectableTag }); | ||
Denis Laxalde
|
r31785 | |||
// add a "followlines-select" class to change cursor type in CSS | ||||
Denis Laxalde
|
r32993 | for (var i = 0; i < selectableElements.length; i++) { | ||
selectableElements[i].classList.add('followlines-select'); | ||||
Denis Laxalde
|
r31785 | } | ||
var lineSelectedCSSClass = 'followlines-selected'; | ||||
Denis Laxalde
|
r32993 | //** add CSS class on selectable elements in `from`-`to` line range */ | ||
Denis Laxalde
|
r31785 | function addSelectedCSSClass(from, to) { | ||
for (var i = from; i <= to; i++) { | ||||
Denis Laxalde
|
r32993 | selectableElements[i].classList.add(lineSelectedCSSClass); | ||
Denis Laxalde
|
r31785 | } | ||
} | ||||
//** remove CSS class from previously selected lines */ | ||||
function removeSelectedCSSClass() { | ||||
var elements = sourcelines.getElementsByClassName( | ||||
lineSelectedCSSClass); | ||||
while (elements.length) { | ||||
elements[0].classList.remove(lineSelectedCSSClass); | ||||
} | ||||
} | ||||
Denis Laxalde
|
r32993 | // ** return the element of type "selectableTag" parent of `element` */ | ||
function selectableParent(element) { | ||||
Denis Laxalde
|
r31785 | var parent = element.parentElement; | ||
if (parent === null) { | ||||
return null; | ||||
} | ||||
Denis Laxalde
|
r32993 | if (element.tagName == selectableTag && parent.isSameNode(sourcelines)) { | ||
Denis Laxalde
|
r31785 | return element; | ||
} | ||||
Denis Laxalde
|
r32993 | return selectableParent(parent); | ||
Denis Laxalde
|
r31785 | } | ||
//** event handler for "click" on the first line of a block */ | ||||
function lineSelectStart(e) { | ||||
Denis Laxalde
|
r32993 | var startElement = selectableParent(e.target); | ||
Denis Laxalde
|
r31785 | if (startElement === null) { | ||
Denis Laxalde
|
r32993 | // not a "selectable" element (maybe <a>): abort, keeping event | ||
// listener registered for other click with a "selectable" target | ||||
Denis Laxalde
|
r31785 | return; | ||
} | ||||
Denis Laxalde
|
r31848 | |||
// update tooltip text | ||||
tooltip.textContent = 'click again to terminate line block selection here'; | ||||
Denis Laxalde
|
r31785 | var startId = parseInt(startElement.id.slice(1)); | ||
startElement.classList.add(lineSelectedCSSClass); // CSS | ||||
// remove this event listener | ||||
sourcelines.removeEventListener('click', lineSelectStart); | ||||
//** event handler for "click" on the last line of the block */ | ||||
function lineSelectEnd(e) { | ||||
Denis Laxalde
|
r32993 | var endElement = selectableParent(e.target); | ||
Denis Laxalde
|
r31785 | if (endElement === null) { | ||
// not a <span> (maybe <a>): abort, keeping event listener | ||||
// registered for other click with <span> target | ||||
return; | ||||
} | ||||
// remove this event listener | ||||
sourcelines.removeEventListener('click', lineSelectEnd); | ||||
Denis Laxalde
|
r31848 | // hide tooltip and disable motion tracking | ||
tooltip.classList.add('hidden'); | ||||
sourcelines.removeEventListener('mousemove', moveAndShowTooltip); | ||||
window.clearTimeout(tooltipTimeoutID); | ||||
//* restore initial "tooltip" state */ | ||||
function restoreTooltip() { | ||||
tooltip.textContent = initTooltipText; | ||||
sourcelines.addEventListener('mousemove', moveAndShowTooltip); | ||||
} | ||||
Denis Laxalde
|
r31785 | // compute line range (startId, endId) | ||
var endId = parseInt(endElement.id.slice(1)); | ||||
if (endId == startId) { | ||||
// clicked twice the same line, cancel and reset initial state | ||||
Denis Laxalde
|
r31848 | // (CSS, event listener for selection start, tooltip) | ||
Denis Laxalde
|
r31785 | removeSelectedCSSClass(); | ||
sourcelines.addEventListener('click', lineSelectStart); | ||||
Denis Laxalde
|
r31848 | restoreTooltip(); | ||
Denis Laxalde
|
r31785 | return; | ||
} | ||||
var inviteElement = endElement; | ||||
if (endId < startId) { | ||||
var tmp = endId; | ||||
endId = startId; | ||||
startId = tmp; | ||||
inviteElement = startElement; | ||||
} | ||||
addSelectedCSSClass(startId - 1, endId -1); // CSS | ||||
// append the <div id="followlines"> element to last line of the | ||||
// selection block | ||||
Denis Laxalde
|
r32070 | var divAndButton = followlinesBox(targetUri, startId, endId, isHead); | ||
Denis Laxalde
|
r31785 | var div = divAndButton[0], | ||
button = divAndButton[1]; | ||||
inviteElement.appendChild(div); | ||||
Denis Laxalde
|
r31849 | // set position close to cursor (top-right) | ||
positionTopRight(div, e); | ||||
Denis Laxalde
|
r31785 | |||
//** event handler for cancelling selection */ | ||||
function cancel() { | ||||
// remove invite box | ||||
div.parentNode.removeChild(div); | ||||
// restore initial event listeners | ||||
sourcelines.addEventListener('click', lineSelectStart); | ||||
sourcelines.removeEventListener('click', cancel); | ||||
// remove styles on selected lines | ||||
removeSelectedCSSClass(); | ||||
Denis Laxalde
|
r31848 | // restore tooltip element | ||
restoreTooltip(); | ||||
Denis Laxalde
|
r31785 | } | ||
// bind cancel event to click on <button> | ||||
button.addEventListener('click', cancel); | ||||
// as well as on an click on any source line | ||||
sourcelines.addEventListener('click', cancel); | ||||
} | ||||
sourcelines.addEventListener('click', lineSelectEnd); | ||||
} | ||||
sourcelines.addEventListener('click', lineSelectStart); | ||||
//** return a <div id="followlines"> and inner cancel <button> elements */ | ||||
Denis Laxalde
|
r32070 | function followlinesBox(targetUri, fromline, toline, isHead) { | ||
Denis Laxalde
|
r31785 | // <div id="followlines"> | ||
var div = document.createElement('div'); | ||||
div.id = 'followlines'; | ||||
// <div class="followlines-cancel"> | ||||
var buttonDiv = document.createElement('div'); | ||||
buttonDiv.classList.add('followlines-cancel'); | ||||
// <button>x</button> | ||||
var button = document.createElement('button'); | ||||
button.textContent = 'x'; | ||||
buttonDiv.appendChild(button); | ||||
div.appendChild(buttonDiv); | ||||
// <div class="followlines-link"> | ||||
var aDiv = document.createElement('div'); | ||||
aDiv.classList.add('followlines-link'); | ||||
Denis Laxalde
|
r31940 | aDiv.textContent = 'follow history of lines ' + fromline + ':' + toline + ':'; | ||
var linesep = document.createElement('br'); | ||||
aDiv.appendChild(linesep); | ||||
// link to "ascending" followlines | ||||
var aAsc = document.createElement('a'); | ||||
Denis Laxalde
|
r31785 | var url = targetUri + '?patch=&linerange=' + fromline + ':' + toline; | ||
Denis Laxalde
|
r31940 | aAsc.setAttribute('href', url); | ||
Denis Laxalde
|
r32071 | aAsc.textContent = 'older'; | ||
Denis Laxalde
|
r31940 | aDiv.appendChild(aAsc); | ||
Denis Laxalde
|
r32070 | |||
if (!isHead) { | ||||
var sep = document.createTextNode(' / '); | ||||
aDiv.appendChild(sep); | ||||
// link to "descending" followlines | ||||
var aDesc = document.createElement('a'); | ||||
aDesc.setAttribute('href', url + '&descend='); | ||||
Denis Laxalde
|
r32071 | aDesc.textContent = 'newer'; | ||
Denis Laxalde
|
r32070 | aDiv.appendChild(aDesc); | ||
} | ||||
Denis Laxalde
|
r31940 | |||
Denis Laxalde
|
r31785 | div.appendChild(aDiv); | ||
return [div, button]; | ||||
} | ||||
}, false); | ||||