helpers.py
383 lines
| 14.0 KiB
| text/x-python
|
PythonLexer
r547 | """Helper functions | |||
Consists of functions to typically be used within templates, but also | ||||
available to Controllers. This module is available to both as 'h'. | ||||
""" | ||||
from pygments.formatters import HtmlFormatter | ||||
from pygments import highlight as code_highlight | ||||
from pylons import url, app_globals as g | ||||
from pylons.i18n.translation import _, ungettext | ||||
from vcs.utils.annotate import annotate_highlight | ||||
from webhelpers.html import literal, HTML, escape | ||||
from webhelpers.html.tools import * | ||||
from webhelpers.html.builder import make_tag | ||||
from webhelpers.html.tags import auto_discovery_link, checkbox, css_classes, \ | ||||
end_form, file, form, hidden, image, javascript_link, link_to, link_to_if, \ | ||||
link_to_unless, ol, required_legend, select, stylesheet_link, submit, text, \ | ||||
password, textarea, title, ul, xml_declaration, radio | ||||
from webhelpers.html.tools import auto_link, button_to, highlight, js_obfuscate, \ | ||||
mail_to, strip_links, strip_tags, tag_re | ||||
from webhelpers.number import format_byte_size, format_bit_size | ||||
from webhelpers.pylonslib import Flash as _Flash | ||||
from webhelpers.pylonslib.secure_form import secure_form | ||||
from webhelpers.text import chop_at, collapse, convert_accented_entities, \ | ||||
convert_misc_entities, lchop, plural, rchop, remove_formatting, \ | ||||
replace_whitespace, urlify, truncate, wrap_paragraphs | ||||
#Custom helpers here :) | ||||
class _Link(object): | ||||
''' | ||||
Make a url based on label and url with help of url_for | ||||
r604 | :param label:name of link if not defined url is used | |||
:param url: the url for link | ||||
r547 | ''' | |||
def __call__(self, label='', *url_, **urlargs): | ||||
if label is None or '': | ||||
label = url | ||||
link_fn = link_to(label, url(*url_, **urlargs)) | ||||
return link_fn | ||||
link = _Link() | ||||
class _GetError(object): | ||||
def __call__(self, field_name, form_errors): | ||||
tmpl = """<span class="error_msg">%s</span>""" | ||||
if form_errors and form_errors.has_key(field_name): | ||||
return literal(tmpl % form_errors.get(field_name)) | ||||
get_error = _GetError() | ||||
def recursive_replace(str, replace=' '): | ||||
""" | ||||
Recursive replace of given sign to just one instance | ||||
r604 | :param str: given string | |||
:param replace:char to find and replace multiple instances | ||||
r547 | ||||
Examples:: | ||||
>>> recursive_replace("Mighty---Mighty-Bo--sstones",'-') | ||||
'Mighty-Mighty-Bo-sstones' | ||||
""" | ||||
if str.find(replace * 2) == -1: | ||||
return str | ||||
else: | ||||
str = str.replace(replace * 2, replace) | ||||
r671 | return recursive_replace(str, replace) | |||
r547 | ||||
class _ToolTip(object): | ||||
r671 | ||||
r547 | def __call__(self, tooltip_title, trim_at=50): | |||
""" | ||||
Special function just to wrap our text into nice formatted autowrapped | ||||
text | ||||
r604 | :param tooltip_title: | |||
r547 | """ | |||
r671 | ||||
r547 | return wrap_paragraphs(escape(tooltip_title), trim_at)\ | |||
.replace('\n', '<br/>') | ||||
r671 | ||||
r547 | def activate(self): | |||
""" | ||||
Adds tooltip mechanism to the given Html all tooltips have to have | ||||
set class tooltip and set attribute tooltip_title. | ||||
Then a tooltip will be generated based on that | ||||
All with yui js tooltip | ||||
""" | ||||
r671 | ||||
r547 | js = ''' | |||
YAHOO.util.Event.onDOMReady(function(){ | ||||
function toolTipsId(){ | ||||
var ids = []; | ||||
var tts = YAHOO.util.Dom.getElementsByClassName('tooltip'); | ||||
for (var i = 0; i < tts.length; i++) { | ||||
//if element doesn not have and id autgenerate one for tooltip | ||||
if (!tts[i].id){ | ||||
tts[i].id='tt'+i*100; | ||||
} | ||||
ids.push(tts[i].id); | ||||
} | ||||
return ids | ||||
}; | ||||
var myToolTips = new YAHOO.widget.Tooltip("tooltip", { | ||||
context: toolTipsId(), | ||||
monitorresize:false, | ||||
xyoffset :[0,0], | ||||
autodismissdelay:300000, | ||||
hidedelay:5, | ||||
showdelay:20, | ||||
}); | ||||
//Mouse Over event disabled for new repositories since they dont | ||||
//have last commit message | ||||
myToolTips.contextMouseOverEvent.subscribe( | ||||
function(type, args) { | ||||
var context = args[0]; | ||||
var txt = context.getAttribute('tooltip_title'); | ||||
if(txt){ | ||||
return true; | ||||
} | ||||
else{ | ||||
return false; | ||||
} | ||||
}); | ||||
// Set the text for the tooltip just before we display it. Lazy method | ||||
myToolTips.contextTriggerEvent.subscribe( | ||||
function(type, args) { | ||||
var context = args[0]; | ||||
var txt = context.getAttribute('tooltip_title'); | ||||
this.cfg.setProperty("text", txt); | ||||
// positioning of tooltip | ||||
var tt_w = this.element.clientWidth; | ||||
var tt_h = this.element.clientHeight; | ||||
var context_w = context.offsetWidth; | ||||
var context_h = context.offsetHeight; | ||||
var pos_x = YAHOO.util.Dom.getX(context); | ||||
var pos_y = YAHOO.util.Dom.getY(context); | ||||
var display_strategy = 'top'; | ||||
var xy_pos = [0,0]; | ||||
switch (display_strategy){ | ||||
case 'top': | ||||
var cur_x = (pos_x+context_w/2)-(tt_w/2); | ||||
var cur_y = pos_y-tt_h-4; | ||||
xy_pos = [cur_x,cur_y]; | ||||
break; | ||||
case 'bottom': | ||||
var cur_x = (pos_x+context_w/2)-(tt_w/2); | ||||
var cur_y = pos_y+context_h+4; | ||||
xy_pos = [cur_x,cur_y]; | ||||
break; | ||||
case 'left': | ||||
var cur_x = (pos_x-tt_w-4); | ||||
var cur_y = pos_y-((tt_h/2)-context_h/2); | ||||
xy_pos = [cur_x,cur_y]; | ||||
break; | ||||
case 'right': | ||||
var cur_x = (pos_x+context_w+4); | ||||
var cur_y = pos_y-((tt_h/2)-context_h/2); | ||||
xy_pos = [cur_x,cur_y]; | ||||
break; | ||||
default: | ||||
var cur_x = (pos_x+context_w/2)-(tt_w/2); | ||||
var cur_y = pos_y-tt_h-4; | ||||
xy_pos = [cur_x,cur_y]; | ||||
break; | ||||
} | ||||
this.cfg.setProperty("xy",xy_pos); | ||||
}); | ||||
//Mouse out | ||||
myToolTips.contextMouseOutEvent.subscribe( | ||||
function(type, args) { | ||||
var context = args[0]; | ||||
}); | ||||
}); | ||||
r671 | ''' | |||
r547 | return literal(js) | |||
tooltip = _ToolTip() | ||||
class _FilesBreadCrumbs(object): | ||||
r671 | ||||
r547 | def __call__(self, repo_name, rev, paths): | |||
url_l = [link_to(repo_name, url('files_home', | ||||
repo_name=repo_name, | ||||
revision=rev, f_path=''))] | ||||
paths_l = paths.split('/') | ||||
r671 | ||||
for cnt, p in enumerate(paths_l): | ||||
r547 | if p != '': | |||
url_l.append(link_to(p, url('files_home', | ||||
repo_name=repo_name, | ||||
revision=rev, | ||||
r671 | f_path='/'.join(paths_l[:cnt + 1])))) | |||
r547 | ||||
return literal('/'.join(url_l)) | ||||
files_breadcrumbs = _FilesBreadCrumbs() | ||||
class CodeHtmlFormatter(HtmlFormatter): | ||||
def wrap(self, source, outfile): | ||||
return self._wrap_div(self._wrap_pre(self._wrap_code(source))) | ||||
def _wrap_code(self, source): | ||||
r671 | for cnt, it in enumerate(source): | |||
r547 | i, t = it | |||
r671 | t = '<div id="#S-%s">%s</div>' % (cnt + 1, t) | |||
r547 | yield i, t | |||
def pygmentize(filenode, **kwargs): | ||||
""" | ||||
pygmentize function using pygments | ||||
r604 | :param filenode: | |||
r547 | """ | |||
return literal(code_highlight(filenode.content, | ||||
filenode.lexer, CodeHtmlFormatter(**kwargs))) | ||||
def pygmentize_annotation(filenode, **kwargs): | ||||
""" | ||||
pygmentize function for annotation | ||||
r604 | :param filenode: | |||
r547 | """ | |||
r671 | ||||
r547 | color_dict = {} | |||
def gen_color(): | ||||
"""generator for getting 10k of evenly distibuted colors using hsv color | ||||
and golden ratio. | ||||
r671 | """ | |||
r547 | import colorsys | |||
n = 10000 | ||||
golden_ratio = 0.618033988749895 | ||||
h = 0.22717784590367374 | ||||
#generate 10k nice web friendly colors in the same order | ||||
for c in xrange(n): | ||||
h += golden_ratio | ||||
h %= 1 | ||||
HSV_tuple = [h, 0.95, 0.95] | ||||
RGB_tuple = colorsys.hsv_to_rgb(*HSV_tuple) | ||||
r671 | yield map(lambda x:str(int(x * 256)), RGB_tuple) | |||
r547 | ||||
cgenerator = gen_color() | ||||
r671 | ||||
r547 | def get_color_string(cs): | |||
if color_dict.has_key(cs): | ||||
col = color_dict[cs] | ||||
else: | ||||
col = color_dict[cs] = cgenerator.next() | ||||
return "color: rgb(%s)! important;" % (', '.join(col)) | ||||
r671 | ||||
r547 | def url_func(changeset): | |||
tooltip_html = "<div style='font-size:0.8em'><b>Author:</b>" + \ | ||||
r671 | " %s<br/><b>Date:</b> %s</b><br/><b>Message:</b> %s<br/></div>" | |||
r547 | tooltip_html = tooltip_html % (changeset.author, | |||
changeset.date, | ||||
tooltip(changeset.message)) | ||||
lnk_format = 'r%-5s:%s' % (changeset.revision, | ||||
changeset.short_id) | ||||
uri = link_to( | ||||
lnk_format, | ||||
url('changeset_home', repo_name=changeset.repository.name, | ||||
revision=changeset.short_id), | ||||
style=get_color_string(changeset.short_id), | ||||
class_='tooltip', | ||||
tooltip_title=tooltip_html | ||||
) | ||||
r671 | ||||
r547 | uri += '\n' | |||
r671 | return uri | |||
r547 | return literal(annotate_highlight(filenode, url_func, **kwargs)) | |||
r671 | ||||
r547 | def repo_name_slug(value): | |||
"""Return slug of name of repository | ||||
This function is called on each creation/modification | ||||
of repository to prevent bad names in repo | ||||
""" | ||||
slug = remove_formatting(value) | ||||
slug = strip_tags(slug) | ||||
r671 | ||||
r547 | for c in """=[]\;'"<>,/~!@#$%^&*()+{}|: """: | |||
slug = slug.replace(c, '-') | ||||
slug = recursive_replace(slug, '-') | ||||
slug = collapse(slug, '-') | ||||
return slug | ||||
def get_changeset_safe(repo, rev): | ||||
from vcs.backends.base import BaseRepository | ||||
from vcs.exceptions import RepositoryError | ||||
if not isinstance(repo, BaseRepository): | ||||
raise Exception('You must pass an Repository ' | ||||
'object as first argument got %s', type(repo)) | ||||
r671 | ||||
r547 | try: | |||
cs = repo.get_changeset(rev) | ||||
except RepositoryError: | ||||
from rhodecode.lib.utils import EmptyChangeset | ||||
cs = EmptyChangeset() | ||||
return cs | ||||
flash = _Flash() | ||||
#=============================================================================== | ||||
# MERCURIAL FILTERS available via h. | ||||
#=============================================================================== | ||||
from mercurial import util | ||||
from mercurial.templatefilters import age as _age, person as _person | ||||
age = lambda x:_age(x) | ||||
capitalize = lambda x: x.capitalize() | ||||
date = lambda x: util.datestr(x) | ||||
email = util.email | ||||
email_or_none = lambda x: util.email(x) if util.email(x) != x else None | ||||
person = lambda x: _person(x) | ||||
hgdate = lambda x: "%d %d" % x | ||||
isodate = lambda x: util.datestr(x, '%Y-%m-%d %H:%M %1%2') | ||||
isodatesec = lambda x: util.datestr(x, '%Y-%m-%d %H:%M:%S %1%2') | ||||
localdate = lambda x: (x[0], util.makedate()[1]) | ||||
rfc822date = lambda x: util.datestr(x, "%a, %d %b %Y %H:%M:%S %1%2") | ||||
rfc822date_notz = lambda x: util.datestr(x, "%a, %d %b %Y %H:%M:%S") | ||||
rfc3339date = lambda x: util.datestr(x, "%Y-%m-%dT%H:%M:%S%1:%2") | ||||
time_ago = lambda x: util.datestr(_age(x), "%a, %d %b %Y %H:%M:%S %1%2") | ||||
#=============================================================================== | ||||
# PERMS | ||||
#=============================================================================== | ||||
from rhodecode.lib.auth import HasPermissionAny, HasPermissionAll, \ | ||||
HasRepoPermissionAny, HasRepoPermissionAll | ||||
#=============================================================================== | ||||
# GRAVATAR URL | ||||
#=============================================================================== | ||||
import hashlib | ||||
import urllib | ||||
from pylons import request | ||||
def gravatar_url(email_address, size=30): | ||||
ssl_enabled = 'https' == request.environ.get('HTTP_X_URL_SCHEME') | ||||
default = 'identicon' | ||||
baseurl_nossl = "http://www.gravatar.com/avatar/" | ||||
baseurl_ssl = "https://secure.gravatar.com/avatar/" | ||||
baseurl = baseurl_ssl if ssl_enabled else baseurl_nossl | ||||
r671 | ||||
r547 | # construct the url | |||
gravatar_url = baseurl + hashlib.md5(email_address.lower()).hexdigest() + "?" | ||||
gravatar_url += urllib.urlencode({'d':default, 's':str(size)}) | ||||
return gravatar_url | ||||
def safe_unicode(str): | ||||
"""safe unicode function. In case of UnicodeDecode error we try to return | ||||
unicode with errors replace, if this failes we return unicode with | ||||
string_escape decoding """ | ||||
r671 | ||||
r547 | try: | |||
u_str = unicode(str) | ||||
except UnicodeDecodeError: | ||||
try: | ||||
u_str = unicode(str, 'utf-8', 'replace') | ||||
except UnicodeDecodeError: | ||||
#incase we have a decode error just represent as byte string | ||||
u_str = unicode(str(str).encode('string_escape')) | ||||
r671 | ||||
r547 | return u_str | |||