rhodecode.js
1324 lines
| 38.6 KiB
| application/javascript
|
JavascriptLexer
r1421 | /** | ||
RhodeCode JS Files | |||
**/ | |||
if (typeof console == "undefined" || typeof console.log == "undefined"){ | |||
console = { log: function() {} } | |||
} | |||
r1699 | var str_repeat = function(i, m) { | ||
r1421 | for (var o = []; m > 0; o[--m] = i); | ||
return o.join(''); | |||
r1699 | }; | ||
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() { | |||
function format() { | |||
var str = this; | |||
var len = arguments.length+1; | |||
var safe = undefined; | |||
var arg = undefined; | |||
// 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++]) { | |||
safe = typeof arg === 'object' ? JSON.stringify(arg) : arg; | |||
str = str.replace(RegExp('\\{'+(i-1)+'\\}', 'g'), safe); | |||
} | |||
return str; | |||
} | |||
// 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; | |||
// Replace the prototype property | |||
return format; | |||
r1458 | }(); | ||
r1421 | |||
r2369 | String.prototype.strip = function(char) { | ||
if(char === undefined){ | |||
char = '\\s'; | |||
} | |||
return this.replace(new RegExp('^'+char+'+|'+char+'+$','g'), ''); | |||
} | |||
String.prototype.lstrip = function(char) { | |||
if(char === undefined){ | |||
char = '\\s'; | |||
} | |||
return this.replace(new RegExp('^'+char+'+'),''); | |||
} | |||
String.prototype.rstrip = function(char) { | |||
if(char === undefined){ | |||
char = '\\s'; | |||
} | |||
return this.replace(new RegExp(''+char+'+$'),''); | |||
} | |||
r1465 | |||
/** | |||
* SmartColorGenerator | |||
* | |||
*usage:: | |||
* var CG = new ColorGenerator(); | |||
* var col = CG.getColor(key); //returns array of RGB | |||
* 'rgb({0})'.format(col.join(',') | |||
* | |||
* @returns {ColorGenerator} | |||
*/ | |||
r1699 | var ColorGenerator = function(){ | ||
r1465 | this.GOLDEN_RATIO = 0.618033988749895; | ||
this.CURRENT_RATIO = 0.22717784590367374 // this can be random | |||
this.HSV_1 = 0.75;//saturation | |||
this.HSV_2 = 0.95; | |||
this.color; | |||
this.cacheColorMap = {}; | |||
}; | |||
ColorGenerator.prototype = { | |||
getColor:function(key){ | |||
if(this.cacheColorMap[key] !== undefined){ | |||
return this.cacheColorMap[key]; | |||
} | |||
else{ | |||
this.cacheColorMap[key] = this.generateColor(); | |||
return this.cacheColorMap[key]; | |||
} | |||
}, | |||
_hsvToRgb:function(h,s,v){ | |||
if (s == 0.0) | |||
return [v, v, v]; | |||
i = parseInt(h * 6.0) | |||
f = (h * 6.0) - i | |||
p = v * (1.0 - s) | |||
q = v * (1.0 - s * f) | |||
t = v * (1.0 - s * (1.0 - f)) | |||
i = i % 6 | |||
if (i == 0) | |||
return [v, t, p] | |||
if (i == 1) | |||
return [q, v, p] | |||
if (i == 2) | |||
return [p, v, t] | |||
if (i == 3) | |||
return [p, q, v] | |||
if (i == 4) | |||
return [t, p, v] | |||
if (i == 5) | |||
return [v, p, q] | |||
}, | |||
generateColor:function(){ | |||
this.CURRENT_RATIO = this.CURRENT_RATIO+this.GOLDEN_RATIO; | |||
this.CURRENT_RATIO = this.CURRENT_RATIO %= 1; | |||
HSV_tuple = [this.CURRENT_RATIO, this.HSV_1, this.HSV_2] | |||
RGB_tuple = this._hsvToRgb(HSV_tuple[0],HSV_tuple[1],HSV_tuple[2]); | |||
function toRgb(v){ | |||
return ""+parseInt(v*256) | |||
} | |||
return [toRgb(RGB_tuple[0]),toRgb(RGB_tuple[1]),toRgb(RGB_tuple[2])]; | |||
} | |||
} | |||
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; | |||
// defines if push state is enabled for this browser ? | |||
var push_state_enabled = Boolean( | |||
window.history && window.history.pushState && window.history.replaceState | |||
&& !( /* disable for versions of iOS before version 4.3 (8F190) */ | |||
(/ Mobile\/([1-7][a-z]|(8([abcde]|f(1[0-8]))))/i).test(navigator.userAgent) | |||
/* disable for the mercury iOS browser, or at least older versions of the webkit engine */ | |||
|| (/AppleWebKit\/5([0-2]|3[0-2])/i).test(navigator.userAgent) | |||
) | |||
r1699 | ); | ||
r1421 | |||
r1717 | var _run_callbacks = function(callbacks){ | ||
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){}; | |||
} | |||
} | |||
} | |||
} | |||
r1421 | /** | ||
* Partial Ajax Implementation | |||
* | |||
* @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 | |||
* @param args arguments | |||
*/ | |||
function ypjax(url,container,s_call,f_call,args){ | |||
var method='GET'; | |||
if(args===undefined){ | |||
args=null; | |||
} | |||
// 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){ | |||
YUD.get(container).innerHTML=o.responseText; | |||
YUD.setStyle(container,'opacity','1.0'); | |||
//execute the given original callback | |||
if (s_call !== undefined){ | |||
s_call(o); | |||
} | |||
} | |||
})() | |||
YUD.setStyle(container,'opacity','0.3'); | |||
YUC.asyncRequest(method,url,{ | |||
success:s_wrapper, | |||
failure:function(o){ | |||
r1651 | console.log(o); | ||
YUD.get(container).innerHTML='ERROR'; | |||
YUD.setStyle(container,'opacity','1.0'); | |||
YUD.setStyle(container,'color','red'); | |||
r1421 | } | ||
},args); | |||
r1699 | }; | ||
r1421 | |||
r2187 | var ajaxPOST = function(url,postData,success) { | ||
r2189 | // Set special header for ajax == HTTP_X_PARTIAL_XHR | ||
YUC.initHeader('X-PARTIAL-XHR',true); | |||
r2187 | var toQueryString = function(o) { | ||
if(typeof o !== 'object') { | |||
return false; | |||
} | |||
var _p, _qs = []; | |||
for(_p in o) { | |||
_qs.push(encodeURIComponent(_p) + '=' + encodeURIComponent(o[_p])); | |||
} | |||
return _qs.join('&'); | |||
}; | |||
var sUrl = url; | |||
var callback = { | |||
success: success, | |||
failure: function (o) { | |||
alert("error"); | |||
}, | |||
}; | |||
var postData = toQueryString(postData); | |||
var request = YAHOO.util.Connect.asyncRequest('POST', sUrl, callback, postData); | |||
return request; | |||
}; | |||
r1421 | /** | ||
* tooltip activate | |||
*/ | |||
r1458 | var tooltip_activate = function(){ | ||
r1421 | function toolTipsId(){ | ||
var ids = []; | |||
var tts = YUQ('.tooltip'); | |||
for (var i = 0; i < tts.length; i++) { | |||
// if element doesn't not have and id | |||
// autogenerate one for tooltip | |||
if (!tts[i].id){ | |||
tts[i].id='tt'+((i*100)+tts.length); | |||
} | |||
ids.push(tts[i].id); | |||
} | |||
return ids | |||
}; | |||
var myToolTips = new YAHOO.widget.Tooltip("tooltip", { | |||
context: [[toolTipsId()],"tl","bl",null,[0,5]], | |||
monitorresize:false, | |||
xyoffset :[0,0], | |||
autodismissdelay:300000, | |||
hidedelay:5, | |||
showdelay:20, | |||
}); | |||
r1699 | }; | ||
r1426 | |||
/** | |||
* show more | |||
*/ | |||
r1458 | var show_more_event = function(){ | ||
r1426 | YUE.on(YUD.getElementsByClassName('show_more'),'click',function(e){ | ||
var el = e.target; | |||
YUD.setStyle(YUD.get(el.id.substring(1)),'display',''); | |||
YUD.setStyle(el.parentNode,'display','none'); | |||
}); | |||
r1699 | }; | ||
r1426 | |||
r1653 | |||
/** | |||
* Quick filter widget | |||
* | |||
* @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 | |||
* | |||
*/ | |||
var q_filter = function(target,nodes,display_element){ | |||
var nodes = nodes; | |||
var q_filter_field = YUD.get(target); | |||
var F = YAHOO.namespace(target); | |||
YUE.on(q_filter_field,'click',function(){ | |||
q_filter_field.value = ''; | |||
}); | |||
YUE.on(q_filter_field,'keyup',function(e){ | |||
clearTimeout(F.filterTimeout); | |||
F.filterTimeout = setTimeout(F.updateFilter,600); | |||
}); | |||
F.filterTimeout = null; | |||
var show_node = function(node){ | |||
YUD.setStyle(node,'display','') | |||
} | |||
var hide_node = function(node){ | |||
YUD.setStyle(node,'display','none'); | |||
} | |||
F.updateFilter = function() { | |||
// Reset timeout | |||
F.filterTimeout = null; | |||
var obsolete = []; | |||
var req = q_filter_field.value.toLowerCase(); | |||
var l = nodes.length; | |||
var i; | |||
r1667 | var showing = 0; | ||
r1653 | for (i=0;i<l;i++ ){ | ||
var n = nodes[i]; | |||
var target_element = display_element(n) | |||
if(req && n.innerHTML.toLowerCase().indexOf(req) == -1){ | |||
hide_node(target_element); | |||
} | |||
else{ | |||
show_node(target_element); | |||
r1667 | showing+=1; | ||
r1653 | } | ||
} | |||
r1667 | // if repo_count is set update the number | ||
var cnt = YUD.get('repo_count'); | |||
if(cnt){ | |||
YUD.get('repo_count').innerHTML = showing; | |||
} | |||
r1653 | } | ||
r1699 | }; | ||
r1653 | |||
r2187 | var tableTr = function(cls,body){ | ||
var tr = document.createElement('tr'); | |||
YUD.addClass(tr, cls); | |||
var cont = new YAHOO.util.Element(body); | |||
var comment_id = fromHTML(body).children[0].id.split('comment-')[1]; | |||
tr.id = 'comment-tr-{0}'.format(comment_id); | |||
tr.innerHTML = '<td class="lineno-inline new-inline"></td>'+ | |||
'<td class="lineno-inline old-inline"></td>'+ | |||
'<td>{0}</td>'.format(body); | |||
return tr; | |||
r1674 | }; | ||
r1653 | |||
r1677 | /** comments **/ | ||
var removeInlineForm = function(form) { | |||
form.parentNode.removeChild(form); | |||
r1699 | }; | ||
r1677 | |||
r1682 | var createInlineForm = function(parent_tr, f_path, line) { | ||
r1677 | var tmpl = YUD.get('comment-inline-form-template').innerHTML; | ||
tmpl = tmpl.format(f_path, line); | |||
r1682 | var form = tableTr('comment-form-inline',tmpl) | ||
r1677 | // create event for hide button | ||
form = new YAHOO.util.Element(form); | |||
var form_hide_button = new YAHOO.util.Element(form.getElementsByClassName('hide-inline-form')[0]); | |||
form_hide_button.on('click', function(e) { | |||
var newtr = e.currentTarget.parentNode.parentNode.parentNode.parentNode.parentNode; | |||
r2187 | if(YUD.hasClass(newtr.nextElementSibling,'inline-comments-button')){ | ||
YUD.setStyle(newtr.nextElementSibling,'display',''); | |||
} | |||
r1677 | removeInlineForm(newtr); | ||
YUD.removeClass(parent_tr, 'form-open'); | |||
r2187 | |||
r1677 | }); | ||
r2187 | |||
r1677 | return form | ||
r1699 | }; | ||
r2187 | |||
/** | |||
* Inject inline comment for on given TR this tr should be always an .line | |||
* tr containing the line. Code will detect comment, and always put the comment | |||
* block at the very bottom | |||
*/ | |||
r1705 | var injectInlineForm = function(tr){ | ||
r2187 | if(!YUD.hasClass(tr, 'line')){ | ||
return | |||
} | |||
var submit_url = AJAX_COMMENT_URL; | |||
r1787 | if(YUD.hasClass(tr,'form-open') || YUD.hasClass(tr,'context') || YUD.hasClass(tr,'no-comment')){ | ||
r1705 | return | ||
} | |||
YUD.addClass(tr,'form-open'); | |||
var node = tr.parentNode.parentNode.parentNode.getElementsByClassName('full_f_path')[0]; | |||
var f_path = YUD.getAttribute(node,'path'); | |||
var lineno = getLineNo(tr); | |||
r2187 | var form = createInlineForm(tr, f_path, lineno, submit_url); | ||
var parent = tr; | |||
while (1){ | |||
var n = parent.nextElementSibling; | |||
// next element are comments ! | |||
if(YUD.hasClass(n,'inline-comments')){ | |||
parent = n; | |||
} | |||
else{ | |||
break; | |||
} | |||
} | |||
YUD.insertAfter(form,parent); | |||
r1772 | YUD.get('text_'+lineno).focus(); | ||
r2187 | var f = YUD.get(form); | ||
var overlay = f.getElementsByClassName('overlay')[0]; | |||
var _form = f.getElementsByClassName('inline-form')[0]; | |||
form.on('submit',function(e){ | |||
YUE.preventDefault(e); | |||
//ajax submit | |||
var text = YUD.get('text_'+lineno).value; | |||
var postData = { | |||
'text':text, | |||
'f_path':f_path, | |||
'line':lineno | |||
}; | |||
if(lineno === undefined){ | |||
alert('missing line !'); | |||
return | |||
} | |||
if(f_path === undefined){ | |||
alert('missing file path !'); | |||
return | |||
} | |||
r2189 | if(text == ""){ | ||
return | |||
} | |||
r2187 | var success = function(o){ | ||
YUD.removeClass(tr, 'form-open'); | |||
removeInlineForm(f); | |||
var json_data = JSON.parse(o.responseText); | |||
renderInlineComment(json_data); | |||
}; | |||
if (YUD.hasClass(overlay,'overlay')){ | |||
var w = _form.offsetWidth; | |||
var h = _form.offsetHeight; | |||
YUD.setStyle(overlay,'width',w+'px'); | |||
YUD.setStyle(overlay,'height',h+'px'); | |||
} | |||
YUD.addClass(overlay, 'submitting'); | |||
ajaxPOST(submit_url, postData, success); | |||
}); | |||
r2369 | // callbacks | ||
r1772 | tooltip_activate(); | ||
r1705 | }; | ||
r2187 | var deleteComment = function(comment_id){ | ||
var url = AJAX_COMMENT_DELETE_URL.replace('__COMMENT_ID__',comment_id); | |||
var postData = {'_method':'delete'}; | |||
var success = function(o){ | |||
var n = YUD.get('comment-tr-'+comment_id); | |||
var root = n.previousElementSibling.previousElementSibling; | |||
n.parentNode.removeChild(n); | |||
// scann nodes, and attach add button to last one | |||
r2189 | placeAddButton(root); | ||
r2187 | } | ||
ajaxPOST(url,postData,success); | |||
} | |||
var createInlineAddButton = function(tr){ | |||
var label = TRANSLATION_MAP['add another comment']; | |||
var html_el = document.createElement('div'); | |||
YUD.addClass(html_el, 'add-comment'); | |||
html_el.innerHTML = '<span class="ui-btn">{0}</span>'.format(label); | |||
var add = new YAHOO.util.Element(html_el); | |||
r1705 | add.on('click', function(e) { | ||
injectInlineForm(tr); | |||
}); | |||
return add; | |||
}; | |||
r1699 | |||
r1677 | var getLineNo = function(tr) { | ||
var line; | |||
var o = tr.children[0].id.split('_'); | |||
var n = tr.children[1].id.split('_'); | |||
r1679 | if (n.length >= 2) { | ||
line = n[n.length-1]; | |||
} else if (o.length >= 2) { | |||
r1685 | line = o[o.length-1]; | ||
r1677 | } | ||
return line | |||
r1699 | }; | ||
r2187 | var placeAddButton = function(target_tr){ | ||
if(!target_tr){ | |||
return | |||
} | |||
var last_node = target_tr; | |||
//scann | |||
while (1){ | |||
var n = last_node.nextElementSibling; | |||
// next element are comments ! | |||
if(YUD.hasClass(n,'inline-comments')){ | |||
last_node = n; | |||
//also remove the comment button from previos | |||
var comment_add_buttons = last_node.getElementsByClassName('add-comment'); | |||
for(var i=0;i<comment_add_buttons.length;i++){ | |||
var b = comment_add_buttons[i]; | |||
b.parentNode.removeChild(b); | |||
} | |||
} | |||
else{ | |||
break; | |||
} | |||
} | |||
var add = createInlineAddButton(target_tr); | |||
// get the comment div | |||
var comment_block = last_node.getElementsByClassName('comment')[0]; | |||
// attach add button | |||
YUD.insertAfter(add,comment_block); | |||
} | |||
/** | |||
* Places the inline comment into the changeset block in proper line position | |||
*/ | |||
var placeInline = function(target_container,lineno,html){ | |||
var lineid = "{0}_{1}".format(target_container,lineno); | |||
var target_line = YUD.get(lineid); | |||
var comment = new YAHOO.util.Element(tableTr('inline-comments',html)) | |||
// check if there are comments already ! | |||
var parent = target_line.parentNode; | |||
var root_parent = parent; | |||
while (1){ | |||
var n = parent.nextElementSibling; | |||
// next element are comments ! | |||
if(YUD.hasClass(n,'inline-comments')){ | |||
parent = n; | |||
} | |||
else{ | |||
break; | |||
} | |||
} | |||
// put in the comment at the bottom | |||
YUD.insertAfter(comment,parent); | |||
// scann nodes, and attach add button to last one | |||
placeAddButton(root_parent); | |||
return target_line; | |||
} | |||
/** | |||
* make a single inline comment and place it inside | |||
*/ | |||
var renderInlineComment = function(json_data){ | |||
try{ | |||
var html = json_data['rendered_text']; | |||
var lineno = json_data['line_no']; | |||
var target_id = json_data['target_id']; | |||
placeInline(target_id, lineno, html); | |||
}catch(e){ | |||
console.log(e); | |||
} | |||
} | |||
/** | |||
* Iterates over all the inlines, and places them inside proper blocks of data | |||
*/ | |||
var renderInlineComments = function(file_comments){ | |||
for (f in file_comments){ | |||
// holding all comments for a FILE | |||
var box = file_comments[f]; | |||
var target_id = YUD.getAttribute(box,'target_id'); | |||
// actually comments with line numbers | |||
var comments = box.children; | |||
for(var i=0; i<comments.length; i++){ | |||
var data = { | |||
'rendered_text': comments[i].outerHTML, | |||
'line_no': YUD.getAttribute(comments[i],'line'), | |||
'target_id': target_id | |||
} | |||
renderInlineComment(data); | |||
} | |||
} | |||
} | |||
r1699 | |||
r1705 | var fileBrowserListeners = function(current_url, node_list_url, url_base, | ||
truncated_lbl, nomatch_lbl){ | |||
r1699 | var current_url_branch = +"?branch=__BRANCH__"; | ||
var url = url_base; | |||
var node_url = node_list_url; | |||
YUE.on('stay_at_branch','click',function(e){ | |||
if(e.target.checked){ | |||
var uri = current_url_branch; | |||
uri = uri.replace('__BRANCH__',e.target.value); | |||
window.location = uri; | |||
} | |||
else{ | |||
window.location = current_url; | |||
} | |||
}) | |||
var n_filter = YUD.get('node_filter'); | |||
var F = YAHOO.namespace('node_filter'); | |||
F.filterTimeout = null; | |||
var nodes = null; | |||
F.initFilter = function(){ | |||
YUD.setStyle('node_filter_box_loading','display',''); | |||
YUD.setStyle('search_activate_id','display','none'); | |||
YUD.setStyle('add_node_id','display','none'); | |||
YUC.initHeader('X-PARTIAL-XHR',true); | |||
YUC.asyncRequest('GET',url,{ | |||
success:function(o){ | |||
nodes = JSON.parse(o.responseText); | |||
YUD.setStyle('node_filter_box_loading','display','none'); | |||
YUD.setStyle('node_filter_box','display',''); | |||
r1810 | n_filter.focus(); | ||
if(YUD.hasClass(n_filter,'init')){ | |||
n_filter.value = ''; | |||
YUD.removeClass(n_filter,'init'); | |||
} | |||
r1699 | }, | ||
failure:function(o){ | |||
console.log('failed to load'); | |||
} | |||
},null); | |||
} | |||
F.updateFilter = function(e) { | |||
return function(){ | |||
// Reset timeout | |||
F.filterTimeout = null; | |||
r1810 | var query = e.target.value.toLowerCase(); | ||
r1699 | var match = []; | ||
var matches = 0; | |||
var matches_max = 20; | |||
if (query != ""){ | |||
for(var i=0;i<nodes.length;i++){ | |||
r1810 | |||
var pos = nodes[i].name.toLowerCase().indexOf(query) | |||
r1699 | if(query && pos != -1){ | ||
matches++ | |||
//show only certain amount to not kill browser | |||
if (matches > matches_max){ | |||
break; | |||
} | |||
r1810 | var n = nodes[i].name; | ||
var t = nodes[i].type; | |||
r1699 | var n_hl = n.substring(0,pos) | ||
+"<b>{0}</b>".format(n.substring(pos,pos+query.length)) | |||
+n.substring(pos+query.length) | |||
r1810 | match.push('<tr><td><a class="browser-{0}" href="{1}">{2}</a></td><td colspan="5"></td></tr>'.format(t,node_url.replace('__FPATH__',n),n_hl)); | ||
r1699 | } | ||
if(match.length >= matches_max){ | |||
r1705 | match.push('<tr><td>{0}</td><td colspan="5"></td></tr>'.format(truncated_lbl)); | ||
r1699 | } | ||
} | |||
} | |||
if(query != ""){ | |||
YUD.setStyle('tbody','display','none'); | |||
YUD.setStyle('tbody_filtered','display',''); | |||
if (match.length==0){ | |||
r1705 | match.push('<tr><td>{0}</td><td colspan="5"></td></tr>'.format(nomatch_lbl)); | ||
r1699 | } | ||
YUD.get('tbody_filtered').innerHTML = match.join(""); | |||
} | |||
else{ | |||
YUD.setStyle('tbody','display',''); | |||
YUD.setStyle('tbody_filtered','display','none'); | |||
} | |||
} | |||
}; | |||
YUE.on(YUD.get('filter_activate'),'click',function(){ | |||
F.initFilter(); | |||
}) | |||
YUE.on(n_filter,'click',function(){ | |||
r1810 | if(YUD.hasClass(n_filter,'init')){ | ||
n_filter.value = ''; | |||
YUD.removeClass(n_filter,'init'); | |||
} | |||
r1699 | }); | ||
YUE.on(n_filter,'keyup',function(e){ | |||
clearTimeout(F.filterTimeout); | |||
F.filterTimeout = setTimeout(F.updateFilter(e),600); | |||
}); | |||
}; | |||
var initCodeMirror = function(textAreadId,resetUrl){ | |||
var myCodeMirror = CodeMirror.fromTextArea(YUD.get(textAreadId),{ | |||
mode: "null", | |||
lineNumbers:true | |||
}); | |||
YUE.on('reset','click',function(e){ | |||
window.location=resetUrl | |||
}); | |||
YUE.on('file_enable','click',function(){ | |||
YUD.setStyle('editor_container','display',''); | |||
YUD.setStyle('upload_file_container','display','none'); | |||
YUD.setStyle('filename_container','display',''); | |||
}); | |||
YUE.on('upload_file_enable','click',function(){ | |||
YUD.setStyle('editor_container','display','none'); | |||
YUD.setStyle('upload_file_container','display',''); | |||
YUD.setStyle('filename_container','display','none'); | |||
}); | |||
}; | |||
var getIdentNode = function(n){ | |||
//iterate thru nodes untill matched interesting node ! | |||
if (typeof n == 'undefined'){ | |||
return -1 | |||
} | |||
if(typeof n.id != "undefined" && n.id.match('L[0-9]+')){ | |||
return n | |||
} | |||
else{ | |||
return getIdentNode(n.parentNode); | |||
} | |||
}; | |||
var getSelectionLink = function(selection_link_label) { | |||
return function(){ | |||
//get selection from start/to nodes | |||
if (typeof window.getSelection != "undefined") { | |||
s = window.getSelection(); | |||
from = getIdentNode(s.anchorNode); | |||
till = getIdentNode(s.focusNode); | |||
f_int = parseInt(from.id.replace('L','')); | |||
t_int = parseInt(till.id.replace('L','')); | |||
if (f_int > t_int){ | |||
//highlight from bottom | |||
offset = -35; | |||
ranges = [t_int,f_int]; | |||
} | |||
else{ | |||
//highligth from top | |||
offset = 35; | |||
ranges = [f_int,t_int]; | |||
} | |||
if (ranges[0] != ranges[1]){ | |||
if(YUD.get('linktt') == null){ | |||
hl_div = document.createElement('div'); | |||
hl_div.id = 'linktt'; | |||
} | |||
anchor = '#L'+ranges[0]+'-'+ranges[1]; | |||
hl_div.innerHTML = ''; | |||
l = document.createElement('a'); | |||
l.href = location.href.substring(0,location.href.indexOf('#'))+anchor; | |||
l.innerHTML = selection_link_label; | |||
hl_div.appendChild(l); | |||
YUD.get('body').appendChild(hl_div); | |||
xy = YUD.getXY(till.id); | |||
YUD.addClass('linktt','yui-tt'); | |||
YUD.setStyle('linktt','top',xy[1]+offset+'px'); | |||
YUD.setStyle('linktt','left',xy[0]+'px'); | |||
YUD.setStyle('linktt','visibility','visible'); | |||
} | |||
else{ | |||
YUD.setStyle('linktt','visibility','hidden'); | |||
} | |||
} | |||
} | |||
}; | |||
r1712 | |||
r1717 | var deleteNotification = function(url, notification_id,callbacks){ | ||
r1712 | var callback = { | ||
success:function(o){ | |||
var obj = YUD.get(String("notification_"+notification_id)); | |||
r1717 | if(obj.parentNode !== undefined){ | ||
obj.parentNode.removeChild(obj); | |||
} | |||
_run_callbacks(callbacks); | |||
r1712 | }, | ||
failure:function(o){ | |||
alert("error"); | |||
}, | |||
}; | |||
var postData = '_method=delete'; | |||
var sUrl = url.replace('__NOTIFICATION_ID__',notification_id); | |||
var request = YAHOO.util.Connect.asyncRequest('POST', sUrl, | |||
callback, postData); | |||
}; | |||
r1778 | |||
r2142 | /** MEMBERS AUTOCOMPLETE WIDGET **/ | ||
r2369 | var MembersAutoComplete = function (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]; | |||
r2369 | if (((contact.fname+"").toLowerCase().indexOf(query) > -1) || | ||
((contact.lname+"").toLowerCase().indexOf(query) > -1) || | |||
((contact.nname) && ((contact.nname).toLowerCase().indexOf(query) > -1))) { | |||
matches[matches.length] = contact; | |||
} | |||
r2142 | } | ||
return matches; | |||
}; | |||
// Define a custom search function for the DataSource of usersGroups | |||
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 | |||
var membersAC = new YAHOO.widget.AutoComplete("perm_new_member_name", "perm_container", memberDS); | |||
membersAC.useShadow = false; | |||
membersAC.resultTypeList = false; | |||
// Instantiate AutoComplete for owner | |||
var ownerAC = new YAHOO.widget.AutoComplete("user", "owner_container", ownerDS); | |||
ownerAC.useShadow = false; | |||
ownerAC.resultTypeList = false; | |||
// Helper highlight function for the formatter | |||
var highlightMatch = function (full, snippet, matchindex) { | |||
return full.substring(0, matchindex) | |||
+ "<span class='match'>" | |||
+ full.substr(matchindex, snippet.length) | |||
+ "</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){ | |||
if (group !== undefined){ | |||
em = '/images/icons/group.png' | |||
} | |||
r2145 | tmpl = '<div class="ac-container-wrap"><img class="perm-gravatar-ac" src="{0}"/>{1}</div>' | ||
r2142 | return tmpl.format(em,res) | ||
} | |||
// 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); | |||
} | |||
return _gravatar(grprefix + oResultData.grname + grsuffix, null,true); | |||
// Users | |||
r2369 | } else if (oResultData.nname != undefined) { | ||
var fname = oResultData.fname || ""; | |||
var lname = oResultData.lname || ""; | |||
var nname = oResultData.nname; | |||
// 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) { | |||
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; | |||
YUD.get('perm_new_member_type').value = 'user'; | |||
} else { | |||
//groups | |||
myAC.getInputEl().value = oData.grname; | |||
YUD.get('perm_new_member_type').value = 'users_group'; | |||
} | |||
}; | |||
membersAC.itemSelectEvent.subscribe(myHandler); | |||
if(ownerAC.itemSelectEvent){ | |||
ownerAC.itemSelectEvent.subscribe(myHandler); | |||
} | |||
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) { | |||
var org_sQuery = sQuery; | |||
if(this.mentionQuery == null){ | |||
return [] | |||
} | |||
sQuery = this.mentionQuery; | |||
// 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]; | |||
r2369 | 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; | |||
// Helper highlight function for the formatter | |||
var highlightMatch = function (full, snippet, matchindex) { | |||
return full.substring(0, matchindex) | |||
+ "<span class='match'>" | |||
+ full.substr(matchindex, snippet.length) | |||
+ "</span>" + full.substring(matchindex + snippet.length); | |||
}; | |||
// Custom formatter to highlight the matching letters | |||
ownerAC.formatResult = function (oResultData, sQuery, sResultMatch) { | |||
var org_sQuery = sQuery; | |||
if(this.dataSource.mentionQuery != null){ | |||
sQuery = this.dataSource.mentionQuery; | |||
} | |||
var query = sQuery.toLowerCase(); | |||
var _gravatar = function(res, em, group){ | |||
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) | |||
} | |||
r2369 | if (oResultData.nname != undefined) { | ||
var fname = oResultData.fname || ""; | |||
var lname = oResultData.lname || ""; | |||
var nname = oResultData.nname; | |||
// 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){ | |||
ownerAC.itemSelectEvent.subscribe(function (sType, aArgs) { | |||
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 | |||
//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+' '); | |||
myAC.getInputEl().value = chunks.join('') | |||
YUD.get(myAC.getInputEl()).focus(); // Y U NO WORK !? | |||
} else { | |||
//groups | |||
myAC.getInputEl().value = oData.grname; | |||
YUD.get('perm_new_member_type').value = 'users_group'; | |||
} | |||
}); | |||
} | |||
// 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) { | |||
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); | |||
chunks.push(org.substr(0,at_pos))// prefix chunk | |||
chunks.push(msg2) // search chunk | |||
chunks.push(org.substr(max_pos)) // postfix chunk | |||
// clean up msg2 for filtering and regex match | |||
r2369 | var msg2 = msg2.lstrip(' ').lstrip('\n'); | ||
r2368 | |||
if(re.test(msg2)){ | |||
var unam = re.exec(msg2)[1]; | |||
return [unam, chunks]; | |||
} | |||
return [null, null]; | |||
}; | |||
ownerAC.textboxKeyUpEvent.subscribe(function(type, args){ | |||
var ac_obj = args[0]; | |||
var currentMessage = args[1]; | |||
var currentCaretPosition = args[0]._elTextbox.selectionStart; | |||
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; | |||
}) | |||
return { | |||
ownerDS: ownerDS, | |||
ownerAC: ownerAC, | |||
}; | |||
} | |||
r1778 | /** | ||
r1779 | * QUICK REPO MENU | ||
*/ | |||
var quick_repo_menu = function(){ | |||
r2088 | YUE.on(YUQ('.quick_repo_menu'),'mouseenter',function(e){ | ||
var menu = e.currentTarget.firstElementChild.firstElementChild; | |||
if(YUD.hasClass(menu,'hidden')){ | |||
YUD.replaceClass(e.currentTarget,'hidden', 'active'); | |||
YUD.replaceClass(menu, 'hidden', 'active'); | |||
} | |||
}) | |||
YUE.on(YUQ('.quick_repo_menu'),'mouseleave',function(e){ | |||
var menu = e.currentTarget.firstElementChild.firstElementChild; | |||
if(YUD.hasClass(menu,'active')){ | |||
YUD.replaceClass(e.currentTarget, 'active', 'hidden'); | |||
YUD.replaceClass(menu, 'active', 'hidden'); | |||
} | |||
}) | |||
r1779 | }; | ||
/** | |||
r1778 | * TABLE SORTING | ||
*/ | |||
// returns a node from given html; | |||
var fromHTML = function(html){ | |||
var _html = document.createElement('element'); | |||
_html.innerHTML = html; | |||
return _html; | |||
} | |||
var get_rev = function(node){ | |||
var n = node.firstElementChild.firstElementChild; | |||
if (n===null){ | |||
return -1 | |||
} | |||
else{ | |||
out = n.firstElementChild.innerHTML.split(':')[0].replace('r',''); | |||
return parseInt(out); | |||
} | |||
} | |||
var get_name = function(node){ | |||
var name = node.firstElementChild.children[2].innerHTML; | |||
return name | |||
} | |||
var get_group_name = function(node){ | |||
var name = node.firstElementChild.children[1].innerHTML; | |||
return name | |||
} | |||
r1782 | var get_date = function(node){ | ||
var date_ = node.firstElementChild.innerHTML; | |||
return date_ | |||
} | |||
r1778 | var revisionSort = function(a, b, desc, field) { | ||
var a_ = fromHTML(a.getData(field)); | |||
var b_ = fromHTML(b.getData(field)); | |||
// extract revisions from string nodes | |||
a_ = get_rev(a_) | |||
b_ = get_rev(b_) | |||
var comp = YAHOO.util.Sort.compare; | |||
var compState = comp(a_, b_, desc); | |||
return compState; | |||
}; | |||
var ageSort = function(a, b, desc, field) { | |||
var a_ = a.getData(field); | |||
var b_ = b.getData(field); | |||
var comp = YAHOO.util.Sort.compare; | |||
var compState = comp(a_, b_, desc); | |||
return compState; | |||
}; | |||
var nameSort = function(a, b, desc, field) { | |||
var a_ = fromHTML(a.getData(field)); | |||
var b_ = fromHTML(b.getData(field)); | |||
// extract name from table | |||
a_ = get_name(a_) | |||
b_ = get_name(b_) | |||
var comp = YAHOO.util.Sort.compare; | |||
var compState = comp(a_, b_, desc); | |||
return compState; | |||
}; | |||
r2127 | var permNameSort = function(a, b, desc, field) { | ||
var a_ = fromHTML(a.getData(field)); | |||
var b_ = fromHTML(b.getData(field)); | |||
// extract name from table | |||
r2128 | |||
a_ = a_.children[0].innerHTML; | |||
b_ = b_.children[0].innerHTML; | |||
r2127 | |||
var comp = YAHOO.util.Sort.compare; | |||
var compState = comp(a_, b_, desc); | |||
return compState; | |||
}; | |||
r1778 | var groupNameSort = function(a, b, desc, field) { | ||
var a_ = fromHTML(a.getData(field)); | |||
var b_ = fromHTML(b.getData(field)); | |||
// extract name from table | |||
a_ = get_group_name(a_) | |||
b_ = get_group_name(b_) | |||
var comp = YAHOO.util.Sort.compare; | |||
var compState = comp(a_, b_, desc); | |||
return compState; | |||
r1782 | }; | ||
var dateSort = function(a, b, desc, field) { | |||
var a_ = fromHTML(a.getData(field)); | |||
var b_ = fromHTML(b.getData(field)); | |||
// extract name from table | |||
a_ = get_date(a_) | |||
b_ = get_date(b_) | |||
var comp = YAHOO.util.Sort.compare; | |||
var compState = comp(a_, b_, desc); | |||
return compState; | |||
r1778 | }; |