rhodecode.js
2149 lines
| 73.0 KiB
| application/javascript
|
JavascriptLexer
r1421 | /** | |||
RhodeCode JS Files | ||||
**/ | ||||
if (typeof console == "undefined" || typeof console.log == "undefined"){ | ||||
r3696 | console = { log: function() {} } | |||
r1421 | } | |||
/** | ||||
* INJECT .format function into String | ||||
* Usage: "My name is {0} {1}".format("Johny","Bravo") | ||||
* Return "My name is Johny Bravo" | ||||
* Inspired by https://gist.github.com/1049426 | ||||
*/ | ||||
String.prototype.format = function() { | ||||
Mads Kiilerich
|
r4158 | function format() { | ||
r3696 | var str = this; | |||
var len = arguments.length+1; | ||||
var safe = undefined; | ||||
var arg = undefined; | ||||
r1421 | ||||
r3696 | // For each {0} {1} {n...} replace with the argument in that position. If | |||
// the argument is an object or an array it will be stringified to JSON. | ||||
for (var i=0; i < len; arg = arguments[i++]) { | ||||
Mads Kiilerich
|
r4158 | safe = typeof arg === 'object' ? JSON.stringify(arg) : arg; | ||
str = str.replace(RegExp('\\{'+(i-1)+'\\}', 'g'), safe); | ||||
r3696 | } | |||
return str; | ||||
Mads Kiilerich
|
r4158 | } | ||
r1421 | ||||
Mads Kiilerich
|
r4158 | // Save a reference of what may already exist under the property native. | ||
// Allows for doing something like: if("".format.native) { /* use native */ } | ||||
format.native = String.prototype.format; | ||||
r3696 | ||||
Mads Kiilerich
|
r4158 | // Replace the prototype property | ||
return format; | ||||
r1421 | ||||
r1458 | }(); | |||
r1421 | ||||
r2369 | String.prototype.strip = function(char) { | |||
r3696 | if(char === undefined){ | |||
char = '\\s'; | ||||
} | ||||
return this.replace(new RegExp('^'+char+'+|'+char+'+$','g'), ''); | ||||
r2369 | } | |||
Mads Kiilerich
|
r4157 | |||
r2369 | String.prototype.lstrip = function(char) { | |||
r3696 | if(char === undefined){ | |||
char = '\\s'; | ||||
} | ||||
return this.replace(new RegExp('^'+char+'+'),''); | ||||
r2369 | } | |||
Mads Kiilerich
|
r4157 | |||
r2369 | String.prototype.rstrip = function(char) { | |||
r3696 | if(char === undefined){ | |||
char = '\\s'; | ||||
} | ||||
return this.replace(new RegExp(''+char+'+$'),''); | ||||
r2369 | } | |||
r1465 | ||||
Mads Kiilerich
|
r4162 | /* https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/indexOf#Polyfill | ||
under MIT license / public domain, see | ||||
https://developer.mozilla.org/en-US/docs/MDN/About#Copyrights_and_licenses */ | ||||
r2612 | if(!Array.prototype.indexOf) { | |||
Mads Kiilerich
|
r4162 | Array.prototype.indexOf = function (searchElement, fromIndex) { | ||
if ( this === undefined || this === null ) { | ||||
throw new TypeError( '"this" is null or not defined' ); | ||||
} | ||||
var length = this.length >>> 0; // Hack to convert object.length to a UInt32 | ||||
fromIndex = +fromIndex || 0; | ||||
if (Math.abs(fromIndex) === Infinity) { | ||||
fromIndex = 0; | ||||
} | ||||
if (fromIndex < 0) { | ||||
fromIndex += length; | ||||
if (fromIndex < 0) { | ||||
fromIndex = 0; | ||||
r2612 | } | |||
} | ||||
Mads Kiilerich
|
r4162 | |||
for (;fromIndex < length; fromIndex++) { | ||||
if (this[fromIndex] === searchElement) { | ||||
return fromIndex; | ||||
} | ||||
} | ||||
r2612 | return -1; | |||
}; | ||||
} | ||||
Mads Kiilerich
|
r4161 | /* https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/filter#Compatibility | ||
under MIT license / public domain, see | ||||
https://developer.mozilla.org/en-US/docs/MDN/About#Copyrights_and_licenses */ | ||||
if (!Array.prototype.filter) | ||||
{ | ||||
Array.prototype.filter = function(fun /*, thisArg */) | ||||
{ | ||||
"use strict"; | ||||
if (this === void 0 || this === null) | ||||
throw new TypeError(); | ||||
var t = Object(this); | ||||
var len = t.length >>> 0; | ||||
if (typeof fun !== "function") | ||||
throw new TypeError(); | ||||
var res = []; | ||||
var thisArg = arguments.length >= 2 ? arguments[1] : void 0; | ||||
for (var i = 0; i < len; i++) | ||||
{ | ||||
if (i in t) | ||||
{ | ||||
var val = t[i]; | ||||
// NOTE: Technically this should Object.defineProperty at | ||||
// the next index, as push can be affected by | ||||
// properties on Object.prototype and Array.prototype. | ||||
// But that method's new, and collisions should be | ||||
// rare, so use the more-compatible alternative. | ||||
if (fun.call(thisArg, val, i, t)) | ||||
res.push(val); | ||||
} | ||||
} | ||||
return res; | ||||
}; | ||||
} | ||||
r1465 | /** | |||
Bradley M. Kuhn
|
r4135 | * A customized version of PyRoutes.JS from https://pypi.python.org/pypi/pyroutes.js/ | ||
* which is copyright Stephane Klein and was made available under the BSD License. | ||||
r3696 | * | |||
r3388 | * Usage pyroutes.url('mark_error_fixed',{"error_id":error_id}) // /mark_error_fixed/<error_id> | |||
*/ | ||||
var pyroutes = (function() { | ||||
r3696 | // access global map defined in special file pyroutes | |||
r3388 | var matchlist = PROUTES_MAP; | |||
var sprintf = (function() { | ||||
function get_type(variable) { | ||||
return Object.prototype.toString.call(variable).slice(8, -1).toLowerCase(); | ||||
} | ||||
function str_repeat(input, multiplier) { | ||||
for (var output = []; multiplier > 0; output[--multiplier] = input) {/* do nothing */} | ||||
return output.join(''); | ||||
} | ||||
r1465 | ||||
r3388 | var str_format = function() { | |||
if (!str_format.cache.hasOwnProperty(arguments[0])) { | ||||
str_format.cache[arguments[0]] = str_format.parse(arguments[0]); | ||||
} | ||||
return str_format.format.call(null, str_format.cache[arguments[0]], arguments); | ||||
}; | ||||
str_format.format = function(parse_tree, argv) { | ||||
var cursor = 1, tree_length = parse_tree.length, node_type = '', arg, output = [], i, k, match, pad, pad_character, pad_length; | ||||
for (i = 0; i < tree_length; i++) { | ||||
node_type = get_type(parse_tree[i]); | ||||
if (node_type === 'string') { | ||||
output.push(parse_tree[i]); | ||||
} | ||||
else if (node_type === 'array') { | ||||
match = parse_tree[i]; // convenience purposes only | ||||
if (match[2]) { // keyword argument | ||||
arg = argv[cursor]; | ||||
for (k = 0; k < match[2].length; k++) { | ||||
if (!arg.hasOwnProperty(match[2][k])) { | ||||
throw(sprintf('[sprintf] property "%s" does not exist', match[2][k])); | ||||
} | ||||
arg = arg[match[2][k]]; | ||||
} | ||||
} | ||||
else if (match[1]) { // positional argument (explicit) | ||||
arg = argv[match[1]]; | ||||
} | ||||
else { // positional argument (implicit) | ||||
arg = argv[cursor++]; | ||||
} | ||||
if (/[^s]/.test(match[8]) && (get_type(arg) != 'number')) { | ||||
throw(sprintf('[sprintf] expecting number but found %s', get_type(arg))); | ||||
} | ||||
switch (match[8]) { | ||||
case 'b': arg = arg.toString(2); break; | ||||
case 'c': arg = String.fromCharCode(arg); break; | ||||
case 'd': arg = parseInt(arg, 10); break; | ||||
case 'e': arg = match[7] ? arg.toExponential(match[7]) : arg.toExponential(); break; | ||||
case 'f': arg = match[7] ? parseFloat(arg).toFixed(match[7]) : parseFloat(arg); break; | ||||
case 'o': arg = arg.toString(8); break; | ||||
case 's': arg = ((arg = String(arg)) && match[7] ? arg.substring(0, match[7]) : arg); break; | ||||
case 'u': arg = Math.abs(arg); break; | ||||
case 'x': arg = arg.toString(16); break; | ||||
case 'X': arg = arg.toString(16).toUpperCase(); break; | ||||
} | ||||
arg = (/[def]/.test(match[8]) && match[3] && arg >= 0 ? '+'+ arg : arg); | ||||
pad_character = match[4] ? match[4] == '0' ? '0' : match[4].charAt(1) : ' '; | ||||
pad_length = match[6] - String(arg).length; | ||||
pad = match[6] ? str_repeat(pad_character, pad_length) : ''; | ||||
output.push(match[5] ? arg + pad : pad + arg); | ||||
} | ||||
} | ||||
return output.join(''); | ||||
}; | ||||
str_format.cache = {}; | ||||
str_format.parse = function(fmt) { | ||||
var _fmt = fmt, match = [], parse_tree = [], arg_names = 0; | ||||
while (_fmt) { | ||||
if ((match = /^[^\x25]+/.exec(_fmt)) !== null) { | ||||
parse_tree.push(match[0]); | ||||
} | ||||
else if ((match = /^\x25{2}/.exec(_fmt)) !== null) { | ||||
parse_tree.push('%'); | ||||
} | ||||
else if ((match = /^\x25(?:([1-9]\d*)\$|\(([^\)]+)\))?(\+)?(0|'[^$])?(-)?(\d+)?(?:\.(\d+))?([b-fosuxX])/.exec(_fmt)) !== null) { | ||||
if (match[2]) { | ||||
arg_names |= 1; | ||||
var field_list = [], replacement_field = match[2], field_match = []; | ||||
if ((field_match = /^([a-z_][a-z_\d]*)/i.exec(replacement_field)) !== null) { | ||||
field_list.push(field_match[1]); | ||||
while ((replacement_field = replacement_field.substring(field_match[0].length)) !== '') { | ||||
if ((field_match = /^\.([a-z_][a-z_\d]*)/i.exec(replacement_field)) !== null) { | ||||
field_list.push(field_match[1]); | ||||
} | ||||
else if ((field_match = /^\[(\d+)\]/.exec(replacement_field)) !== null) { | ||||
field_list.push(field_match[1]); | ||||
} | ||||
else { | ||||
throw('[sprintf] huh?'); | ||||
} | ||||
} | ||||
} | ||||
else { | ||||
throw('[sprintf] huh?'); | ||||
} | ||||
match[2] = field_list; | ||||
} | ||||
else { | ||||
arg_names |= 2; | ||||
} | ||||
if (arg_names === 3) { | ||||
throw('[sprintf] mixing positional and named placeholders is not (yet) supported'); | ||||
} | ||||
parse_tree.push(match); | ||||
} | ||||
else { | ||||
throw('[sprintf] huh?'); | ||||
} | ||||
_fmt = _fmt.substring(match[0].length); | ||||
} | ||||
return parse_tree; | ||||
}; | ||||
return str_format; | ||||
})(); | ||||
var vsprintf = function(fmt, argv) { | ||||
argv.unshift(fmt); | ||||
return sprintf.apply(null, argv); | ||||
}; | ||||
return { | ||||
'url': function(route_name, params) { | ||||
var result = route_name; | ||||
if (typeof(params) != 'object'){ | ||||
r3696 | params = {}; | |||
r3388 | } | |||
if (matchlist.hasOwnProperty(route_name)) { | ||||
var route = matchlist[route_name]; | ||||
r3403 | // param substitution | |||
r3388 | for(var i=0; i < route[1].length; i++) { | |||
if (!params.hasOwnProperty(route[1][i])) | ||||
throw new Error(route[1][i] + ' missing in "' + route_name + '" route generation'); | ||||
} | ||||
result = sprintf(route[0], params); | ||||
r3696 | ||||
r3403 | var ret = []; | |||
//extra params => GET | ||||
for(param in params){ | ||||
r3696 | if (route[1].indexOf(param) == -1){ | |||
ret.push(encodeURIComponent(param) + "=" + encodeURIComponent(params[param])); | ||||
} | ||||
r3403 | } | |||
var _parts = ret.join("&"); | ||||
if(_parts){ | ||||
r3696 | result = result +'?'+ _parts | |||
r3403 | } | |||
r3388 | } | |||
return result; | ||||
}, | ||||
r3696 | 'register': function(route_name, route_tmpl, req_params) { | |||
if (typeof(req_params) != 'object') { | ||||
req_params = []; | ||||
} | ||||
//fix escape | ||||
route_tmpl = unescape(route_tmpl); | ||||
keys = []; | ||||
Mads Kiilerich
|
r4169 | for (var i=0; i < req_params.length; i++) { | ||
keys.push(req_params[i]) | ||||
r3696 | } | |||
matchlist[route_name] = [ | ||||
route_tmpl, | ||||
keys | ||||
] | ||||
}, | ||||
'_routes': function(){ | ||||
return matchlist; | ||||
} | ||||
r3388 | } | |||
})(); | ||||
r1465 | ||||
r1421 | /** | |||
* GLOBAL YUI Shortcuts | ||||
*/ | ||||
var YUC = YAHOO.util.Connect; | ||||
var YUD = YAHOO.util.Dom; | ||||
var YUE = YAHOO.util.Event; | ||||
var YUQ = YAHOO.util.Selector.query; | ||||
Mads Kiilerich
|
r4160 | /* Invoke all functions in callbacks */ | ||
r1717 | var _run_callbacks = function(callbacks){ | |||
r3696 | if (callbacks !== undefined){ | |||
var _l = callbacks.length; | ||||
for (var i=0;i<_l;i++){ | ||||
var func = callbacks[i]; | ||||
if(typeof(func)=='function'){ | ||||
try{ | ||||
func(); | ||||
}catch (err){}; | ||||
} | ||||
} | ||||
} | ||||
r1717 | } | |||
r1421 | /** | |||
r3715 | * turns objects into GET query string | |||
*/ | ||||
Mads Kiilerich
|
r4160 | var _toQueryString = function(o) { | ||
r3715 | if(typeof o !== 'object') { | |||
return false; | ||||
} | ||||
var _p, _qs = []; | ||||
for(_p in o) { | ||||
_qs.push(encodeURIComponent(_p) + '=' + encodeURIComponent(o[_p])); | ||||
} | ||||
return _qs.join('&'); | ||||
}; | ||||
/** | ||||
r1421 | * Partial Ajax Implementation | |||
r3696 | * | |||
r1421 | * @param url: defines url to make partial request | |||
* @param container: defines id of container to input partial result | ||||
* @param s_call: success callback function that takes o as arg | ||||
* o.tId | ||||
* o.status | ||||
* o.statusText | ||||
* o.getResponseHeader[ ] | ||||
* o.getAllResponseHeaders | ||||
* o.responseText | ||||
* o.responseXML | ||||
* o.argument | ||||
* @param f_call: failure callback | ||||
r3696 | * @param args arguments | |||
r1421 | */ | |||
function ypjax(url,container,s_call,f_call,args){ | ||||
r3696 | var method='GET'; | |||
if(args===undefined){ | ||||
args=null; | ||||
} | ||||
Mads Kiilerich
|
r4160 | $container = $('#' + container); | ||
r3696 | ||||
// Set special header for partial ajax == HTTP_X_PARTIAL_XHR | ||||
YUC.initHeader('X-PARTIAL-XHR',true); | ||||
// wrapper of passed callback | ||||
var s_wrapper = (function(o){ | ||||
return function(o){ | ||||
Mads Kiilerich
|
r4160 | $container.html(o.responseText); | ||
$container.css('opacity','1.0'); | ||||
r3696 | //execute the given original callback | |||
if (s_call !== undefined){ | ||||
s_call(o); | ||||
} | ||||
} | ||||
})() | ||||
Mads Kiilerich
|
r4160 | $container.css('opacity','0.3'); | ||
r3696 | YUC.asyncRequest(method,url,{ | |||
success:s_wrapper, | ||||
failure:function(o){ | ||||
Mads Kiilerich
|
r4160 | console.log('ypjax failure: '+o); | ||
$container.html('<span class="error_red">ERROR: {0}</span>'.format(o.status)); | ||||
$container.css('opacity','1.0'); | ||||
r3696 | }, | |||
cache:false | ||||
},args); | ||||
r1699 | }; | |||
r1421 | ||||
r2971 | var ajaxGET = function(url,success) { | |||
r3696 | // Set special header for ajax == HTTP_X_PARTIAL_XHR | |||
YUC.initHeader('X-PARTIAL-XHR',true); | ||||
r2971 | ||||
var sUrl = url; | ||||
var callback = { | ||||
success: success, | ||||
failure: function (o) { | ||||
Mads Kiilerich
|
r3553 | if (o.status != 0) { | ||
alert("error: " + o.statusText); | ||||
}; | ||||
r2971 | }, | |||
}; | ||||
var request = YAHOO.util.Connect.asyncRequest('GET', sUrl, callback); | ||||
return request; | ||||
}; | ||||
r2187 | var ajaxPOST = function(url,postData,success) { | |||
r3696 | // Set special header for ajax == HTTP_X_PARTIAL_XHR | |||
YUC.initHeader('X-PARTIAL-XHR',true); | ||||
r2187 | var sUrl = url; | |||
var callback = { | ||||
success: success, | ||||
failure: function (o) { | ||||
alert("error"); | ||||
}, | ||||
}; | ||||
Mads Kiilerich
|
r4160 | var postData = _toQueryString(postData); | ||
r2187 | var request = YAHOO.util.Connect.asyncRequest('POST', sUrl, callback, postData); | |||
return request; | ||||
}; | ||||
r1421 | /** | |||
Mads Kiilerich
|
r4160 | * activate .show_more links | ||
* the .show_more must have an id that is the the id of an element to hide prefixed with _ | ||||
* the parentnode will be displayed | ||||
r1426 | */ | |||
r1458 | var show_more_event = function(){ | |||
Mads Kiilerich
|
r4160 | $('.show_more').click(function(e){ | ||
var el = e.currentTarget; | ||||
$('#' + el.id.substring(1)).hide(); | ||||
$(el.parentNode).show(); | ||||
r1426 | }); | |||
r1699 | }; | |||
r1426 | ||||
r2971 | /** | |||
Mads Kiilerich
|
r4160 | * activate .lazy-cs mouseover for showing changeset tooltip | ||
r2971 | */ | |||
var show_changeset_tooltip = function(){ | ||||
Mads Kiilerich
|
r4160 | $('.lazy-cs').mouseover(function(e){ | ||
var $target = $(e.currentTarget); | ||||
var rid = $target.attr('raw_id'); | ||||
var repo_name = $target.attr('repo_name'); | ||||
if(rid && !$target.hasClass('tooltip')){ | ||||
_show_tooltip(e, _TM['loading ...']); | ||||
var url = pyroutes.url('changeset_info', {"repo_name": repo_name, "revision": rid}); | ||||
ajaxGET(url, function(o){ | ||||
var json = JSON.parse(o.responseText); | ||||
$target.addClass('tooltip') | ||||
_show_tooltip(e, json['message']); | ||||
_activate_tooltip($target); | ||||
}); | ||||
r3696 | } | |||
}); | ||||
r2971 | }; | |||
Mads Kiilerich
|
r4160 | var _onSuccessFollow = function(target){ | ||
var $target = $(target); | ||||
var $f_cnt = $('#current_followers_count'); | ||||
if($target.hasClass('follow')){ | ||||
$target.attr('class', 'following'); | ||||
$target.attr('title', _TM['Stop following this repository']); | ||||
if($f_cnt.html()){ | ||||
var cnt = Number($f_cnt.html())+1; | ||||
$f_cnt.html(cnt); | ||||
r3066 | } | |||
} | ||||
else{ | ||||
Mads Kiilerich
|
r4160 | $target.attr('class', 'follow'); | ||
$target.attr('title', _TM['Start following this repository']); | ||||
if($f_cnt.html()){ | ||||
var cnt = Number($f_cnt.html())-1; | ||||
$f_cnt.html(cnt); | ||||
r3066 | } | |||
} | ||||
} | ||||
Mads Kiilerich
|
r4160 | var toggleFollowingRepo = function(target, follows_repo_id, token, user_id){ | ||
args = 'follows_repo_id='+follows_repo_id; | ||||
r3066 | args+= '&auth_token='+token; | |||
if(user_id != undefined){ | ||||
args+="&user_id="+user_id; | ||||
} | ||||
Andrew Shadura
|
r4159 | $.post(TOGGLE_FOLLOW_URL, args, function(data){ | ||
Mads Kiilerich
|
r4160 | _onSuccessFollow(target); | ||
Andrew Shadura
|
r4159 | }); | ||
r3066 | return false; | |||
Mads Kiilerich
|
r4160 | }; | ||
r3066 | ||||
r3246 | var showRepoSize = function(target, repo_name, token){ | |||
var args= 'auth_token='+token; | ||||
r3696 | ||||
Andrew Shadura
|
r4159 | if(!$("#" + target).hasClass('loaded')){ | ||
$("#" + target).html(_TM['Loading ...']); | ||||
r3388 | var url = pyroutes.url('repo_size', {"repo_name":repo_name}); | |||
Andrew Shadura
|
r4159 | $.post(url, args, function(data) { | ||
$("#" + target).html(data); | ||||
$("#" + target).addClass('loaded'); | ||||
}); | ||||
r3247 | } | |||
r3696 | return false; | |||
Mads Kiilerich
|
r4160 | }; | ||
r2971 | ||||
/** | ||||
Mads Kiilerich
|
r4160 | * tooltips | ||
r2971 | */ | |||
Mads Kiilerich
|
r4160 | var tooltip_activate = function(){ | ||
$(document).ready(_init_tooltip); | ||||
}; | ||||
r2976 | ||||
Mads Kiilerich
|
r4160 | var _activate_tooltip = function($tt){ | ||
$tt.mouseover(_show_tooltip); | ||||
$tt.mousemove(_move_tooltip); | ||||
$tt.mouseout(_close_tooltip); | ||||
}; | ||||
r2971 | ||||
Mads Kiilerich
|
r4160 | var _init_tooltip = function(){ | ||
var $tipBox = $('#tip-box'); | ||||
if(!$tipBox.length){ | ||||
$tipBox = $('<div id="tip-box"></div>') | ||||
$(document.body).append($tipBox); | ||||
} | ||||
r2971 | ||||
Mads Kiilerich
|
r4160 | $tipBox.hide(); | ||
$tipBox.css('position', 'absolute'); | ||||
$tipBox.css('max-width', '600px'); | ||||
r2971 | ||||
Mads Kiilerich
|
r4160 | _activate_tooltip($('.tooltip')); | ||
}; | ||||
r2971 | ||||
Mads Kiilerich
|
r4160 | var _show_tooltip = function(e, tipText){ | ||
e.stopImmediatePropagation(); | ||||
var el = e.currentTarget; | ||||
if(tipText){ | ||||
// just use it | ||||
} else if(el.tagName.toLowerCase() === 'img'){ | ||||
tipText = el.alt ? el.alt : ''; | ||||
} else { | ||||
tipText = el.title ? el.title : ''; | ||||
} | ||||
r2971 | ||||
Mads Kiilerich
|
r4160 | if(tipText !== ''){ | ||
// save org title | ||||
$(el).attr('tt_title', tipText); | ||||
// reset title to not show org tooltips | ||||
$(el).attr('title', ''); | ||||
r2971 | ||||
Mads Kiilerich
|
r4160 | var $tipBox = $('#tip-box'); | ||
$tipBox.html(tipText); | ||||
$tipBox.css('display', 'block'); | ||||
} | ||||
}; | ||||
r2971 | ||||
Mads Kiilerich
|
r4160 | var _move_tooltip = function(e){ | ||
e.stopImmediatePropagation(); | ||||
var $tipBox = $('#tip-box'); | ||||
$tipBox.css('top', (e.pageY + 15) + 'px'); | ||||
$tipBox.css('left', (e.pageX + 15) + 'px'); | ||||
}; | ||||
r2971 | ||||
Mads Kiilerich
|
r4160 | var _close_tooltip = function(e){ | ||
e.stopImmediatePropagation(); | ||||
var $tipBox = $('#tip-box'); | ||||
$tipBox.hide(); | ||||
var el = e.currentTarget; | ||||
$(el).attr('title', $(el).attr('tt_title')); | ||||
}; | ||||
r1653 | ||||
/** | ||||
* Quick filter widget | ||||
r3696 | * | |||
r1653 | * @param target: filter input target | |||
* @param nodes: list of nodes in html we want to filter. | ||||
* @param display_element function that takes current node from nodes and | ||||
* does hide or show based on the node | ||||
*/ | ||||
Mads Kiilerich
|
r4160 | var q_filter = function(target, nodes, display_element){ | ||
r3696 | var nodes = nodes; | |||
Mads Kiilerich
|
r4160 | var $q_filter_field = $('#' + target); | ||
r3696 | var F = YAHOO.namespace(target); | |||
Mads Kiilerich
|
r4160 | $q_filter_field.keyup(function(e){ | ||
r3696 | clearTimeout(F.filterTimeout); | |||
Mads Kiilerich
|
r4160 | F.filterTimeout = setTimeout(F.updateFilter, 600); | ||
r3696 | }); | |||
F.filterTimeout = null; | ||||
r1653 | ||||
r3696 | F.updateFilter = function() { | |||
Mads Kiilerich
|
r4158 | // Reset timeout | ||
F.filterTimeout = null; | ||||
r3696 | ||||
Mads Kiilerich
|
r4158 | var obsolete = []; | ||
r1653 | ||||
Mads Kiilerich
|
r4160 | var req = $q_filter_field.val().toLowerCase(); | ||
r3696 | ||||
Mads Kiilerich
|
r4158 | var l = nodes.length; | ||
var i; | ||||
var showing = 0; | ||||
r3696 | ||||
Mads Kiilerich
|
r4160 | for (i=0; i<l; i++ ){ | ||
Mads Kiilerich
|
r4158 | var n = nodes[i]; | ||
var target_element = display_element(n) | ||||
if(req && n.innerHTML.toLowerCase().indexOf(req) == -1){ | ||||
Mads Kiilerich
|
r4160 | $(target_element).hide(); | ||
Mads Kiilerich
|
r4158 | } | ||
else{ | ||||
Mads Kiilerich
|
r4160 | $(target_element).show(); | ||
showing += 1; | ||||
Mads Kiilerich
|
r4158 | } | ||
} | ||||
r1653 | ||||
Mads Kiilerich
|
r4160 | $('#repo_count').html(showing); /* FIXME: don't hardcode */ | ||
r3696 | } | |||
r1699 | }; | |||
r1653 | ||||
Mads Kiilerich
|
r4160 | /* return jQuery expression with a tr with body in 3rd column and class cls and id named after the body */ | ||
var _table_tr = function(cls, body){ | ||||
// like: <div class="comment" id="comment-8" line="o92"><div class="comment-wrapp">... | ||||
// except new inlines which are different ... | ||||
var comment_id = ($(body).attr('id') || 'comment-new').split('comment-')[1]; | ||||
var tr_id = 'comment-tr-{0}'.format(comment_id); | ||||
return $(('<tr id="{0}" class="{1}">'+ | ||||
r3696 | '<td class="lineno-inline new-inline"></td>'+ | |||
'<td class="lineno-inline old-inline"></td>'+ | ||||
r2787 | '<td>{2}</td>'+ | |||
Mads Kiilerich
|
r4160 | '</tr>').format(tr_id, cls, body)); | ||
r1674 | }; | |||
r1653 | ||||
Mads Kiilerich
|
r4160 | /** return jQuery expression with new inline form based on template **/ | ||
var _createInlineForm = function(parent_tr, f_path, line) { | ||||
var $tmpl = $('#comment-inline-form-template').html().format(f_path, line); | ||||
var $form = _table_tr('comment-form-inline', $tmpl) | ||||
r2787 | ||||
r3696 | // create event for hide button | |||
Mads Kiilerich
|
r4160 | $form.find('.hide-inline-form').click(function(e) { | ||
r3696 | var newtr = e.currentTarget.parentNode.parentNode.parentNode.parentNode.parentNode; | |||
Mads Kiilerich
|
r4160 | if($(newtr).next().hasClass('inline-comments-button')){ | ||
$(newtr).next().show(); | ||||
r3696 | } | |||
Mads Kiilerich
|
r4160 | $(newtr).remove(); | ||
$(parent_tr).removeClass('form-open'); | ||||
$(parent_tr).removeClass('hl-comment'); | ||||
r3696 | }); | |||
Mads Kiilerich
|
r4160 | return $form | ||
r1699 | }; | |||
r2187 | ||||
/** | ||||
Mads Kiilerich
|
r4160 | * Inject inline comment for an given TR. This tr should always be a .line . | ||
* The form will be inject after any comments. | ||||
r2187 | */ | |||
r1705 | var injectInlineForm = function(tr){ | |||
Mads Kiilerich
|
r4160 | $tr = $(tr); | ||
if(!$tr.hasClass('line')){ | ||||
Mads Kiilerich
|
r4158 | return | ||
} | ||||
var submit_url = AJAX_COMMENT_URL; | ||||
Mads Kiilerich
|
r4160 | var $td = $tr.find('.code'); | ||
if($tr.hasClass('form-open') || $tr.hasClass('context') || $td.hasClass('no-comment')){ | ||||
Mads Kiilerich
|
r4158 | return | ||
} | ||||
Mads Kiilerich
|
r4160 | $tr.addClass('form-open hl-comment'); | ||
var $node = $tr.parent().parent().parent().find('.full_f_path'); | ||||
var f_path = $node.attr('path'); | ||||
var lineno = _getLineNo(tr); | ||||
var $form = _createInlineForm(tr, f_path, lineno, submit_url); | ||||
r3696 | ||||
Mads Kiilerich
|
r4160 | var $parent = $tr; | ||
while ($parent.next().hasClass('inline-comments')){ | ||||
var $parent = $parent.next(); | ||||
Mads Kiilerich
|
r4158 | } | ||
Mads Kiilerich
|
r4160 | $form.insertAfter($parent); | ||
var $overlay = $form.find('.overlay'); | ||||
var $inlineform = $form.find('.inline-form'); | ||||
r3696 | ||||
Mads Kiilerich
|
r4160 | $form.submit(function(e){ | ||
e.preventDefault(); | ||||
r3696 | ||||
Mads Kiilerich
|
r4158 | if(lineno === undefined){ | ||
alert('missing line !'); | ||||
return | ||||
} | ||||
if(f_path === undefined){ | ||||
alert('missing file path !'); | ||||
return | ||||
} | ||||
r2187 | ||||
Mads Kiilerich
|
r4160 | var text = $('#text_'+lineno).val(); | ||
Mads Kiilerich
|
r4158 | if(text == ""){ | ||
return | ||||
} | ||||
r3696 | ||||
Mads Kiilerich
|
r4160 | if ($overlay.hasClass('overlay')){ | ||
$overlay.css('width', $inlineform.offsetWidth + 'px'); | ||||
$overlay.css('height', $inlineform.offsetHeight + 'px'); | ||||
} | ||||
$overlay.addClass('submitting'); | ||||
r3695 | ||||
Mads Kiilerich
|
r4160 | var success = function(o){ | ||
$tr.removeClass('form-open'); | ||||
$form.remove(); | ||||
var json_data = JSON.parse(o.responseText); | ||||
_renderInlineComment(json_data); | ||||
}; | ||||
var postData = { | ||||
'text': text, | ||||
'f_path': f_path, | ||||
'line': lineno | ||||
}; | ||||
Mads Kiilerich
|
r4158 | ajaxPOST(submit_url, postData, success); | ||
}); | ||||
r3695 | ||||
Mads Kiilerich
|
r4160 | $('#preview-btn_'+lineno).click(function(e){ | ||
var text = $('#text_'+lineno).val(); | ||||
if(!text){ | ||||
Mads Kiilerich
|
r4158 | return | ||
} | ||||
Mads Kiilerich
|
r4160 | $('#preview-box_'+lineno).addClass('unloaded'); | ||
$('#preview-box_'+lineno).html(_TM['Loading ...']); | ||||
$('#edit-container_'+lineno).hide(); | ||||
$('#preview-container_'+lineno).show(); | ||||
r3696 | ||||
Mads Kiilerich
|
r4158 | var url = pyroutes.url('changeset_comment_preview', {'repo_name': REPO_NAME}); | ||
Mads Kiilerich
|
r4160 | var post_data = {'text': text}; | ||
ajaxPOST(url, post_data, function(o){ | ||||
$('#preview-box_'+lineno).html(o.responseText); | ||||
$('#preview-box_'+lineno).removeClass('unloaded'); | ||||
Mads Kiilerich
|
r4158 | }) | ||
}) | ||||
Mads Kiilerich
|
r4160 | $('#edit-btn_'+lineno).click(function(e){ | ||
$('#edit-container_'+lineno).show(); | ||||
$('#preview-container_'+lineno).hide(); | ||||
Mads Kiilerich
|
r4158 | }) | ||
r3696 | ||||
Mads Kiilerich
|
r4158 | setTimeout(function(){ | ||
// callbacks | ||||
tooltip_activate(); | ||||
MentionsAutoComplete('text_'+lineno, 'mentions_container_'+lineno, | ||||
r3696 | _USERS_AC_DATA, _GROUPS_AC_DATA); | |||
Mads Kiilerich
|
r4160 | $('#text_'+lineno).focus(); | ||
Mads Kiilerich
|
r4158 | },10) | ||
r1705 | }; | |||
r2187 | var deleteComment = function(comment_id){ | |||
r3696 | var url = AJAX_COMMENT_DELETE_URL.replace('__COMMENT_ID__',comment_id); | |||
r2187 | var postData = {'_method':'delete'}; | |||
var success = function(o){ | ||||
Mads Kiilerich
|
r4160 | var $deleted = $('#comment-tr-'+comment_id); | ||
var $prev = $deleted.prev('tr'); | ||||
$deleted.remove(); | ||||
_placeAddButton($prev); | ||||
r2187 | } | |||
ajaxPOST(url,postData,success); | ||||
} | ||||
Mads Kiilerich
|
r4160 | var _getLineNo = function(tr) { | ||
r3696 | var line; | |||
Mads Kiilerich
|
r4160 | var o = $(tr).children()[0].id.split('_'); | ||
var n = $(tr).children()[1].id.split('_'); | ||||
r1677 | ||||
r3696 | if (n.length >= 2) { | |||
line = n[n.length-1]; | ||||
} else if (o.length >= 2) { | ||||
line = o[o.length-1]; | ||||
} | ||||
r1677 | ||||
r3696 | return line | |||
r1699 | }; | |||
Mads Kiilerich
|
r4160 | var _placeAddButton = function($line_tr){ | ||
var $tr = $line_tr; | ||||
while ($tr.next().hasClass('inline-comments')){ | ||||
$tr.find('.add-comment').remove(); | ||||
$tr = $tr.next(); | ||||
r3696 | } | |||
Mads Kiilerich
|
r4160 | $tr.find('.add-comment').remove(); | ||
var label = TRANSLATION_MAP['Add another comment']; | ||||
Mads Kiilerich
|
r4171 | var $html_el = $('<div class="add-comment"><span class="btn btn-mini">{0}</span></div>'.format(label)); | ||
Mads Kiilerich
|
r4160 | $html_el.click(function(e) { | ||
injectInlineForm($line_tr); | ||||
}); | ||||
$tr.find('.comment').after($html_el); | ||||
}; | ||||
r2187 | ||||
/** | ||||
* Places the inline comment into the changeset block in proper line position | ||||
*/ | ||||
Mads Kiilerich
|
r4160 | var _placeInline = function(target_id, lineno, html){ | ||
var $td = $("#{0}_{1}".format(target_id, lineno)); | ||||
r3696 | ||||
Mads Kiilerich
|
r4158 | // check if there are comments already ! | ||
Mads Kiilerich
|
r4160 | var $line_tr = $td.parent(); // the tr | ||
var $after_tr = $line_tr; | ||||
while ($after_tr.next().hasClass('inline-comments')){ | ||||
$after_tr = $after_tr.next(); | ||||
Mads Kiilerich
|
r4158 | } | ||
// put in the comment at the bottom | ||||
Mads Kiilerich
|
r4160 | $after_tr.after(_table_tr('inline-comments', html)); | ||
r3696 | ||||
Mads Kiilerich
|
r4160 | // scan nodes, and attach add button to last one | ||
_placeAddButton($line_tr); | ||||
r2187 | } | |||
/** | ||||
* make a single inline comment and place it inside | ||||
*/ | ||||
Mads Kiilerich
|
r4160 | var _renderInlineComment = function(json_data){ | ||
var html = json_data['rendered_text']; | ||||
var lineno = json_data['line_no']; | ||||
var target_id = json_data['target_id']; | ||||
_placeInline(target_id, lineno, html); | ||||
r2187 | } | |||
/** | ||||
* Iterates over all the inlines, and places them inside proper blocks of data | ||||
*/ | ||||
var renderInlineComments = function(file_comments){ | ||||
r3696 | for (f in file_comments){ | |||
r2187 | // holding all comments for a FILE | |||
r3696 | var box = file_comments[f]; | |||
r2187 | ||||
Mads Kiilerich
|
r4160 | var target_id = $(box).attr('target_id'); | ||
// actual comments with line numbers | ||||
r2187 | var comments = box.children; | |||
for(var i=0; i<comments.length; i++){ | ||||
r3696 | var data = { | |||
'rendered_text': comments[i].outerHTML, | ||||
Mads Kiilerich
|
r4160 | 'line_no': $(comments[i]).attr('line'), | ||
r3696 | 'target_id': target_id | |||
} | ||||
Mads Kiilerich
|
r4160 | _renderInlineComment(data); | ||
r2187 | } | |||
r3696 | } | |||
r2187 | } | |||
Mads Kiilerich
|
r4160 | /* activate files.html stuff */ | ||
r2428 | var fileBrowserListeners = function(current_url, node_list_url, url_base){ | |||
Mads Kiilerich
|
r4160 | var current_url_branch = "?branch=__BRANCH__"; | ||
r1699 | ||||
Andrew Shadura
|
r4159 | $('#stay_at_branch').on('click',function(e){ | ||
Mads Kiilerich
|
r4160 | if(e.currentTarget.checked){ | ||
r3696 | var uri = current_url_branch; | |||
Mads Kiilerich
|
r4160 | uri = uri.replace('__BRANCH__',e.currentTarget.value); | ||
r3696 | window.location = uri; | |||
} | ||||
else{ | ||||
window.location = current_url; | ||||
} | ||||
}) | ||||
r1699 | ||||
Mads Kiilerich
|
r4160 | var $node_filter = $('#node_filter'); | ||
r3696 | ||||
Mads Kiilerich
|
r4160 | var filterTimeout = null; | ||
r3696 | var nodes = null; | |||
r1699 | ||||
Mads Kiilerich
|
r4160 | var initFilter = function(){ | ||
$('#node_filter_box_loading').show(); | ||||
$('#search_activate_id').hide(); | ||||
$('#add_node_id').hide(); | ||||
Mads Kiilerich
|
r4158 | YUC.initHeader('X-PARTIAL-XHR',true); | ||
YUC.asyncRequest('GET', node_list_url, { | ||||
success:function(o){ | ||||
nodes = JSON.parse(o.responseText).nodes; | ||||
Mads Kiilerich
|
r4160 | $('#node_filter_box_loading').hide(); | ||
$('#node_filter_box').show(); | ||||
$node_filter.focus(); | ||||
if($node_filter.hasClass('init')){ | ||||
$node_filter.val(''); | ||||
$node_filter.removeClass('init'); | ||||
Mads Kiilerich
|
r4158 | } | ||
}, | ||||
failure:function(o){ | ||||
console.log('failed to load'); | ||||
r3696 | } | |||
Mads Kiilerich
|
r4158 | },null); | ||
r3696 | } | |||
Mads Kiilerich
|
r4160 | var updateFilter = function(e) { | ||
r3696 | return function(){ | |||
// Reset timeout | ||||
Mads Kiilerich
|
r4160 | filterTimeout = null; | ||
var query = e.currentTarget.value.toLowerCase(); | ||||
r3696 | var match = []; | |||
var matches = 0; | ||||
var matches_max = 20; | ||||
if (query != ""){ | ||||
for(var i=0;i<nodes.length;i++){ | ||||
var pos = nodes[i].name.toLowerCase().indexOf(query) | ||||
if(query && pos != -1){ | ||||
matches++ | ||||
//show only certain amount to not kill browser | ||||
if (matches > matches_max){ | ||||
break; | ||||
} | ||||
var n = nodes[i].name; | ||||
var t = nodes[i].type; | ||||
var n_hl = n.substring(0,pos) | ||||
+"<b>{0}</b>".format(n.substring(pos,pos+query.length)) | ||||
+n.substring(pos+query.length) | ||||
var new_url = url_base.replace('__FPATH__',n); | ||||
match.push('<tr><td><a class="browser-{0}" href="{1}">{2}</a></td><td colspan="5"></td></tr>'.format(t,new_url,n_hl)); | ||||
} | ||||
if(match.length >= matches_max){ | ||||
match.push('<tr><td>{0}</td><td colspan="5"></td></tr>'.format(_TM['Search truncated'])); | ||||
Mads Kiilerich
|
r4166 | break; | ||
r3696 | } | |||
} | ||||
} | ||||
if(query != ""){ | ||||
Mads Kiilerich
|
r4160 | $('#tbody').hide(); | ||
$('#tbody_filtered').show(); | ||||
r1699 | ||||
r3696 | if (match.length==0){ | |||
match.push('<tr><td>{0}</td><td colspan="5"></td></tr>'.format(_TM['No matching files'])); | ||||
} | ||||
Mads Kiilerich
|
r4160 | $('#tbody_filtered').html(match.join("")); | ||
r3696 | } | |||
else{ | ||||
Mads Kiilerich
|
r4160 | $('#tbody').show(); | ||
$('#tbody_filtered').hide(); | ||||
r3696 | } | |||
} | ||||
}; | ||||
Mads Kiilerich
|
r4160 | $('#filter_activate').click(function(){ | ||
initFilter(); | ||||
}); | ||||
$node_filter.click(function(){ | ||||
if($node_filter.hasClass('init')){ | ||||
$node_filter.val(''); | ||||
$node_filter.removeClass('init'); | ||||
Mads Kiilerich
|
r4158 | } | ||
}); | ||||
Mads Kiilerich
|
r4160 | $node_filter.keyup(function(e){ | ||
clearTimeout(filterTimeout); | ||||
filterTimeout = setTimeout(updateFilter(e),600); | ||||
Mads Kiilerich
|
r4158 | }); | ||
r1699 | }; | |||
Mads Kiilerich
|
r4160 | var initCodeMirror = function(textarea_id, resetUrl){ | ||
var myCodeMirror = CodeMirror.fromTextArea($('#' + textarea_id)[0], { | ||||
mode: "null", | ||||
lineNumbers: true, | ||||
Mads Kiilerich
|
r4168 | indentUnit: 4, | ||
autofocus: true, | ||||
Mads Kiilerich
|
r4158 | }); | ||
Mads Kiilerich
|
r4160 | $('#reset').click(function(e){ | ||
window.location=resetUrl; | ||||
Mads Kiilerich
|
r4158 | }); | ||
r3696 | ||||
Mads Kiilerich
|
r4160 | $('#file_enable').click(function(){ | ||
$('#editor_container').show(); | ||||
$('#upload_file_container').hide(); | ||||
$('#filename_container').show(); | ||||
Mads Kiilerich
|
r4170 | $('#set_mode_header').show(); | ||
Mads Kiilerich
|
r4158 | }); | ||
r3696 | ||||
Mads Kiilerich
|
r4160 | $('#upload_file_enable').click(function(){ | ||
$('#editor_container').hide(); | ||||
$('#upload_file_container').show(); | ||||
$('#filename_container').hide(); | ||||
Mads Kiilerich
|
r4170 | $('#set_mode_header').hide(); | ||
Mads Kiilerich
|
r4158 | }); | ||
r4026 | ||||
return myCodeMirror | ||||
r1699 | }; | |||
r4026 | var setCodeMirrorMode = function(codeMirrorInstance, mode) { | |||
Mads Kiilerich
|
r4158 | codeMirrorInstance.setOption("mode", mode); | ||
CodeMirror.autoLoadMode(codeMirrorInstance, mode); | ||||
r4026 | } | |||
r1699 | ||||
Mads Kiilerich
|
r4160 | var _getIdentNode = function(n){ | ||
//iterate thrugh nodes until matching interesting node | ||||
r3696 | ||||
if (typeof n == 'undefined'){ | ||||
return -1 | ||||
} | ||||
if(typeof n.id != "undefined" && n.id.match('L[0-9]+')){ | ||||
Mads Kiilerich
|
r4158 | return n | ||
} | ||||
r3696 | else{ | |||
Mads Kiilerich
|
r4160 | return _getIdentNode(n.parentNode); | ||
r3696 | } | |||
r1699 | }; | |||
Mads Kiilerich
|
r4160 | /* generate links for multi line selects that can be shown by files.html page_highlights. | ||
* This is a mouseup handler for hlcode from CodeHtmlFormatter and pygmentize */ | ||||
Mads Kiilerich
|
r4158 | var getSelectionLink = function(e) { | ||
r3696 | //get selection from start/to nodes | |||
if (typeof window.getSelection != "undefined") { | ||||
s = window.getSelection(); | ||||
Mads Kiilerich
|
r4160 | from = _getIdentNode(s.anchorNode); | ||
till = _getIdentNode(s.focusNode); | ||||
r3696 | ||||
f_int = parseInt(from.id.replace('L','')); | ||||
t_int = parseInt(till.id.replace('L','')); | ||||
Mads Kiilerich
|
r4160 | var yoffset = 35; | ||
var ranges = [parseInt(from.id.replace('L','')), parseInt(till.id.replace('L',''))]; | ||||
if (ranges[0] > ranges[1]){ | ||||
r3696 | //highlight from bottom | |||
Mads Kiilerich
|
r4160 | yoffset = -yoffset; | ||
ranges = [ranges[1], ranges[0]]; | ||||
r3696 | } | |||
Mads Kiilerich
|
r4160 | var $hl_div = $('div#linktt'); | ||
r3696 | // if we select more than 2 lines | |||
if (ranges[0] != ranges[1]){ | ||||
Mads Kiilerich
|
r4160 | if ($hl_div.length) { | ||
$hl_div.html(''); | ||||
} else { | ||||
$hl_div = $('<div id="linktt" class="hl-tip-box">'); | ||||
$('body').prepend($hl_div); | ||||
r3696 | } | |||
r3081 | ||||
Mads Kiilerich
|
r4160 | $hl_div.append($('<a>').html(_TM['Selection link']).attr('href', location.href.substring(0, location.href.indexOf('#')) + '#L' + ranges[0] + '-'+ranges[1])); | ||
xy = $(till).offset(); | ||||
$hl_div.css('top', (xy.top + yoffset) + 'px').css('left', xy.left + 'px'); | ||||
$hl_div.show(); | ||||
r3696 | } | |||
else{ | ||||
Mads Kiilerich
|
r4160 | $hl_div.hide(); | ||
r3696 | } | |||
} | ||||
r1699 | }; | |||
r1712 | ||||
Mads Kiilerich
|
r4160 | var deleteNotification = function(url, notification_id, callbacks){ | ||
r3696 | var callback = { | |||
success:function(o){ | ||||
Mads Kiilerich
|
r4160 | $("#notification_"+notification_id).remove(); | ||
r3696 | _run_callbacks(callbacks); | |||
}, | ||||
failure:function(o){ | ||||
alert("error"); | ||||
}, | ||||
}; | ||||
r1712 | var postData = '_method=delete'; | |||
var sUrl = url.replace('__NOTIFICATION_ID__',notification_id); | ||||
r3696 | var request = YAHOO.util.Connect.asyncRequest('POST', sUrl, | |||
callback, postData); | ||||
}; | ||||
r1778 | ||||
Mads Kiilerich
|
r4160 | var readNotification = function(url, notification_id, callbacks){ | ||
r3696 | var callback = { | |||
success:function(o){ | ||||
Mads Kiilerich
|
r4160 | var $obj = $("#notification_"+notification_id); | ||
$obj.removeClass('unread'); | ||||
$obj.find('.read-notification').remove(); | ||||
r3696 | _run_callbacks(callbacks); | |||
}, | ||||
failure:function(o){ | ||||
alert("error"); | ||||
}, | ||||
}; | ||||
r2610 | var postData = '_method=put'; | |||
var sUrl = url.replace('__NOTIFICATION_ID__',notification_id); | ||||
r3696 | var request = YAHOO.util.Connect.asyncRequest('POST', sUrl, | |||
callback, postData); | ||||
}; | ||||
r1778 | ||||
r2142 | /** MEMBERS AUTOCOMPLETE WIDGET **/ | |||
Mads Kiilerich
|
r4160 | var _MembersAutoComplete = function (divid, cont, users_list, groups_list) { | ||
r2142 | var myUsers = users_list; | |||
var myGroups = groups_list; | ||||
// Define a custom search function for the DataSource of users | ||||
var matchUsers = function (sQuery) { | ||||
// Case insensitive matching | ||||
var query = sQuery.toLowerCase(); | ||||
var i = 0; | ||||
var l = myUsers.length; | ||||
var matches = []; | ||||
// Match against each name of each contact | ||||
for (; i < l; i++) { | ||||
contact = myUsers[i]; | ||||
r3696 | if (((contact.fname+"").toLowerCase().indexOf(query) > -1) || | |||
((contact.lname+"").toLowerCase().indexOf(query) > -1) || | ||||
((contact.nname) && ((contact.nname).toLowerCase().indexOf(query) > -1))) { | ||||
Mads Kiilerich
|
r4158 | matches[matches.length] = contact; | ||
} | ||||
r2142 | } | |||
return matches; | ||||
}; | ||||
Mads Kiilerich
|
r3417 | // Define a custom search function for the DataSource of userGroups | ||
r2142 | var matchGroups = function (sQuery) { | |||
// Case insensitive matching | ||||
var query = sQuery.toLowerCase(); | ||||
var i = 0; | ||||
var l = myGroups.length; | ||||
var matches = []; | ||||
// Match against each name of each contact | ||||
for (; i < l; i++) { | ||||
matched_group = myGroups[i]; | ||||
if (matched_group.grname.toLowerCase().indexOf(query) > -1) { | ||||
matches[matches.length] = matched_group; | ||||
} | ||||
} | ||||
return matches; | ||||
}; | ||||
//match all | ||||
var matchAll = function (sQuery) { | ||||
u = matchUsers(sQuery); | ||||
g = matchGroups(sQuery); | ||||
return u.concat(g); | ||||
}; | ||||
// DataScheme for members | ||||
var memberDS = new YAHOO.util.FunctionDataSource(matchAll); | ||||
memberDS.responseSchema = { | ||||
fields: ["id", "fname", "lname", "nname", "grname", "grmembers", "gravatar_lnk"] | ||||
}; | ||||
// DataScheme for owner | ||||
var ownerDS = new YAHOO.util.FunctionDataSource(matchUsers); | ||||
ownerDS.responseSchema = { | ||||
fields: ["id", "fname", "lname", "nname", "gravatar_lnk"] | ||||
}; | ||||
// Instantiate AutoComplete for perms | ||||
r2759 | var membersAC = new YAHOO.widget.AutoComplete(divid, cont, memberDS); | |||
r2142 | membersAC.useShadow = false; | |||
membersAC.resultTypeList = false; | ||||
r2611 | membersAC.animVert = false; | |||
r3696 | membersAC.animHoriz = false; | |||
r2611 | membersAC.animSpeed = 0.1; | |||
r2142 | ||||
// Instantiate AutoComplete for owner | ||||
var ownerAC = new YAHOO.widget.AutoComplete("user", "owner_container", ownerDS); | ||||
ownerAC.useShadow = false; | ||||
ownerAC.resultTypeList = false; | ||||
r2611 | ownerAC.animVert = false; | |||
ownerAC.animHoriz = false; | ||||
ownerAC.animSpeed = 0.1; | ||||
r2142 | ||||
// Helper highlight function for the formatter | ||||
var highlightMatch = function (full, snippet, matchindex) { | ||||
r3696 | return full.substring(0, matchindex) | |||
+ "<span class='match'>" | ||||
+ full.substr(matchindex, snippet.length) | ||||
r2142 | + "</span>" + full.substring(matchindex + snippet.length); | |||
}; | ||||
// Custom formatter to highlight the matching letters | ||||
var custom_formatter = function (oResultData, sQuery, sResultMatch) { | ||||
var query = sQuery.toLowerCase(); | ||||
var _gravatar = function(res, em, group){ | ||||
r3696 | if (group !== undefined){ | |||
em = '/images/icons/group.png' | ||||
} | ||||
tmpl = '<div class="ac-container-wrap"><img class="perm-gravatar-ac" src="{0}"/>{1}</div>' | ||||
return tmpl.format(em,res) | ||||
r2142 | } | |||
// group | ||||
if (oResultData.grname != undefined) { | ||||
var grname = oResultData.grname; | ||||
var grmembers = oResultData.grmembers; | ||||
var grnameMatchIndex = grname.toLowerCase().indexOf(query); | ||||
r2369 | var grprefix = "{0}: ".format(_TM['Group']); | |||
r2142 | var grsuffix = " (" + grmembers + " )"; | |||
r2369 | var grsuffix = " ({0} {1})".format(grmembers, _TM['members']); | |||
r2142 | ||||
if (grnameMatchIndex > -1) { | ||||
return _gravatar(grprefix + highlightMatch(grname, query, grnameMatchIndex) + grsuffix,null,true); | ||||
} | ||||
r3696 | return _gravatar(grprefix + oResultData.grname + grsuffix, null,true); | |||
r2142 | // Users | |||
r2369 | } else if (oResultData.nname != undefined) { | |||
var fname = oResultData.fname || ""; | ||||
var lname = oResultData.lname || ""; | ||||
var nname = oResultData.nname; | ||||
r3696 | ||||
r2369 | // Guard against null value | |||
var fnameMatchIndex = fname.toLowerCase().indexOf(query), | ||||
r2142 | lnameMatchIndex = lname.toLowerCase().indexOf(query), | |||
nnameMatchIndex = nname.toLowerCase().indexOf(query), | ||||
displayfname, displaylname, displaynname; | ||||
if (fnameMatchIndex > -1) { | ||||
displayfname = highlightMatch(fname, query, fnameMatchIndex); | ||||
} else { | ||||
displayfname = fname; | ||||
} | ||||
if (lnameMatchIndex > -1) { | ||||
displaylname = highlightMatch(lname, query, lnameMatchIndex); | ||||
} else { | ||||
displaylname = lname; | ||||
} | ||||
if (nnameMatchIndex > -1) { | ||||
displaynname = "(" + highlightMatch(nname, query, nnameMatchIndex) + ")"; | ||||
} else { | ||||
displaynname = nname ? "(" + nname + ")" : ""; | ||||
} | ||||
return _gravatar(displayfname + " " + displaylname + " " + displaynname, oResultData.gravatar_lnk); | ||||
} else { | ||||
return ''; | ||||
} | ||||
}; | ||||
membersAC.formatResult = custom_formatter; | ||||
ownerAC.formatResult = custom_formatter; | ||||
var myHandler = function (sType, aArgs) { | ||||
r3696 | var nextId = divid.split('perm_new_member_name_')[1]; | |||
r2142 | var myAC = aArgs[0]; // reference back to the AC instance | |||
var elLI = aArgs[1]; // reference to the selected LI element | ||||
var oData = aArgs[2]; // object literal of selected item's result data | ||||
//fill the autocomplete with value | ||||
if (oData.nname != undefined) { | ||||
//users | ||||
myAC.getInputEl().value = oData.nname; | ||||
Mads Kiilerich
|
r4160 | $('#perm_new_member_type_'+nextId).val('user'); | ||
r2142 | } else { | |||
//groups | ||||
myAC.getInputEl().value = oData.grname; | ||||
Mads Kiilerich
|
r4160 | $('#perm_new_member_type_'+nextId).val('users_group'); | ||
r2142 | } | |||
}; | ||||
membersAC.itemSelectEvent.subscribe(myHandler); | ||||
if(ownerAC.itemSelectEvent){ | ||||
r3696 | ownerAC.itemSelectEvent.subscribe(myHandler); | |||
r2142 | } | |||
return { | ||||
memberDS: memberDS, | ||||
ownerDS: ownerDS, | ||||
membersAC: membersAC, | ||||
ownerAC: ownerAC, | ||||
}; | ||||
} | ||||
r2369 | var MentionsAutoComplete = function (divid, cont, users_list, groups_list) { | |||
r2368 | var myUsers = users_list; | |||
var myGroups = groups_list; | ||||
// Define a custom search function for the DataSource of users | ||||
var matchUsers = function (sQuery) { | ||||
r3696 | var org_sQuery = sQuery; | |||
if(this.mentionQuery == null){ | ||||
return [] | ||||
} | ||||
sQuery = this.mentionQuery; | ||||
r2368 | // Case insensitive matching | |||
var query = sQuery.toLowerCase(); | ||||
var i = 0; | ||||
var l = myUsers.length; | ||||
var matches = []; | ||||
// Match against each name of each contact | ||||
for (; i < l; i++) { | ||||
contact = myUsers[i]; | ||||
r3696 | if (((contact.fname+"").toLowerCase().indexOf(query) > -1) || | |||
((contact.lname+"").toLowerCase().indexOf(query) > -1) || | ||||
((contact.nname) && ((contact.nname).toLowerCase().indexOf(query) > -1))) { | ||||
r2368 | matches[matches.length] = contact; | |||
} | ||||
} | ||||
return matches | ||||
}; | ||||
//match all | ||||
var matchAll = function (sQuery) { | ||||
u = matchUsers(sQuery); | ||||
return u | ||||
}; | ||||
// DataScheme for owner | ||||
var ownerDS = new YAHOO.util.FunctionDataSource(matchUsers); | ||||
r2369 | ||||
r2368 | ownerDS.responseSchema = { | |||
fields: ["id", "fname", "lname", "nname", "gravatar_lnk"] | ||||
}; | ||||
// Instantiate AutoComplete for mentions | ||||
var ownerAC = new YAHOO.widget.AutoComplete(divid, cont, ownerDS); | ||||
ownerAC.useShadow = false; | ||||
ownerAC.resultTypeList = false; | ||||
ownerAC.suppressInputUpdate = true; | ||||
r2611 | ownerAC.animVert = false; | |||
r3696 | ownerAC.animHoriz = false; | |||
r2611 | ownerAC.animSpeed = 0.1; | |||
r3696 | ||||
r2368 | // Helper highlight function for the formatter | |||
var highlightMatch = function (full, snippet, matchindex) { | ||||
r3696 | return full.substring(0, matchindex) | |||
+ "<span class='match'>" | ||||
+ full.substr(matchindex, snippet.length) | ||||
r2368 | + "</span>" + full.substring(matchindex + snippet.length); | |||
}; | ||||
// Custom formatter to highlight the matching letters | ||||
ownerAC.formatResult = function (oResultData, sQuery, sResultMatch) { | ||||
r3696 | var org_sQuery = sQuery; | |||
if(this.dataSource.mentionQuery != null){ | ||||
sQuery = this.dataSource.mentionQuery; | ||||
} | ||||
r2368 | ||||
var query = sQuery.toLowerCase(); | ||||
var _gravatar = function(res, em, group){ | ||||
r3696 | if (group !== undefined){ | |||
em = '/images/icons/group.png' | ||||
} | ||||
tmpl = '<div class="ac-container-wrap"><img class="perm-gravatar-ac" src="{0}"/>{1}</div>' | ||||
return tmpl.format(em,res) | ||||
r2368 | } | |||
r2369 | if (oResultData.nname != undefined) { | |||
var fname = oResultData.fname || ""; | ||||
var lname = oResultData.lname || ""; | ||||
var nname = oResultData.nname; | ||||
r3696 | ||||
r2369 | // Guard against null value | |||
var fnameMatchIndex = fname.toLowerCase().indexOf(query), | ||||
r2368 | lnameMatchIndex = lname.toLowerCase().indexOf(query), | |||
nnameMatchIndex = nname.toLowerCase().indexOf(query), | ||||
displayfname, displaylname, displaynname; | ||||
if (fnameMatchIndex > -1) { | ||||
displayfname = highlightMatch(fname, query, fnameMatchIndex); | ||||
} else { | ||||
displayfname = fname; | ||||
} | ||||
if (lnameMatchIndex > -1) { | ||||
displaylname = highlightMatch(lname, query, lnameMatchIndex); | ||||
} else { | ||||
displaylname = lname; | ||||
} | ||||
if (nnameMatchIndex > -1) { | ||||
displaynname = "(" + highlightMatch(nname, query, nnameMatchIndex) + ")"; | ||||
} else { | ||||
displaynname = nname ? "(" + nname + ")" : ""; | ||||
} | ||||
return _gravatar(displayfname + " " + displaylname + " " + displaynname, oResultData.gravatar_lnk); | ||||
} else { | ||||
return ''; | ||||
} | ||||
}; | ||||
if(ownerAC.itemSelectEvent){ | ||||
r3696 | ownerAC.itemSelectEvent.subscribe(function (sType, aArgs) { | |||
r2368 | var myAC = aArgs[0]; // reference back to the AC instance | |||
var elLI = aArgs[1]; // reference to the selected LI element | ||||
var oData = aArgs[2]; // object literal of selected item's result data | ||||
//fill the autocomplete with value | ||||
if (oData.nname != undefined) { | ||||
//users | ||||
r3696 | //Replace the mention name with replaced | |||
var re = new RegExp(); | ||||
var org = myAC.getInputEl().value; | ||||
var chunks = myAC.dataSource.chunks | ||||
// replace middle chunk(the search term) with actuall match | ||||
chunks[1] = chunks[1].replace('@'+myAC.dataSource.mentionQuery, | ||||
'@'+oData.nname+' '); | ||||
r2368 | myAC.getInputEl().value = chunks.join('') | |||
Mads Kiilerich
|
r4160 | myAC.getInputEl().focus(); // Y U NO WORK !? | ||
r2368 | } else { | |||
//groups | ||||
myAC.getInputEl().value = oData.grname; | ||||
Mads Kiilerich
|
r4160 | $('#perm_new_member_type').val('users_group'); | ||
r2368 | } | |||
}); | ||||
} | ||||
// in this keybuffer we will gather current value of search ! | ||||
// since we need to get this just when someone does `@` then we do the | ||||
// search | ||||
ownerAC.dataSource.chunks = []; | ||||
ownerAC.dataSource.mentionQuery = null; | ||||
ownerAC.get_mention = function(msg, max_pos) { | ||||
r3696 | var org = msg; | |||
var re = new RegExp('(?:^@|\s@)([a-zA-Z0-9]{1}[a-zA-Z0-9\-\_\.]+)$') | ||||
var chunks = []; | ||||
// cut first chunk until curret pos | ||||
var to_max = msg.substr(0, max_pos); | ||||
var at_pos = Math.max(0,to_max.lastIndexOf('@')-1); | ||||
var msg2 = to_max.substr(at_pos); | ||||
r2368 | ||||
r3696 | chunks.push(org.substr(0,at_pos))// prefix chunk | |||
chunks.push(msg2) // search chunk | ||||
chunks.push(org.substr(max_pos)) // postfix chunk | ||||
r2368 | ||||
r3696 | // clean up msg2 for filtering and regex match | |||
var msg2 = msg2.lstrip(' ').lstrip('\n'); | ||||
r2368 | ||||
r3696 | if(re.test(msg2)){ | |||
var unam = re.exec(msg2)[1]; | ||||
return [unam, chunks]; | ||||
} | ||||
return [null, null]; | ||||
r2612 | }; | |||
r3696 | ||||
Bradley M. Kuhn
|
r4173 | $divid = $('#'+divid); | ||
$divid.keyup(function(e){ | ||||
var currentMessage = $divid.val(); | ||||
var currentCaretPosition = $divid[0].selectionStart; | ||||
r3696 | ||||
var unam = ownerAC.get_mention(currentMessage, currentCaretPosition); | ||||
var curr_search = null; | ||||
if(unam[0]){ | ||||
curr_search = unam[0]; | ||||
} | ||||
ownerAC.dataSource.chunks = unam[1]; | ||||
ownerAC.dataSource.mentionQuery = curr_search; | ||||
Bradley M. Kuhn
|
r4173 | }); | ||
r2368 | } | |||
r3388 | var addReviewMember = function(id,fname,lname,nname,gravatar_link){ | |||
Mads Kiilerich
|
r4172 | var displayname = "{0} {1} ({2})".format(fname, lname, nname); | ||
var element = ( | ||||
' <li id="reviewer_{2}">\n'+ | ||||
' <div class="reviewers_member">\n'+ | ||||
' <div class="reviewer_status tooltip" title="not_reviewed">\n'+ | ||||
' <img src="/images/icons/flag_status_not_reviewed.png"/>\n'+ | ||||
' </div>\n'+ | ||||
' <div class="reviewer_gravatar gravatar"><img alt="gravatar" src="{0}"/> </div>\n'+ | ||||
' <div style="float:left;">{1}</div>\n'+ | ||||
' <input type="hidden" value="{2}" name="review_members" />\n'+ | ||||
' <div class="reviewer_member_remove action_button" onclick="removeReviewMember({2})">\n'+ | ||||
' <i class="icon-remove-sign" style="color: #FF4444;"></i>\n'+ | ||||
' </div>\n'+ | ||||
' </div>\n'+ | ||||
' </li>\n' | ||||
).format(gravatar_link, displayname, id); | ||||
r3696 | // check if we don't have this ID already in | |||
var ids = []; | ||||
Mads Kiilerich
|
r4160 | $('#review_members').find('li').each(function() { | ||
ids.push(this.id); | ||||
}); | ||||
r3696 | if(ids.indexOf('reviewer_'+id) == -1){ | |||
//only add if it's not there | ||||
Mads Kiilerich
|
r4160 | $('#review_members').append(element); | ||
r3696 | } | |||
r3388 | } | |||
var removeReviewMember = function(reviewer_id, repo_name, pull_request_id){ | ||||
Mads Kiilerich
|
r4160 | $('#reviewer_{0}'.format(reviewer_id)).remove(); | ||
r3388 | } | |||
Mads Kiilerich
|
r4160 | /* handle "Save Changes" of addReviewMember and removeReviewMember on PR */ | ||
r3388 | var updateReviewers = function(reviewers_ids, repo_name, pull_request_id){ | |||
r3696 | if (reviewers_ids === undefined){ | |||
Mads Kiilerich
|
r4158 | var reviewers_ids = []; | ||
Mads Kiilerich
|
r4160 | $('#review_members').find('input').each(function(){ | ||
reviewers_ids.push(this.value); | ||||
}); | ||||
r3696 | } | |||
var url = pyroutes.url('pullrequest_update', {"repo_name":repo_name, | ||||
"pull_request_id": pull_request_id}); | ||||
var postData = {'_method':'put', | ||||
'reviewers_ids': reviewers_ids}; | ||||
var success = function(o){ | ||||
window.location.reload(); | ||||
} | ||||
ajaxPOST(url,postData,success); | ||||
r3388 | } | |||
r2368 | ||||
Mads Kiilerich
|
r4160 | /* activate auto completion of users and groups ... but only used for users as PR reviewers */ | ||
r2612 | var PullRequestAutoComplete = function (divid, cont, users_list, groups_list) { | |||
var myUsers = users_list; | ||||
var myGroups = groups_list; | ||||
// Define a custom search function for the DataSource of users | ||||
var matchUsers = function (sQuery) { | ||||
// Case insensitive matching | ||||
var query = sQuery.toLowerCase(); | ||||
var i = 0; | ||||
var l = myUsers.length; | ||||
var matches = []; | ||||
// Match against each name of each contact | ||||
for (; i < l; i++) { | ||||
contact = myUsers[i]; | ||||
r3696 | if (((contact.fname+"").toLowerCase().indexOf(query) > -1) || | |||
((contact.lname+"").toLowerCase().indexOf(query) > -1) || | ||||
((contact.nname) && ((contact.nname).toLowerCase().indexOf(query) > -1))) { | ||||
Mads Kiilerich
|
r4158 | matches[matches.length] = contact; | ||
} | ||||
r2612 | } | |||
return matches; | ||||
}; | ||||
Mads Kiilerich
|
r3417 | // Define a custom search function for the DataSource of userGroups | ||
r2612 | var matchGroups = function (sQuery) { | |||
// Case insensitive matching | ||||
var query = sQuery.toLowerCase(); | ||||
var i = 0; | ||||
var l = myGroups.length; | ||||
var matches = []; | ||||
// Match against each name of each contact | ||||
for (; i < l; i++) { | ||||
matched_group = myGroups[i]; | ||||
if (matched_group.grname.toLowerCase().indexOf(query) > -1) { | ||||
matches[matches.length] = matched_group; | ||||
} | ||||
} | ||||
return matches; | ||||
}; | ||||
//match all | ||||
var matchAll = function (sQuery) { | ||||
u = matchUsers(sQuery); | ||||
return u | ||||
}; | ||||
// DataScheme for owner | ||||
var ownerDS = new YAHOO.util.FunctionDataSource(matchUsers); | ||||
ownerDS.responseSchema = { | ||||
fields: ["id", "fname", "lname", "nname", "gravatar_lnk"] | ||||
}; | ||||
// Instantiate AutoComplete for mentions | ||||
var reviewerAC = new YAHOO.widget.AutoComplete(divid, cont, ownerDS); | ||||
reviewerAC.useShadow = false; | ||||
reviewerAC.resultTypeList = false; | ||||
reviewerAC.suppressInputUpdate = true; | ||||
reviewerAC.animVert = false; | ||||
r3696 | reviewerAC.animHoriz = false; | |||
r2612 | reviewerAC.animSpeed = 0.1; | |||
r3696 | ||||
r2612 | // Helper highlight function for the formatter | |||
var highlightMatch = function (full, snippet, matchindex) { | ||||
r3696 | return full.substring(0, matchindex) | |||
+ "<span class='match'>" | ||||
+ full.substr(matchindex, snippet.length) | ||||
r2612 | + "</span>" + full.substring(matchindex + snippet.length); | |||
}; | ||||
// Custom formatter to highlight the matching letters | ||||
reviewerAC.formatResult = function (oResultData, sQuery, sResultMatch) { | ||||
r3696 | var org_sQuery = sQuery; | |||
if(this.dataSource.mentionQuery != null){ | ||||
sQuery = this.dataSource.mentionQuery; | ||||
} | ||||
r2612 | ||||
var query = sQuery.toLowerCase(); | ||||
var _gravatar = function(res, em, group){ | ||||
r3696 | if (group !== undefined){ | |||
em = '/images/icons/group.png' | ||||
} | ||||
tmpl = '<div class="ac-container-wrap"><img class="perm-gravatar-ac" src="{0}"/>{1}</div>' | ||||
return tmpl.format(em,res) | ||||
r2612 | } | |||
if (oResultData.nname != undefined) { | ||||
var fname = oResultData.fname || ""; | ||||
var lname = oResultData.lname || ""; | ||||
var nname = oResultData.nname; | ||||
r3696 | ||||
r2612 | // Guard against null value | |||
var fnameMatchIndex = fname.toLowerCase().indexOf(query), | ||||
lnameMatchIndex = lname.toLowerCase().indexOf(query), | ||||
nnameMatchIndex = nname.toLowerCase().indexOf(query), | ||||
displayfname, displaylname, displaynname; | ||||
if (fnameMatchIndex > -1) { | ||||
displayfname = highlightMatch(fname, query, fnameMatchIndex); | ||||
} else { | ||||
displayfname = fname; | ||||
} | ||||
if (lnameMatchIndex > -1) { | ||||
displaylname = highlightMatch(lname, query, lnameMatchIndex); | ||||
} else { | ||||
displaylname = lname; | ||||
} | ||||
if (nnameMatchIndex > -1) { | ||||
displaynname = "(" + highlightMatch(nname, query, nnameMatchIndex) + ")"; | ||||
} else { | ||||
displaynname = nname ? "(" + nname + ")" : ""; | ||||
} | ||||
return _gravatar(displayfname + " " + displaylname + " " + displaynname, oResultData.gravatar_lnk); | ||||
} else { | ||||
return ''; | ||||
} | ||||
}; | ||||
r3696 | ||||
r2612 | //members cache to catch duplicates | |||
reviewerAC.dataSource.cache = []; | ||||
// hack into select event | ||||
if(reviewerAC.itemSelectEvent){ | ||||
r3696 | reviewerAC.itemSelectEvent.subscribe(function (sType, aArgs) { | |||
r2612 | ||||
var myAC = aArgs[0]; // reference back to the AC instance | ||||
var elLI = aArgs[1]; // reference to the selected LI element | ||||
var oData = aArgs[2]; // object literal of selected item's result data | ||||
r3696 | ||||
r2612 | //fill the autocomplete with value | |||
if (oData.nname != undefined) { | ||||
r3696 | addReviewMember(oData.id, oData.fname, oData.lname, oData.nname, | |||
oData.gravatar_lnk); | ||||
myAC.dataSource.cache.push(oData.id); | ||||
Mads Kiilerich
|
r4160 | $('#user').val(''); | ||
r2612 | } | |||
r3696 | }); | |||
r2612 | } | |||
} | ||||
r1778 | /** | |||
Mads Kiilerich
|
r4160 | * Activate .quick_repo_menu | ||
r1779 | */ | |||
var quick_repo_menu = function(){ | ||||
Mads Kiilerich
|
r4160 | $(".quick_repo_menu").mouseenter(function(e) { | ||
var $menu = $(e.currentTarget).children().first().children().first(); | ||||
if($menu.hasClass('hidden')){ | ||||
$menu.removeClass('hidden').addClass('active'); | ||||
$(e.currentTarget).removeClass('hidden').addClass('active'); | ||||
r2088 | } | |||
}) | ||||
Mads Kiilerich
|
r4160 | $(".quick_repo_menu").mouseleave(function(e) { | ||
var $menu = $(e.currentTarget).children().first().children().first(); | ||||
if($menu.hasClass('active')){ | ||||
$menu.removeClass('active').addClass('hidden'); | ||||
$(e.currentTarget).removeClass('active').addClass('hidden'); | ||||
r2088 | } | |||
}) | ||||
r1779 | }; | |||
/** | ||||
r1778 | * TABLE SORTING | |||
*/ | ||||
var revisionSort = function(a, b, desc, field) { | ||||
Mads Kiilerich
|
r4165 | var a_ = parseInt(a.getData('last_rev_raw') || 0); | ||
var b_ = parseInt(b.getData('last_rev_raw') || 0); | ||||
r3696 | ||||
Mads Kiilerich
|
r4160 | return YAHOO.util.Sort.compare(a_, b_, desc); | ||
r1778 | }; | |||
Mads Kiilerich
|
r4157 | |||
r1778 | var ageSort = function(a, b, desc, field) { | |||
Mads Kiilerich
|
r4160 | // data is like: <span class="tooltip" date="2014-06-04 18:18:55.325474" title="Wed, 04 Jun 2014 18:18:55">1 day and 23 hours ago</span> | ||
var a_ = $(a.getData(field)).attr('date'); | ||||
var b_ = $(b.getData(field)).attr('date'); | ||||
r3696 | ||||
Mads Kiilerich
|
r4160 | return YAHOO.util.Sort.compare(a_, b_, desc); | ||
r1778 | }; | |||
r2699 | var lastLoginSort = function(a, b, desc, field) { | |||
Mads Kiilerich
|
r4165 | var a_ = parseFloat(a.getData('last_login_raw') || 0); | ||
var b_ = parseFloat(b.getData('last_login_raw') || 0); | ||||
r3696 | ||||
Mads Kiilerich
|
r4160 | return YAHOO.util.Sort.compare(a_, b_, desc); | ||
r2699 | }; | |||
r1778 | var nameSort = function(a, b, desc, field) { | |||
r4087 | var a_ = a.getData('raw_name') || 0; | |||
var b_ = b.getData('raw_name') || 0; | ||||
r3696 | ||||
Mads Kiilerich
|
r4160 | return YAHOO.util.Sort.compare(a_, b_, desc); | ||
r1778 | }; | |||
r1782 | var dateSort = function(a, b, desc, field) { | |||
Mads Kiilerich
|
r4165 | var a_ = parseFloat(a.getData('raw_date') || 0); | ||
var b_ = parseFloat(b.getData('raw_date') || 0); | ||||
r3696 | ||||
Mads Kiilerich
|
r4160 | return YAHOO.util.Sort.compare(a_, b_, desc); | ||
r2394 | }; | |||
r2759 | var addPermAction = function(_html, users_list, groups_list){ | |||
Mads Kiilerich
|
r4160 | var $last_node = $('.last_new_member').last(); // empty tr between last and add | ||
var next_id = $('.new_members').length; | ||||
$last_node.before($('<tr class="new_members">').append(_html.format(next_id))); | ||||
_MembersAutoComplete("perm_new_member_name_"+next_id, | ||||
"perm_container_"+next_id, users_list, groups_list); | ||||
r2759 | } | |||
Mads Kiilerich
|
r4157 | |||
r3715 | function ajaxActionRevokePermission(url, obj_id, obj_type, field_id, extra_data) { | |||
var callback = { | ||||
success: function (o) { | ||||
Mads Kiilerich
|
r4160 | $('#' + field_id).remove(); | ||
r3715 | }, | |||
failure: function (o) { | ||||
alert(_TM['Failed to remoke permission'] + ": " + o.status); | ||||
}, | ||||
}; | ||||
query_params = { | ||||
'_method': 'delete' | ||||
} | ||||
// put extra data into POST | ||||
if (extra_data !== undefined && (typeof extra_data === 'object')){ | ||||
for(k in extra_data){ | ||||
query_params[k] = extra_data[k]; | ||||
} | ||||
} | ||||
r2394 | ||||
r3715 | if (obj_type=='user'){ | |||
query_params['user_id'] = obj_id; | ||||
query_params['obj_type'] = 'user'; | ||||
} | ||||
else if (obj_type=='user_group'){ | ||||
query_params['user_group_id'] = obj_id; | ||||
query_params['obj_type'] = 'user_group'; | ||||
} | ||||
var request = YAHOO.util.Connect.asyncRequest('POST', url, callback, | ||||
Mads Kiilerich
|
r4160 | _toQueryString(query_params)); | ||
r3715 | }; | |||
Mads Kiilerich
|
r4157 | |||
r2394 | /* Multi selectors */ | |||
var MultiSelectWidget = function(selected_id, available_id, form_id){ | ||||
Mads Kiilerich
|
r4160 | var $availableselect = $('#' + available_id); | ||
var $selectedselect = $('#' + selected_id); | ||||
r3696 | ||||
Mads Kiilerich
|
r4160 | //fill available only with those not in selected | ||
var $selectedoptions = $selectedselect.children('option'); | ||||
$availableselect.children('option').filter(function(i, e){ | ||||
for(var j = 0, node; node = $selectedoptions[j]; j++){ | ||||
if(node.value == e.value){ | ||||
return true; | ||||
r3696 | } | |||
} | ||||
Mads Kiilerich
|
r4160 | return false; | ||
}).remove(); | ||||
r3696 | ||||
Mads Kiilerich
|
r4160 | $('#add_element').click(function(e){ | ||
$selectedselect.append($availableselect.children('option:selected')); | ||||
}); | ||||
$('#remove_element').click(function(e){ | ||||
$availableselect.append($selectedselect.children('option:selected')); | ||||
}); | ||||
$('#add_all_elements').click(function(e){ | ||||
$selectedselect.append($availableselect.children('option')); | ||||
}); | ||||
$('#remove_all_elements').click(function(e){ | ||||
$availableselect.append($selectedselect.children('option')); | ||||
}); | ||||
r3696 | ||||
Mads Kiilerich
|
r4160 | $('#'+form_id).submit(function(){ | ||
$selectedselect.children('option').each(function(i, e){ | ||||
e.selected = 'selected'; | ||||
}); | ||||
r3696 | }); | |||
Dies Koper
|
r2723 | } | ||
r3419 | ||||
r3784 | // custom paginator | |||
var YUI_paginator = function(links_per_page, containers){ | ||||
r3776 | ||||
(function () { | ||||
var Paginator = YAHOO.widget.Paginator, | ||||
l = YAHOO.lang, | ||||
setId = YAHOO.util.Dom.generateId; | ||||
Paginator.ui.MyFirstPageLink = function (p) { | ||||
this.paginator = p; | ||||
p.subscribe('recordOffsetChange',this.update,this,true); | ||||
p.subscribe('rowsPerPageChange',this.update,this,true); | ||||
p.subscribe('totalRecordsChange',this.update,this,true); | ||||
p.subscribe('destroy',this.destroy,this,true); | ||||
// TODO: make this work | ||||
p.subscribe('firstPageLinkLabelChange',this.update,this,true); | ||||
p.subscribe('firstPageLinkClassChange',this.update,this,true); | ||||
}; | ||||
Paginator.ui.MyFirstPageLink.init = function (p) { | ||||
p.setAttributeConfig('firstPageLinkLabel', { | ||||
value : 1, | ||||
validator : l.isString | ||||
}); | ||||
p.setAttributeConfig('firstPageLinkClass', { | ||||
value : 'yui-pg-first', | ||||
validator : l.isString | ||||
}); | ||||
p.setAttributeConfig('firstPageLinkTitle', { | ||||
value : 'First Page', | ||||
validator : l.isString | ||||
}); | ||||
}; | ||||
// Instance members and methods | ||||
Paginator.ui.MyFirstPageLink.prototype = { | ||||
current : null, | ||||
leftmost_page: null, | ||||
rightmost_page: null, | ||||
link : null, | ||||
span : null, | ||||
dotdot : null, | ||||
getPos : function(cur_page, max_page, items){ | ||||
var edge = parseInt(items / 2) + 1; | ||||
if (cur_page <= edge){ | ||||
var radius = Math.max(parseInt(items / 2), items - cur_page); | ||||
} | ||||
else if ((max_page - cur_page) < edge) { | ||||
var radius = (items - 1) - (max_page - cur_page); | ||||
} | ||||
else{ | ||||
var radius = parseInt(items / 2); | ||||
} | ||||
var left = Math.max(1, (cur_page - (radius))) | ||||
var right = Math.min(max_page, cur_page + (radius)) | ||||
return [left, cur_page, right] | ||||
}, | ||||
render : function (id_base) { | ||||
var p = this.paginator, | ||||
c = p.get('firstPageLinkClass'), | ||||
label = p.get('firstPageLinkLabel'), | ||||
title = p.get('firstPageLinkTitle'); | ||||
this.link = document.createElement('a'); | ||||
r3784 | this.span = document.createElement('span'); | |||
Mads Kiilerich
|
r4160 | $(this.span).hide(); | ||
r3776 | ||||
var _pos = this.getPos(p.getCurrentPage(), p.getTotalPages(), 5); | ||||
this.leftmost_page = _pos[0]; | ||||
this.rightmost_page = _pos[2]; | ||||
setId(this.link, id_base + '-first-link'); | ||||
this.link.href = '#'; | ||||
this.link.className = c; | ||||
this.link.innerHTML = label; | ||||
this.link.title = title; | ||||
Mads Kiilerich
|
r4160 | YUE.on(this.link,'click',this.onClick,this,true); | ||
r3776 | ||||
setId(this.span, id_base + '-first-span'); | ||||
this.span.className = c; | ||||
this.span.innerHTML = label; | ||||
this.current = p.getCurrentPage() > 1 ? this.link : this.span; | ||||
return this.current; | ||||
}, | ||||
update : function (e) { | ||||
var p = this.paginator; | ||||
var _pos = this.getPos(p.getCurrentPage(), p.getTotalPages(), 5); | ||||
this.leftmost_page = _pos[0]; | ||||
this.rightmost_page = _pos[2]; | ||||
if (e && e.prevValue === e.newValue) { | ||||
return; | ||||
} | ||||
var par = this.current ? this.current.parentNode : null; | ||||
if (this.leftmost_page > 1) { | ||||
if (par && this.current === this.span) { | ||||
par.replaceChild(this.link,this.current); | ||||
this.current = this.link; | ||||
} | ||||
} else { | ||||
if (par && this.current === this.link) { | ||||
par.replaceChild(this.span,this.current); | ||||
this.current = this.span; | ||||
} | ||||
} | ||||
}, | ||||
destroy : function () { | ||||
Mads Kiilerich
|
r4160 | YUE.purgeElement(this.link); | ||
r3776 | this.current.parentNode.removeChild(this.current); | |||
this.link = this.span = null; | ||||
}, | ||||
onClick : function (e) { | ||||
Mads Kiilerich
|
r4160 | YUE.stopEvent(e); | ||
r3776 | this.paginator.setPage(1); | |||
} | ||||
}; | ||||
})(); | ||||
Mads Kiilerich
|
r4157 | |||
r3776 | (function () { | |||
var Paginator = YAHOO.widget.Paginator, | ||||
l = YAHOO.lang, | ||||
setId = YAHOO.util.Dom.generateId; | ||||
Paginator.ui.MyLastPageLink = function (p) { | ||||
this.paginator = p; | ||||
p.subscribe('recordOffsetChange',this.update,this,true); | ||||
p.subscribe('rowsPerPageChange',this.update,this,true); | ||||
p.subscribe('totalRecordsChange',this.update,this,true); | ||||
p.subscribe('destroy',this.destroy,this,true); | ||||
// TODO: make this work | ||||
p.subscribe('lastPageLinkLabelChange',this.update,this,true); | ||||
p.subscribe('lastPageLinkClassChange', this.update,this,true); | ||||
}; | ||||
Paginator.ui.MyLastPageLink.init = function (p) { | ||||
p.setAttributeConfig('lastPageLinkLabel', { | ||||
value : -1, | ||||
validator : l.isString | ||||
}); | ||||
p.setAttributeConfig('lastPageLinkClass', { | ||||
value : 'yui-pg-last', | ||||
validator : l.isString | ||||
}); | ||||
p.setAttributeConfig('lastPageLinkTitle', { | ||||
value : 'Last Page', | ||||
validator : l.isString | ||||
}); | ||||
}; | ||||
Paginator.ui.MyLastPageLink.prototype = { | ||||
current : null, | ||||
leftmost_page: null, | ||||
rightmost_page: null, | ||||
link : null, | ||||
span : null, | ||||
dotdot : null, | ||||
na : null, | ||||
getPos : function(cur_page, max_page, items){ | ||||
var edge = parseInt(items / 2) + 1; | ||||
if (cur_page <= edge){ | ||||
var radius = Math.max(parseInt(items / 2), items - cur_page); | ||||
} | ||||
else if ((max_page - cur_page) < edge) { | ||||
var radius = (items - 1) - (max_page - cur_page); | ||||
} | ||||
else{ | ||||
var radius = parseInt(items / 2); | ||||
} | ||||
var left = Math.max(1, (cur_page - (radius))) | ||||
var right = Math.min(max_page, cur_page + (radius)) | ||||
return [left, cur_page, right] | ||||
}, | ||||
render : function (id_base) { | ||||
var p = this.paginator, | ||||
c = p.get('lastPageLinkClass'), | ||||
label = p.get('lastPageLinkLabel'), | ||||
last = p.getTotalPages(), | ||||
title = p.get('lastPageLinkTitle'); | ||||
var _pos = this.getPos(p.getCurrentPage(), p.getTotalPages(), 5); | ||||
this.leftmost_page = _pos[0]; | ||||
this.rightmost_page = _pos[2]; | ||||
this.link = document.createElement('a'); | ||||
r3784 | this.span = document.createElement('span'); | |||
Mads Kiilerich
|
r4160 | $(this.span).hide(); | ||
r3784 | ||||
r3776 | this.na = this.span.cloneNode(false); | |||
setId(this.link, id_base + '-last-link'); | ||||
this.link.href = '#'; | ||||
this.link.className = c; | ||||
this.link.innerHTML = label; | ||||
this.link.title = title; | ||||
Mads Kiilerich
|
r4160 | YUE.on(this.link,'click',this.onClick,this,true); | ||
r3776 | ||||
setId(this.span, id_base + '-last-span'); | ||||
this.span.className = c; | ||||
this.span.innerHTML = label; | ||||
setId(this.na, id_base + '-last-na'); | ||||
if (this.rightmost_page < p.getTotalPages()){ | ||||
this.current = this.link; | ||||
} | ||||
else{ | ||||
this.current = this.span; | ||||
} | ||||
this.current.innerHTML = p.getTotalPages(); | ||||
return this.current; | ||||
}, | ||||
update : function (e) { | ||||
var p = this.paginator; | ||||
var _pos = this.getPos(p.getCurrentPage(), p.getTotalPages(), 5); | ||||
this.leftmost_page = _pos[0]; | ||||
this.rightmost_page = _pos[2]; | ||||
if (e && e.prevValue === e.newValue) { | ||||
return; | ||||
} | ||||
var par = this.current ? this.current.parentNode : null, | ||||
after = this.link; | ||||
if (par) { | ||||
// only show the last page if the rightmost one is | ||||
// lower, so we don't have doubled entries at the end | ||||
if (!(this.rightmost_page < p.getTotalPages())){ | ||||
after = this.span | ||||
} | ||||
if (this.current !== after) { | ||||
par.replaceChild(after,this.current); | ||||
this.current = after; | ||||
} | ||||
} | ||||
this.current.innerHTML = this.paginator.getTotalPages(); | ||||
}, | ||||
destroy : function () { | ||||
Mads Kiilerich
|
r4160 | YUE.purgeElement(this.link); | ||
r3776 | this.current.parentNode.removeChild(this.current); | |||
this.link = this.span = null; | ||||
}, | ||||
onClick : function (e) { | ||||
Mads Kiilerich
|
r4160 | YUE.stopEvent(e); | ||
r3776 | this.paginator.setPage(this.paginator.getTotalPages()); | |||
} | ||||
}; | ||||
})(); | ||||
var pagi = new YAHOO.widget.Paginator({ | ||||
rowsPerPage: links_per_page, | ||||
alwaysVisible: false, | ||||
template : "{PreviousPageLink} {MyFirstPageLink} {PageLinks} {MyLastPageLink} {NextPageLink}", | ||||
pageLinks: 5, | ||||
containerClass: 'pagination-wh', | ||||
currentPageClass: 'pager_curpage', | ||||
pageLinkClass: 'pager_link', | ||||
nextPageLinkLabel: '>', | ||||
previousPageLinkLabel: '<', | ||||
containers:containers | ||||
}) | ||||
return pagi | ||||
} | ||||
Mads Kiilerich
|
r4164 | var YUI_datatable = function(data, fields, columns, countnode, sortkey, rows){ | ||
Mads Kiilerich
|
r4163 | var myDataSource = new YAHOO.util.DataSource(data); | ||
myDataSource.responseType = YAHOO.util.DataSource.TYPE_JSON; | ||||
myDataSource.responseSchema = { | ||||
resultsList: "records", | ||||
fields: fields, | ||||
}; | ||||
myDataSource.doBeforeCallback = function(req, raw, res, cb) { | ||||
// This is the filter function | ||||
var data = res.results || [], | ||||
filtered = [], | ||||
i, l; | ||||
if (req) { | ||||
req = req.toLowerCase(); | ||||
for (i = 0; i<data.length; i++) { | ||||
var pos = data[i].raw_name.toLowerCase().indexOf(req) | ||||
if (pos != -1) { | ||||
filtered.push(data[i]); | ||||
} | ||||
} | ||||
res.results = filtered; | ||||
} | ||||
$(countnode).html(res.results.length); | ||||
return res; | ||||
} | ||||
var myDataTable = new YAHOO.widget.DataTable("datatable_list_wrap", columns, myDataSource, { | ||||
sortedBy: {key:sortkey, dir:"asc"}, | ||||
Mads Kiilerich
|
r4164 | paginator: YUI_paginator(rows !== undefined && rows ? rows : 25, ['user-paginator']), | ||
Mads Kiilerich
|
r4163 | MSG_SORTASC: _TM['MSG_SORTASC'], | ||
MSG_SORTDESC: _TM['MSG_SORTDESC'], | ||||
MSG_EMPTY: _TM['MSG_EMPTY'], | ||||
MSG_ERROR: _TM['MSG_ERROR'], | ||||
MSG_LOADING: _TM['MSG_LOADING'], | ||||
}); | ||||
myDataTable.subscribe('postRenderEvent',function(oArgs) { | ||||
tooltip_activate(); | ||||
quick_repo_menu(); | ||||
}); | ||||
var filterTimeout = null; | ||||
var $q_filter = $('#q_filter'); | ||||
updateFilter = function () { | ||||
// Reset timeout | ||||
filterTimeout = null; | ||||
// Reset sort | ||||
var state = myDataTable.getState(); | ||||
state.sortedBy = {key:sortkey, dir:YAHOO.widget.DataTable.CLASS_ASC}; | ||||
// Get filtered data | ||||
myDataSource.sendRequest($q_filter.val(), { | ||||
success : myDataTable.onDataReturnInitializeTable, | ||||
failure : myDataTable.onDataReturnInitializeTable, | ||||
scope : myDataTable, | ||||
argument: state}); | ||||
}; | ||||
$q_filter.click(function(){ | ||||
if(!$q_filter.hasClass('loaded')){ | ||||
//TODO: load here full list later to do search within groups | ||||
$q_filter.addClass('loaded'); | ||||
} | ||||
}); | ||||
$q_filter.keyup(function (e) { | ||||
clearTimeout(filterTimeout); | ||||
filterTimeout = setTimeout(updateFilter, 600); | ||||
}); | ||||
} | ||||
r3776 | ||||
r3419 | // global hooks after DOM is loaded | |||
Mads Kiilerich
|
r4160 | $(document).ready(function(){ | ||
$('.diff-collapse-button').click(function(e) { | ||||
var $button = $(e.currentTarget); | ||||
var $target = $('#' + $button.attr('target')); | ||||
if($target.hasClass('hidden')){ | ||||
$target.removeClass('hidden'); | ||||
$button.html("↑ {0} ↑".format(_TM['Collapse diff'])); | ||||
r3696 | } | |||
Mads Kiilerich
|
r4160 | else if(!$target.hasClass('hidden')){ | ||
$target.addClass('hidden'); | ||||
$button.html("↓ {0} ↓".format(_TM['Expand diff'])); | ||||
r3696 | } | |||
}); | ||||
r3419 | }); | |||