Show More
@@ -1,2026 +1,2031 b'' | |||||
1 | # -*- coding: utf-8 -*- |
|
1 | # -*- coding: utf-8 -*- | |
2 |
|
2 | |||
3 | # Copyright (C) 2010-2017 RhodeCode GmbH |
|
3 | # Copyright (C) 2010-2017 RhodeCode GmbH | |
4 | # |
|
4 | # | |
5 | # This program is free software: you can redistribute it and/or modify |
|
5 | # This program is free software: you can redistribute it and/or modify | |
6 | # it under the terms of the GNU Affero General Public License, version 3 |
|
6 | # it under the terms of the GNU Affero General Public License, version 3 | |
7 | # (only), as published by the Free Software Foundation. |
|
7 | # (only), as published by the Free Software Foundation. | |
8 | # |
|
8 | # | |
9 | # This program is distributed in the hope that it will be useful, |
|
9 | # This program is distributed in the hope that it will be useful, | |
10 | # but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
10 | # but WITHOUT ANY WARRANTY; without even the implied warranty of | |
11 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
11 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
12 | # GNU General Public License for more details. |
|
12 | # GNU General Public License for more details. | |
13 | # |
|
13 | # | |
14 | # You should have received a copy of the GNU Affero General Public License |
|
14 | # You should have received a copy of the GNU Affero General Public License | |
15 | # along with this program. If not, see <http://www.gnu.org/licenses/>. |
|
15 | # along with this program. If not, see <http://www.gnu.org/licenses/>. | |
16 | # |
|
16 | # | |
17 | # This program is dual-licensed. If you wish to learn more about the |
|
17 | # This program is dual-licensed. If you wish to learn more about the | |
18 | # RhodeCode Enterprise Edition, including its added features, Support services, |
|
18 | # RhodeCode Enterprise Edition, including its added features, Support services, | |
19 | # and proprietary license terms, please see https://rhodecode.com/licenses/ |
|
19 | # and proprietary license terms, please see https://rhodecode.com/licenses/ | |
20 |
|
20 | |||
21 | """ |
|
21 | """ | |
22 | Helper functions |
|
22 | Helper functions | |
23 |
|
23 | |||
24 | Consists of functions to typically be used within templates, but also |
|
24 | Consists of functions to typically be used within templates, but also | |
25 | available to Controllers. This module is available to both as 'h'. |
|
25 | available to Controllers. This module is available to both as 'h'. | |
26 | """ |
|
26 | """ | |
27 |
|
27 | |||
28 | import random |
|
28 | import random | |
29 | import hashlib |
|
29 | import hashlib | |
30 | import StringIO |
|
30 | import StringIO | |
31 | import urllib |
|
31 | import urllib | |
32 | import math |
|
32 | import math | |
33 | import logging |
|
33 | import logging | |
34 | import re |
|
34 | import re | |
35 | import urlparse |
|
35 | import urlparse | |
36 | import time |
|
36 | import time | |
37 | import string |
|
37 | import string | |
38 | import hashlib |
|
38 | import hashlib | |
39 | from collections import OrderedDict |
|
39 | from collections import OrderedDict | |
40 |
|
40 | |||
41 | import pygments |
|
41 | import pygments | |
42 | import itertools |
|
42 | import itertools | |
43 | import fnmatch |
|
43 | import fnmatch | |
44 |
|
44 | |||
45 | from datetime import datetime |
|
45 | from datetime import datetime | |
46 | from functools import partial |
|
46 | from functools import partial | |
47 | from pygments.formatters.html import HtmlFormatter |
|
47 | from pygments.formatters.html import HtmlFormatter | |
48 | from pygments import highlight as code_highlight |
|
48 | from pygments import highlight as code_highlight | |
49 | from pygments.lexers import ( |
|
49 | from pygments.lexers import ( | |
50 | get_lexer_by_name, get_lexer_for_filename, get_lexer_for_mimetype) |
|
50 | get_lexer_by_name, get_lexer_for_filename, get_lexer_for_mimetype) | |
51 |
|
51 | |||
52 | from pyramid.threadlocal import get_current_request |
|
52 | from pyramid.threadlocal import get_current_request | |
53 |
|
53 | |||
54 | from webhelpers.html import literal, HTML, escape |
|
54 | from webhelpers.html import literal, HTML, escape | |
55 | from webhelpers.html.tools import * |
|
55 | from webhelpers.html.tools import * | |
56 | from webhelpers.html.builder import make_tag |
|
56 | from webhelpers.html.builder import make_tag | |
57 | from webhelpers.html.tags import auto_discovery_link, checkbox, css_classes, \ |
|
57 | from webhelpers.html.tags import auto_discovery_link, checkbox, css_classes, \ | |
58 | end_form, file, form as wh_form, hidden, image, javascript_link, link_to, \ |
|
58 | end_form, file, form as wh_form, hidden, image, javascript_link, link_to, \ | |
59 | link_to_if, link_to_unless, ol, required_legend, select, stylesheet_link, \ |
|
59 | link_to_if, link_to_unless, ol, required_legend, select, stylesheet_link, \ | |
60 | submit, text, password, textarea, title, ul, xml_declaration, radio |
|
60 | submit, text, password, textarea, title, ul, xml_declaration, radio | |
61 | from webhelpers.html.tools import auto_link, button_to, highlight, \ |
|
61 | from webhelpers.html.tools import auto_link, button_to, highlight, \ | |
62 | js_obfuscate, mail_to, strip_links, strip_tags, tag_re |
|
62 | js_obfuscate, mail_to, strip_links, strip_tags, tag_re | |
63 | from webhelpers.pylonslib import Flash as _Flash |
|
63 | from webhelpers.pylonslib import Flash as _Flash | |
64 | from webhelpers.text import chop_at, collapse, convert_accented_entities, \ |
|
64 | from webhelpers.text import chop_at, collapse, convert_accented_entities, \ | |
65 | convert_misc_entities, lchop, plural, rchop, remove_formatting, \ |
|
65 | convert_misc_entities, lchop, plural, rchop, remove_formatting, \ | |
66 | replace_whitespace, urlify, truncate, wrap_paragraphs |
|
66 | replace_whitespace, urlify, truncate, wrap_paragraphs | |
67 | from webhelpers.date import time_ago_in_words |
|
67 | from webhelpers.date import time_ago_in_words | |
68 | from webhelpers.paginate import Page as _Page |
|
68 | from webhelpers.paginate import Page as _Page | |
69 | from webhelpers.html.tags import _set_input_attrs, _set_id_attr, \ |
|
69 | from webhelpers.html.tags import _set_input_attrs, _set_id_attr, \ | |
70 | convert_boolean_attrs, NotGiven, _make_safe_id_component |
|
70 | convert_boolean_attrs, NotGiven, _make_safe_id_component | |
71 | from webhelpers2.number import format_byte_size |
|
71 | from webhelpers2.number import format_byte_size | |
72 |
|
72 | |||
73 | from rhodecode.lib.action_parser import action_parser |
|
73 | from rhodecode.lib.action_parser import action_parser | |
74 | from rhodecode.lib.ext_json import json |
|
74 | from rhodecode.lib.ext_json import json | |
75 | from rhodecode.lib.utils import repo_name_slug, get_custom_lexer |
|
75 | from rhodecode.lib.utils import repo_name_slug, get_custom_lexer | |
76 | from rhodecode.lib.utils2 import str2bool, safe_unicode, safe_str, \ |
|
76 | from rhodecode.lib.utils2 import str2bool, safe_unicode, safe_str, \ | |
77 | get_commit_safe, datetime_to_time, time_to_datetime, time_to_utcdatetime, \ |
|
77 | get_commit_safe, datetime_to_time, time_to_datetime, time_to_utcdatetime, \ | |
78 | AttributeDict, safe_int, md5, md5_safe |
|
78 | AttributeDict, safe_int, md5, md5_safe | |
79 | from rhodecode.lib.markup_renderer import MarkupRenderer, relative_links |
|
79 | from rhodecode.lib.markup_renderer import MarkupRenderer, relative_links | |
80 | from rhodecode.lib.vcs.exceptions import CommitDoesNotExistError |
|
80 | from rhodecode.lib.vcs.exceptions import CommitDoesNotExistError | |
81 | from rhodecode.lib.vcs.backends.base import BaseChangeset, EmptyCommit |
|
81 | from rhodecode.lib.vcs.backends.base import BaseChangeset, EmptyCommit | |
82 | from rhodecode.config.conf import DATE_FORMAT, DATETIME_FORMAT |
|
82 | from rhodecode.config.conf import DATE_FORMAT, DATETIME_FORMAT | |
83 | from rhodecode.model.changeset_status import ChangesetStatusModel |
|
83 | from rhodecode.model.changeset_status import ChangesetStatusModel | |
84 | from rhodecode.model.db import Permission, User, Repository |
|
84 | from rhodecode.model.db import Permission, User, Repository | |
85 | from rhodecode.model.repo_group import RepoGroupModel |
|
85 | from rhodecode.model.repo_group import RepoGroupModel | |
86 | from rhodecode.model.settings import IssueTrackerSettingsModel |
|
86 | from rhodecode.model.settings import IssueTrackerSettingsModel | |
87 |
|
87 | |||
88 | log = logging.getLogger(__name__) |
|
88 | log = logging.getLogger(__name__) | |
89 |
|
89 | |||
90 |
|
90 | |||
91 | DEFAULT_USER = User.DEFAULT_USER |
|
91 | DEFAULT_USER = User.DEFAULT_USER | |
92 | DEFAULT_USER_EMAIL = User.DEFAULT_USER_EMAIL |
|
92 | DEFAULT_USER_EMAIL = User.DEFAULT_USER_EMAIL | |
93 |
|
93 | |||
94 |
|
94 | |||
95 | def url(*args, **kw): |
|
95 | def url(*args, **kw): | |
96 | from pylons import url as pylons_url |
|
96 | from pylons import url as pylons_url | |
97 | return pylons_url(*args, **kw) |
|
97 | return pylons_url(*args, **kw) | |
98 |
|
98 | |||
99 |
|
99 | |||
100 | def pylons_url_current(*args, **kw): |
|
100 | def pylons_url_current(*args, **kw): | |
101 | """ |
|
101 | """ | |
102 | This function overrides pylons.url.current() which returns the current |
|
102 | This function overrides pylons.url.current() which returns the current | |
103 | path so that it will also work from a pyramid only context. This |
|
103 | path so that it will also work from a pyramid only context. This | |
104 | should be removed once port to pyramid is complete. |
|
104 | should be removed once port to pyramid is complete. | |
105 | """ |
|
105 | """ | |
106 | from pylons import url as pylons_url |
|
106 | from pylons import url as pylons_url | |
107 | if not args and not kw: |
|
107 | if not args and not kw: | |
108 | request = get_current_request() |
|
108 | request = get_current_request() | |
109 | return request.path |
|
109 | return request.path | |
110 | return pylons_url.current(*args, **kw) |
|
110 | return pylons_url.current(*args, **kw) | |
111 |
|
111 | |||
112 | url.current = pylons_url_current |
|
112 | url.current = pylons_url_current | |
113 |
|
113 | |||
114 |
|
114 | |||
115 | def url_replace(**qargs): |
|
115 | def url_replace(**qargs): | |
116 | """ Returns the current request url while replacing query string args """ |
|
116 | """ Returns the current request url while replacing query string args """ | |
117 |
|
117 | |||
118 | request = get_current_request() |
|
118 | request = get_current_request() | |
119 | new_args = request.GET.mixed() |
|
119 | new_args = request.GET.mixed() | |
120 | new_args.update(qargs) |
|
120 | new_args.update(qargs) | |
121 | return url('', **new_args) |
|
121 | return url('', **new_args) | |
122 |
|
122 | |||
123 |
|
123 | |||
124 | def asset(path, ver=None, **kwargs): |
|
124 | def asset(path, ver=None, **kwargs): | |
125 | """ |
|
125 | """ | |
126 | Helper to generate a static asset file path for rhodecode assets |
|
126 | Helper to generate a static asset file path for rhodecode assets | |
127 |
|
127 | |||
128 | eg. h.asset('images/image.png', ver='3923') |
|
128 | eg. h.asset('images/image.png', ver='3923') | |
129 |
|
129 | |||
130 | :param path: path of asset |
|
130 | :param path: path of asset | |
131 | :param ver: optional version query param to append as ?ver= |
|
131 | :param ver: optional version query param to append as ?ver= | |
132 | """ |
|
132 | """ | |
133 | request = get_current_request() |
|
133 | request = get_current_request() | |
134 | query = {} |
|
134 | query = {} | |
135 | query.update(kwargs) |
|
135 | query.update(kwargs) | |
136 | if ver: |
|
136 | if ver: | |
137 | query = {'ver': ver} |
|
137 | query = {'ver': ver} | |
138 | return request.static_path( |
|
138 | return request.static_path( | |
139 | 'rhodecode:public/{}'.format(path), _query=query) |
|
139 | 'rhodecode:public/{}'.format(path), _query=query) | |
140 |
|
140 | |||
141 |
|
141 | |||
142 | default_html_escape_table = { |
|
142 | default_html_escape_table = { | |
143 | ord('&'): u'&', |
|
143 | ord('&'): u'&', | |
144 | ord('<'): u'<', |
|
144 | ord('<'): u'<', | |
145 | ord('>'): u'>', |
|
145 | ord('>'): u'>', | |
146 | ord('"'): u'"', |
|
146 | ord('"'): u'"', | |
147 | ord("'"): u''', |
|
147 | ord("'"): u''', | |
148 | } |
|
148 | } | |
149 |
|
149 | |||
150 |
|
150 | |||
151 | def html_escape(text, html_escape_table=default_html_escape_table): |
|
151 | def html_escape(text, html_escape_table=default_html_escape_table): | |
152 | """Produce entities within text.""" |
|
152 | """Produce entities within text.""" | |
153 | return text.translate(html_escape_table) |
|
153 | return text.translate(html_escape_table) | |
154 |
|
154 | |||
155 |
|
155 | |||
156 | def chop_at_smart(s, sub, inclusive=False, suffix_if_chopped=None): |
|
156 | def chop_at_smart(s, sub, inclusive=False, suffix_if_chopped=None): | |
157 | """ |
|
157 | """ | |
158 | Truncate string ``s`` at the first occurrence of ``sub``. |
|
158 | Truncate string ``s`` at the first occurrence of ``sub``. | |
159 |
|
159 | |||
160 | If ``inclusive`` is true, truncate just after ``sub`` rather than at it. |
|
160 | If ``inclusive`` is true, truncate just after ``sub`` rather than at it. | |
161 | """ |
|
161 | """ | |
162 | suffix_if_chopped = suffix_if_chopped or '' |
|
162 | suffix_if_chopped = suffix_if_chopped or '' | |
163 | pos = s.find(sub) |
|
163 | pos = s.find(sub) | |
164 | if pos == -1: |
|
164 | if pos == -1: | |
165 | return s |
|
165 | return s | |
166 |
|
166 | |||
167 | if inclusive: |
|
167 | if inclusive: | |
168 | pos += len(sub) |
|
168 | pos += len(sub) | |
169 |
|
169 | |||
170 | chopped = s[:pos] |
|
170 | chopped = s[:pos] | |
171 | left = s[pos:].strip() |
|
171 | left = s[pos:].strip() | |
172 |
|
172 | |||
173 | if left and suffix_if_chopped: |
|
173 | if left and suffix_if_chopped: | |
174 | chopped += suffix_if_chopped |
|
174 | chopped += suffix_if_chopped | |
175 |
|
175 | |||
176 | return chopped |
|
176 | return chopped | |
177 |
|
177 | |||
178 |
|
178 | |||
179 | def shorter(text, size=20): |
|
179 | def shorter(text, size=20): | |
180 | postfix = '...' |
|
180 | postfix = '...' | |
181 | if len(text) > size: |
|
181 | if len(text) > size: | |
182 | return text[:size - len(postfix)] + postfix |
|
182 | return text[:size - len(postfix)] + postfix | |
183 | return text |
|
183 | return text | |
184 |
|
184 | |||
185 |
|
185 | |||
186 | def _reset(name, value=None, id=NotGiven, type="reset", **attrs): |
|
186 | def _reset(name, value=None, id=NotGiven, type="reset", **attrs): | |
187 | """ |
|
187 | """ | |
188 | Reset button |
|
188 | Reset button | |
189 | """ |
|
189 | """ | |
190 | _set_input_attrs(attrs, type, name, value) |
|
190 | _set_input_attrs(attrs, type, name, value) | |
191 | _set_id_attr(attrs, id, name) |
|
191 | _set_id_attr(attrs, id, name) | |
192 | convert_boolean_attrs(attrs, ["disabled"]) |
|
192 | convert_boolean_attrs(attrs, ["disabled"]) | |
193 | return HTML.input(**attrs) |
|
193 | return HTML.input(**attrs) | |
194 |
|
194 | |||
195 | reset = _reset |
|
195 | reset = _reset | |
196 | safeid = _make_safe_id_component |
|
196 | safeid = _make_safe_id_component | |
197 |
|
197 | |||
198 |
|
198 | |||
199 | def branding(name, length=40): |
|
199 | def branding(name, length=40): | |
200 | return truncate(name, length, indicator="") |
|
200 | return truncate(name, length, indicator="") | |
201 |
|
201 | |||
202 |
|
202 | |||
203 | def FID(raw_id, path): |
|
203 | def FID(raw_id, path): | |
204 | """ |
|
204 | """ | |
205 | Creates a unique ID for filenode based on it's hash of path and commit |
|
205 | Creates a unique ID for filenode based on it's hash of path and commit | |
206 | it's safe to use in urls |
|
206 | it's safe to use in urls | |
207 |
|
207 | |||
208 | :param raw_id: |
|
208 | :param raw_id: | |
209 | :param path: |
|
209 | :param path: | |
210 | """ |
|
210 | """ | |
211 |
|
211 | |||
212 | return 'c-%s-%s' % (short_id(raw_id), md5_safe(path)[:12]) |
|
212 | return 'c-%s-%s' % (short_id(raw_id), md5_safe(path)[:12]) | |
213 |
|
213 | |||
214 |
|
214 | |||
215 | class _GetError(object): |
|
215 | class _GetError(object): | |
216 | """Get error from form_errors, and represent it as span wrapped error |
|
216 | """Get error from form_errors, and represent it as span wrapped error | |
217 | message |
|
217 | message | |
218 |
|
218 | |||
219 | :param field_name: field to fetch errors for |
|
219 | :param field_name: field to fetch errors for | |
220 | :param form_errors: form errors dict |
|
220 | :param form_errors: form errors dict | |
221 | """ |
|
221 | """ | |
222 |
|
222 | |||
223 | def __call__(self, field_name, form_errors): |
|
223 | def __call__(self, field_name, form_errors): | |
224 | tmpl = """<span class="error_msg">%s</span>""" |
|
224 | tmpl = """<span class="error_msg">%s</span>""" | |
225 | if form_errors and field_name in form_errors: |
|
225 | if form_errors and field_name in form_errors: | |
226 | return literal(tmpl % form_errors.get(field_name)) |
|
226 | return literal(tmpl % form_errors.get(field_name)) | |
227 |
|
227 | |||
228 | get_error = _GetError() |
|
228 | get_error = _GetError() | |
229 |
|
229 | |||
230 |
|
230 | |||
231 | class _ToolTip(object): |
|
231 | class _ToolTip(object): | |
232 |
|
232 | |||
233 | def __call__(self, tooltip_title, trim_at=50): |
|
233 | def __call__(self, tooltip_title, trim_at=50): | |
234 | """ |
|
234 | """ | |
235 | Special function just to wrap our text into nice formatted |
|
235 | Special function just to wrap our text into nice formatted | |
236 | autowrapped text |
|
236 | autowrapped text | |
237 |
|
237 | |||
238 | :param tooltip_title: |
|
238 | :param tooltip_title: | |
239 | """ |
|
239 | """ | |
240 | tooltip_title = escape(tooltip_title) |
|
240 | tooltip_title = escape(tooltip_title) | |
241 | tooltip_title = tooltip_title.replace('<', '<').replace('>', '>') |
|
241 | tooltip_title = tooltip_title.replace('<', '<').replace('>', '>') | |
242 | return tooltip_title |
|
242 | return tooltip_title | |
243 | tooltip = _ToolTip() |
|
243 | tooltip = _ToolTip() | |
244 |
|
244 | |||
245 |
|
245 | |||
246 | def files_breadcrumbs(repo_name, commit_id, file_path): |
|
246 | def files_breadcrumbs(repo_name, commit_id, file_path): | |
247 | if isinstance(file_path, str): |
|
247 | if isinstance(file_path, str): | |
248 | file_path = safe_unicode(file_path) |
|
248 | file_path = safe_unicode(file_path) | |
249 |
|
249 | |||
250 | # TODO: johbo: Is this always a url like path, or is this operating |
|
250 | # TODO: johbo: Is this always a url like path, or is this operating | |
251 | # system dependent? |
|
251 | # system dependent? | |
252 | path_segments = file_path.split('/') |
|
252 | path_segments = file_path.split('/') | |
253 |
|
253 | |||
254 | repo_name_html = escape(repo_name) |
|
254 | repo_name_html = escape(repo_name) | |
255 | if len(path_segments) == 1 and path_segments[0] == '': |
|
255 | if len(path_segments) == 1 and path_segments[0] == '': | |
256 | url_segments = [repo_name_html] |
|
256 | url_segments = [repo_name_html] | |
257 | else: |
|
257 | else: | |
258 | url_segments = [ |
|
258 | url_segments = [ | |
259 | link_to( |
|
259 | link_to( | |
260 | repo_name_html, |
|
260 | repo_name_html, | |
261 | url('files_home', |
|
261 | url('files_home', | |
262 | repo_name=repo_name, |
|
262 | repo_name=repo_name, | |
263 | revision=commit_id, |
|
263 | revision=commit_id, | |
264 | f_path=''), |
|
264 | f_path=''), | |
265 | class_='pjax-link')] |
|
265 | class_='pjax-link')] | |
266 |
|
266 | |||
267 | last_cnt = len(path_segments) - 1 |
|
267 | last_cnt = len(path_segments) - 1 | |
268 | for cnt, segment in enumerate(path_segments): |
|
268 | for cnt, segment in enumerate(path_segments): | |
269 | if not segment: |
|
269 | if not segment: | |
270 | continue |
|
270 | continue | |
271 | segment_html = escape(segment) |
|
271 | segment_html = escape(segment) | |
272 |
|
272 | |||
273 | if cnt != last_cnt: |
|
273 | if cnt != last_cnt: | |
274 | url_segments.append( |
|
274 | url_segments.append( | |
275 | link_to( |
|
275 | link_to( | |
276 | segment_html, |
|
276 | segment_html, | |
277 | url('files_home', |
|
277 | url('files_home', | |
278 | repo_name=repo_name, |
|
278 | repo_name=repo_name, | |
279 | revision=commit_id, |
|
279 | revision=commit_id, | |
280 | f_path='/'.join(path_segments[:cnt + 1])), |
|
280 | f_path='/'.join(path_segments[:cnt + 1])), | |
281 | class_='pjax-link')) |
|
281 | class_='pjax-link')) | |
282 | else: |
|
282 | else: | |
283 | url_segments.append(segment_html) |
|
283 | url_segments.append(segment_html) | |
284 |
|
284 | |||
285 | return literal('/'.join(url_segments)) |
|
285 | return literal('/'.join(url_segments)) | |
286 |
|
286 | |||
287 |
|
287 | |||
288 | class CodeHtmlFormatter(HtmlFormatter): |
|
288 | class CodeHtmlFormatter(HtmlFormatter): | |
289 | """ |
|
289 | """ | |
290 | My code Html Formatter for source codes |
|
290 | My code Html Formatter for source codes | |
291 | """ |
|
291 | """ | |
292 |
|
292 | |||
293 | def wrap(self, source, outfile): |
|
293 | def wrap(self, source, outfile): | |
294 | return self._wrap_div(self._wrap_pre(self._wrap_code(source))) |
|
294 | return self._wrap_div(self._wrap_pre(self._wrap_code(source))) | |
295 |
|
295 | |||
296 | def _wrap_code(self, source): |
|
296 | def _wrap_code(self, source): | |
297 | for cnt, it in enumerate(source): |
|
297 | for cnt, it in enumerate(source): | |
298 | i, t = it |
|
298 | i, t = it | |
299 | t = '<div id="L%s">%s</div>' % (cnt + 1, t) |
|
299 | t = '<div id="L%s">%s</div>' % (cnt + 1, t) | |
300 | yield i, t |
|
300 | yield i, t | |
301 |
|
301 | |||
302 | def _wrap_tablelinenos(self, inner): |
|
302 | def _wrap_tablelinenos(self, inner): | |
303 | dummyoutfile = StringIO.StringIO() |
|
303 | dummyoutfile = StringIO.StringIO() | |
304 | lncount = 0 |
|
304 | lncount = 0 | |
305 | for t, line in inner: |
|
305 | for t, line in inner: | |
306 | if t: |
|
306 | if t: | |
307 | lncount += 1 |
|
307 | lncount += 1 | |
308 | dummyoutfile.write(line) |
|
308 | dummyoutfile.write(line) | |
309 |
|
309 | |||
310 | fl = self.linenostart |
|
310 | fl = self.linenostart | |
311 | mw = len(str(lncount + fl - 1)) |
|
311 | mw = len(str(lncount + fl - 1)) | |
312 | sp = self.linenospecial |
|
312 | sp = self.linenospecial | |
313 | st = self.linenostep |
|
313 | st = self.linenostep | |
314 | la = self.lineanchors |
|
314 | la = self.lineanchors | |
315 | aln = self.anchorlinenos |
|
315 | aln = self.anchorlinenos | |
316 | nocls = self.noclasses |
|
316 | nocls = self.noclasses | |
317 | if sp: |
|
317 | if sp: | |
318 | lines = [] |
|
318 | lines = [] | |
319 |
|
319 | |||
320 | for i in range(fl, fl + lncount): |
|
320 | for i in range(fl, fl + lncount): | |
321 | if i % st == 0: |
|
321 | if i % st == 0: | |
322 | if i % sp == 0: |
|
322 | if i % sp == 0: | |
323 | if aln: |
|
323 | if aln: | |
324 | lines.append('<a href="#%s%d" class="special">%*d</a>' % |
|
324 | lines.append('<a href="#%s%d" class="special">%*d</a>' % | |
325 | (la, i, mw, i)) |
|
325 | (la, i, mw, i)) | |
326 | else: |
|
326 | else: | |
327 | lines.append('<span class="special">%*d</span>' % (mw, i)) |
|
327 | lines.append('<span class="special">%*d</span>' % (mw, i)) | |
328 | else: |
|
328 | else: | |
329 | if aln: |
|
329 | if aln: | |
330 | lines.append('<a href="#%s%d">%*d</a>' % (la, i, mw, i)) |
|
330 | lines.append('<a href="#%s%d">%*d</a>' % (la, i, mw, i)) | |
331 | else: |
|
331 | else: | |
332 | lines.append('%*d' % (mw, i)) |
|
332 | lines.append('%*d' % (mw, i)) | |
333 | else: |
|
333 | else: | |
334 | lines.append('') |
|
334 | lines.append('') | |
335 | ls = '\n'.join(lines) |
|
335 | ls = '\n'.join(lines) | |
336 | else: |
|
336 | else: | |
337 | lines = [] |
|
337 | lines = [] | |
338 | for i in range(fl, fl + lncount): |
|
338 | for i in range(fl, fl + lncount): | |
339 | if i % st == 0: |
|
339 | if i % st == 0: | |
340 | if aln: |
|
340 | if aln: | |
341 | lines.append('<a href="#%s%d">%*d</a>' % (la, i, mw, i)) |
|
341 | lines.append('<a href="#%s%d">%*d</a>' % (la, i, mw, i)) | |
342 | else: |
|
342 | else: | |
343 | lines.append('%*d' % (mw, i)) |
|
343 | lines.append('%*d' % (mw, i)) | |
344 | else: |
|
344 | else: | |
345 | lines.append('') |
|
345 | lines.append('') | |
346 | ls = '\n'.join(lines) |
|
346 | ls = '\n'.join(lines) | |
347 |
|
347 | |||
348 | # in case you wonder about the seemingly redundant <div> here: since the |
|
348 | # in case you wonder about the seemingly redundant <div> here: since the | |
349 | # content in the other cell also is wrapped in a div, some browsers in |
|
349 | # content in the other cell also is wrapped in a div, some browsers in | |
350 | # some configurations seem to mess up the formatting... |
|
350 | # some configurations seem to mess up the formatting... | |
351 | if nocls: |
|
351 | if nocls: | |
352 | yield 0, ('<table class="%stable">' % self.cssclass + |
|
352 | yield 0, ('<table class="%stable">' % self.cssclass + | |
353 | '<tr><td><div class="linenodiv" ' |
|
353 | '<tr><td><div class="linenodiv" ' | |
354 | 'style="background-color: #f0f0f0; padding-right: 10px">' |
|
354 | 'style="background-color: #f0f0f0; padding-right: 10px">' | |
355 | '<pre style="line-height: 125%">' + |
|
355 | '<pre style="line-height: 125%">' + | |
356 | ls + '</pre></div></td><td id="hlcode" class="code">') |
|
356 | ls + '</pre></div></td><td id="hlcode" class="code">') | |
357 | else: |
|
357 | else: | |
358 | yield 0, ('<table class="%stable">' % self.cssclass + |
|
358 | yield 0, ('<table class="%stable">' % self.cssclass + | |
359 | '<tr><td class="linenos"><div class="linenodiv"><pre>' + |
|
359 | '<tr><td class="linenos"><div class="linenodiv"><pre>' + | |
360 | ls + '</pre></div></td><td id="hlcode" class="code">') |
|
360 | ls + '</pre></div></td><td id="hlcode" class="code">') | |
361 | yield 0, dummyoutfile.getvalue() |
|
361 | yield 0, dummyoutfile.getvalue() | |
362 | yield 0, '</td></tr></table>' |
|
362 | yield 0, '</td></tr></table>' | |
363 |
|
363 | |||
364 |
|
364 | |||
365 | class SearchContentCodeHtmlFormatter(CodeHtmlFormatter): |
|
365 | class SearchContentCodeHtmlFormatter(CodeHtmlFormatter): | |
366 | def __init__(self, **kw): |
|
366 | def __init__(self, **kw): | |
367 | # only show these line numbers if set |
|
367 | # only show these line numbers if set | |
368 | self.only_lines = kw.pop('only_line_numbers', []) |
|
368 | self.only_lines = kw.pop('only_line_numbers', []) | |
369 | self.query_terms = kw.pop('query_terms', []) |
|
369 | self.query_terms = kw.pop('query_terms', []) | |
370 | self.max_lines = kw.pop('max_lines', 5) |
|
370 | self.max_lines = kw.pop('max_lines', 5) | |
371 | self.line_context = kw.pop('line_context', 3) |
|
371 | self.line_context = kw.pop('line_context', 3) | |
372 | self.url = kw.pop('url', None) |
|
372 | self.url = kw.pop('url', None) | |
373 |
|
373 | |||
374 | super(CodeHtmlFormatter, self).__init__(**kw) |
|
374 | super(CodeHtmlFormatter, self).__init__(**kw) | |
375 |
|
375 | |||
376 | def _wrap_code(self, source): |
|
376 | def _wrap_code(self, source): | |
377 | for cnt, it in enumerate(source): |
|
377 | for cnt, it in enumerate(source): | |
378 | i, t = it |
|
378 | i, t = it | |
379 | t = '<pre>%s</pre>' % t |
|
379 | t = '<pre>%s</pre>' % t | |
380 | yield i, t |
|
380 | yield i, t | |
381 |
|
381 | |||
382 | def _wrap_tablelinenos(self, inner): |
|
382 | def _wrap_tablelinenos(self, inner): | |
383 | yield 0, '<table class="code-highlight %stable">' % self.cssclass |
|
383 | yield 0, '<table class="code-highlight %stable">' % self.cssclass | |
384 |
|
384 | |||
385 | last_shown_line_number = 0 |
|
385 | last_shown_line_number = 0 | |
386 | current_line_number = 1 |
|
386 | current_line_number = 1 | |
387 |
|
387 | |||
388 | for t, line in inner: |
|
388 | for t, line in inner: | |
389 | if not t: |
|
389 | if not t: | |
390 | yield t, line |
|
390 | yield t, line | |
391 | continue |
|
391 | continue | |
392 |
|
392 | |||
393 | if current_line_number in self.only_lines: |
|
393 | if current_line_number in self.only_lines: | |
394 | if last_shown_line_number + 1 != current_line_number: |
|
394 | if last_shown_line_number + 1 != current_line_number: | |
395 | yield 0, '<tr>' |
|
395 | yield 0, '<tr>' | |
396 | yield 0, '<td class="line">...</td>' |
|
396 | yield 0, '<td class="line">...</td>' | |
397 | yield 0, '<td id="hlcode" class="code"></td>' |
|
397 | yield 0, '<td id="hlcode" class="code"></td>' | |
398 | yield 0, '</tr>' |
|
398 | yield 0, '</tr>' | |
399 |
|
399 | |||
400 | yield 0, '<tr>' |
|
400 | yield 0, '<tr>' | |
401 | if self.url: |
|
401 | if self.url: | |
402 | yield 0, '<td class="line"><a href="%s#L%i">%i</a></td>' % ( |
|
402 | yield 0, '<td class="line"><a href="%s#L%i">%i</a></td>' % ( | |
403 | self.url, current_line_number, current_line_number) |
|
403 | self.url, current_line_number, current_line_number) | |
404 | else: |
|
404 | else: | |
405 | yield 0, '<td class="line"><a href="">%i</a></td>' % ( |
|
405 | yield 0, '<td class="line"><a href="">%i</a></td>' % ( | |
406 | current_line_number) |
|
406 | current_line_number) | |
407 | yield 0, '<td id="hlcode" class="code">' + line + '</td>' |
|
407 | yield 0, '<td id="hlcode" class="code">' + line + '</td>' | |
408 | yield 0, '</tr>' |
|
408 | yield 0, '</tr>' | |
409 |
|
409 | |||
410 | last_shown_line_number = current_line_number |
|
410 | last_shown_line_number = current_line_number | |
411 |
|
411 | |||
412 | current_line_number += 1 |
|
412 | current_line_number += 1 | |
413 |
|
413 | |||
414 |
|
414 | |||
415 | yield 0, '</table>' |
|
415 | yield 0, '</table>' | |
416 |
|
416 | |||
417 |
|
417 | |||
418 | def extract_phrases(text_query): |
|
418 | def extract_phrases(text_query): | |
419 | """ |
|
419 | """ | |
420 | Extracts phrases from search term string making sure phrases |
|
420 | Extracts phrases from search term string making sure phrases | |
421 | contained in double quotes are kept together - and discarding empty values |
|
421 | contained in double quotes are kept together - and discarding empty values | |
422 | or fully whitespace values eg. |
|
422 | or fully whitespace values eg. | |
423 |
|
423 | |||
424 | 'some text "a phrase" more' => ['some', 'text', 'a phrase', 'more'] |
|
424 | 'some text "a phrase" more' => ['some', 'text', 'a phrase', 'more'] | |
425 |
|
425 | |||
426 | """ |
|
426 | """ | |
427 |
|
427 | |||
428 | in_phrase = False |
|
428 | in_phrase = False | |
429 | buf = '' |
|
429 | buf = '' | |
430 | phrases = [] |
|
430 | phrases = [] | |
431 | for char in text_query: |
|
431 | for char in text_query: | |
432 | if in_phrase: |
|
432 | if in_phrase: | |
433 | if char == '"': # end phrase |
|
433 | if char == '"': # end phrase | |
434 | phrases.append(buf) |
|
434 | phrases.append(buf) | |
435 | buf = '' |
|
435 | buf = '' | |
436 | in_phrase = False |
|
436 | in_phrase = False | |
437 | continue |
|
437 | continue | |
438 | else: |
|
438 | else: | |
439 | buf += char |
|
439 | buf += char | |
440 | continue |
|
440 | continue | |
441 | else: |
|
441 | else: | |
442 | if char == '"': # start phrase |
|
442 | if char == '"': # start phrase | |
443 | in_phrase = True |
|
443 | in_phrase = True | |
444 | phrases.append(buf) |
|
444 | phrases.append(buf) | |
445 | buf = '' |
|
445 | buf = '' | |
446 | continue |
|
446 | continue | |
447 | elif char == ' ': |
|
447 | elif char == ' ': | |
448 | phrases.append(buf) |
|
448 | phrases.append(buf) | |
449 | buf = '' |
|
449 | buf = '' | |
450 | continue |
|
450 | continue | |
451 | else: |
|
451 | else: | |
452 | buf += char |
|
452 | buf += char | |
453 |
|
453 | |||
454 | phrases.append(buf) |
|
454 | phrases.append(buf) | |
455 | phrases = [phrase.strip() for phrase in phrases if phrase.strip()] |
|
455 | phrases = [phrase.strip() for phrase in phrases if phrase.strip()] | |
456 | return phrases |
|
456 | return phrases | |
457 |
|
457 | |||
458 |
|
458 | |||
459 | def get_matching_offsets(text, phrases): |
|
459 | def get_matching_offsets(text, phrases): | |
460 | """ |
|
460 | """ | |
461 | Returns a list of string offsets in `text` that the list of `terms` match |
|
461 | Returns a list of string offsets in `text` that the list of `terms` match | |
462 |
|
462 | |||
463 | >>> get_matching_offsets('some text here', ['some', 'here']) |
|
463 | >>> get_matching_offsets('some text here', ['some', 'here']) | |
464 | [(0, 4), (10, 14)] |
|
464 | [(0, 4), (10, 14)] | |
465 |
|
465 | |||
466 | """ |
|
466 | """ | |
467 | offsets = [] |
|
467 | offsets = [] | |
468 | for phrase in phrases: |
|
468 | for phrase in phrases: | |
469 | for match in re.finditer(phrase, text): |
|
469 | for match in re.finditer(phrase, text): | |
470 | offsets.append((match.start(), match.end())) |
|
470 | offsets.append((match.start(), match.end())) | |
471 |
|
471 | |||
472 | return offsets |
|
472 | return offsets | |
473 |
|
473 | |||
474 |
|
474 | |||
475 | def normalize_text_for_matching(x): |
|
475 | def normalize_text_for_matching(x): | |
476 | """ |
|
476 | """ | |
477 | Replaces all non alnum characters to spaces and lower cases the string, |
|
477 | Replaces all non alnum characters to spaces and lower cases the string, | |
478 | useful for comparing two text strings without punctuation |
|
478 | useful for comparing two text strings without punctuation | |
479 | """ |
|
479 | """ | |
480 | return re.sub(r'[^\w]', ' ', x.lower()) |
|
480 | return re.sub(r'[^\w]', ' ', x.lower()) | |
481 |
|
481 | |||
482 |
|
482 | |||
483 | def get_matching_line_offsets(lines, terms): |
|
483 | def get_matching_line_offsets(lines, terms): | |
484 | """ Return a set of `lines` indices (starting from 1) matching a |
|
484 | """ Return a set of `lines` indices (starting from 1) matching a | |
485 | text search query, along with `context` lines above/below matching lines |
|
485 | text search query, along with `context` lines above/below matching lines | |
486 |
|
486 | |||
487 | :param lines: list of strings representing lines |
|
487 | :param lines: list of strings representing lines | |
488 | :param terms: search term string to match in lines eg. 'some text' |
|
488 | :param terms: search term string to match in lines eg. 'some text' | |
489 | :param context: number of lines above/below a matching line to add to result |
|
489 | :param context: number of lines above/below a matching line to add to result | |
490 | :param max_lines: cut off for lines of interest |
|
490 | :param max_lines: cut off for lines of interest | |
491 | eg. |
|
491 | eg. | |
492 |
|
492 | |||
493 | text = ''' |
|
493 | text = ''' | |
494 | words words words |
|
494 | words words words | |
495 | words words words |
|
495 | words words words | |
496 | some text some |
|
496 | some text some | |
497 | words words words |
|
497 | words words words | |
498 | words words words |
|
498 | words words words | |
499 | text here what |
|
499 | text here what | |
500 | ''' |
|
500 | ''' | |
501 | get_matching_line_offsets(text, 'text', context=1) |
|
501 | get_matching_line_offsets(text, 'text', context=1) | |
502 | {3: [(5, 9)], 6: [(0, 4)]] |
|
502 | {3: [(5, 9)], 6: [(0, 4)]] | |
503 |
|
503 | |||
504 | """ |
|
504 | """ | |
505 | matching_lines = {} |
|
505 | matching_lines = {} | |
506 | phrases = [normalize_text_for_matching(phrase) |
|
506 | phrases = [normalize_text_for_matching(phrase) | |
507 | for phrase in extract_phrases(terms)] |
|
507 | for phrase in extract_phrases(terms)] | |
508 |
|
508 | |||
509 | for line_index, line in enumerate(lines, start=1): |
|
509 | for line_index, line in enumerate(lines, start=1): | |
510 | match_offsets = get_matching_offsets( |
|
510 | match_offsets = get_matching_offsets( | |
511 | normalize_text_for_matching(line), phrases) |
|
511 | normalize_text_for_matching(line), phrases) | |
512 | if match_offsets: |
|
512 | if match_offsets: | |
513 | matching_lines[line_index] = match_offsets |
|
513 | matching_lines[line_index] = match_offsets | |
514 |
|
514 | |||
515 | return matching_lines |
|
515 | return matching_lines | |
516 |
|
516 | |||
517 |
|
517 | |||
518 | def hsv_to_rgb(h, s, v): |
|
518 | def hsv_to_rgb(h, s, v): | |
519 | """ Convert hsv color values to rgb """ |
|
519 | """ Convert hsv color values to rgb """ | |
520 |
|
520 | |||
521 | if s == 0.0: |
|
521 | if s == 0.0: | |
522 | return v, v, v |
|
522 | return v, v, v | |
523 | i = int(h * 6.0) # XXX assume int() truncates! |
|
523 | i = int(h * 6.0) # XXX assume int() truncates! | |
524 | f = (h * 6.0) - i |
|
524 | f = (h * 6.0) - i | |
525 | p = v * (1.0 - s) |
|
525 | p = v * (1.0 - s) | |
526 | q = v * (1.0 - s * f) |
|
526 | q = v * (1.0 - s * f) | |
527 | t = v * (1.0 - s * (1.0 - f)) |
|
527 | t = v * (1.0 - s * (1.0 - f)) | |
528 | i = i % 6 |
|
528 | i = i % 6 | |
529 | if i == 0: |
|
529 | if i == 0: | |
530 | return v, t, p |
|
530 | return v, t, p | |
531 | if i == 1: |
|
531 | if i == 1: | |
532 | return q, v, p |
|
532 | return q, v, p | |
533 | if i == 2: |
|
533 | if i == 2: | |
534 | return p, v, t |
|
534 | return p, v, t | |
535 | if i == 3: |
|
535 | if i == 3: | |
536 | return p, q, v |
|
536 | return p, q, v | |
537 | if i == 4: |
|
537 | if i == 4: | |
538 | return t, p, v |
|
538 | return t, p, v | |
539 | if i == 5: |
|
539 | if i == 5: | |
540 | return v, p, q |
|
540 | return v, p, q | |
541 |
|
541 | |||
542 |
|
542 | |||
543 | def unique_color_generator(n=10000, saturation=0.10, lightness=0.95): |
|
543 | def unique_color_generator(n=10000, saturation=0.10, lightness=0.95): | |
544 | """ |
|
544 | """ | |
545 | Generator for getting n of evenly distributed colors using |
|
545 | Generator for getting n of evenly distributed colors using | |
546 | hsv color and golden ratio. It always return same order of colors |
|
546 | hsv color and golden ratio. It always return same order of colors | |
547 |
|
547 | |||
548 | :param n: number of colors to generate |
|
548 | :param n: number of colors to generate | |
549 | :param saturation: saturation of returned colors |
|
549 | :param saturation: saturation of returned colors | |
550 | :param lightness: lightness of returned colors |
|
550 | :param lightness: lightness of returned colors | |
551 | :returns: RGB tuple |
|
551 | :returns: RGB tuple | |
552 | """ |
|
552 | """ | |
553 |
|
553 | |||
554 | golden_ratio = 0.618033988749895 |
|
554 | golden_ratio = 0.618033988749895 | |
555 | h = 0.22717784590367374 |
|
555 | h = 0.22717784590367374 | |
556 |
|
556 | |||
557 | for _ in xrange(n): |
|
557 | for _ in xrange(n): | |
558 | h += golden_ratio |
|
558 | h += golden_ratio | |
559 | h %= 1 |
|
559 | h %= 1 | |
560 | HSV_tuple = [h, saturation, lightness] |
|
560 | HSV_tuple = [h, saturation, lightness] | |
561 | RGB_tuple = hsv_to_rgb(*HSV_tuple) |
|
561 | RGB_tuple = hsv_to_rgb(*HSV_tuple) | |
562 | yield map(lambda x: str(int(x * 256)), RGB_tuple) |
|
562 | yield map(lambda x: str(int(x * 256)), RGB_tuple) | |
563 |
|
563 | |||
564 |
|
564 | |||
565 | def color_hasher(n=10000, saturation=0.10, lightness=0.95): |
|
565 | def color_hasher(n=10000, saturation=0.10, lightness=0.95): | |
566 | """ |
|
566 | """ | |
567 | Returns a function which when called with an argument returns a unique |
|
567 | Returns a function which when called with an argument returns a unique | |
568 | color for that argument, eg. |
|
568 | color for that argument, eg. | |
569 |
|
569 | |||
570 | :param n: number of colors to generate |
|
570 | :param n: number of colors to generate | |
571 | :param saturation: saturation of returned colors |
|
571 | :param saturation: saturation of returned colors | |
572 | :param lightness: lightness of returned colors |
|
572 | :param lightness: lightness of returned colors | |
573 | :returns: css RGB string |
|
573 | :returns: css RGB string | |
574 |
|
574 | |||
575 | >>> color_hash = color_hasher() |
|
575 | >>> color_hash = color_hasher() | |
576 | >>> color_hash('hello') |
|
576 | >>> color_hash('hello') | |
577 | 'rgb(34, 12, 59)' |
|
577 | 'rgb(34, 12, 59)' | |
578 | >>> color_hash('hello') |
|
578 | >>> color_hash('hello') | |
579 | 'rgb(34, 12, 59)' |
|
579 | 'rgb(34, 12, 59)' | |
580 | >>> color_hash('other') |
|
580 | >>> color_hash('other') | |
581 | 'rgb(90, 224, 159)' |
|
581 | 'rgb(90, 224, 159)' | |
582 | """ |
|
582 | """ | |
583 |
|
583 | |||
584 | color_dict = {} |
|
584 | color_dict = {} | |
585 | cgenerator = unique_color_generator( |
|
585 | cgenerator = unique_color_generator( | |
586 | saturation=saturation, lightness=lightness) |
|
586 | saturation=saturation, lightness=lightness) | |
587 |
|
587 | |||
588 | def get_color_string(thing): |
|
588 | def get_color_string(thing): | |
589 | if thing in color_dict: |
|
589 | if thing in color_dict: | |
590 | col = color_dict[thing] |
|
590 | col = color_dict[thing] | |
591 | else: |
|
591 | else: | |
592 | col = color_dict[thing] = cgenerator.next() |
|
592 | col = color_dict[thing] = cgenerator.next() | |
593 | return "rgb(%s)" % (', '.join(col)) |
|
593 | return "rgb(%s)" % (', '.join(col)) | |
594 |
|
594 | |||
595 | return get_color_string |
|
595 | return get_color_string | |
596 |
|
596 | |||
597 |
|
597 | |||
598 | def get_lexer_safe(mimetype=None, filepath=None): |
|
598 | def get_lexer_safe(mimetype=None, filepath=None): | |
599 | """ |
|
599 | """ | |
600 | Tries to return a relevant pygments lexer using mimetype/filepath name, |
|
600 | Tries to return a relevant pygments lexer using mimetype/filepath name, | |
601 | defaulting to plain text if none could be found |
|
601 | defaulting to plain text if none could be found | |
602 | """ |
|
602 | """ | |
603 | lexer = None |
|
603 | lexer = None | |
604 | try: |
|
604 | try: | |
605 | if mimetype: |
|
605 | if mimetype: | |
606 | lexer = get_lexer_for_mimetype(mimetype) |
|
606 | lexer = get_lexer_for_mimetype(mimetype) | |
607 | if not lexer: |
|
607 | if not lexer: | |
608 | lexer = get_lexer_for_filename(filepath) |
|
608 | lexer = get_lexer_for_filename(filepath) | |
609 | except pygments.util.ClassNotFound: |
|
609 | except pygments.util.ClassNotFound: | |
610 | pass |
|
610 | pass | |
611 |
|
611 | |||
612 | if not lexer: |
|
612 | if not lexer: | |
613 | lexer = get_lexer_by_name('text') |
|
613 | lexer = get_lexer_by_name('text') | |
614 |
|
614 | |||
615 | return lexer |
|
615 | return lexer | |
616 |
|
616 | |||
617 |
|
617 | |||
618 | def get_lexer_for_filenode(filenode): |
|
618 | def get_lexer_for_filenode(filenode): | |
619 | lexer = get_custom_lexer(filenode.extension) or filenode.lexer |
|
619 | lexer = get_custom_lexer(filenode.extension) or filenode.lexer | |
620 | return lexer |
|
620 | return lexer | |
621 |
|
621 | |||
622 |
|
622 | |||
623 | def pygmentize(filenode, **kwargs): |
|
623 | def pygmentize(filenode, **kwargs): | |
624 | """ |
|
624 | """ | |
625 | pygmentize function using pygments |
|
625 | pygmentize function using pygments | |
626 |
|
626 | |||
627 | :param filenode: |
|
627 | :param filenode: | |
628 | """ |
|
628 | """ | |
629 | lexer = get_lexer_for_filenode(filenode) |
|
629 | lexer = get_lexer_for_filenode(filenode) | |
630 | return literal(code_highlight(filenode.content, lexer, |
|
630 | return literal(code_highlight(filenode.content, lexer, | |
631 | CodeHtmlFormatter(**kwargs))) |
|
631 | CodeHtmlFormatter(**kwargs))) | |
632 |
|
632 | |||
633 |
|
633 | |||
634 | def is_following_repo(repo_name, user_id): |
|
634 | def is_following_repo(repo_name, user_id): | |
635 | from rhodecode.model.scm import ScmModel |
|
635 | from rhodecode.model.scm import ScmModel | |
636 | return ScmModel().is_following_repo(repo_name, user_id) |
|
636 | return ScmModel().is_following_repo(repo_name, user_id) | |
637 |
|
637 | |||
638 |
|
638 | |||
639 | class _Message(object): |
|
639 | class _Message(object): | |
640 | """A message returned by ``Flash.pop_messages()``. |
|
640 | """A message returned by ``Flash.pop_messages()``. | |
641 |
|
641 | |||
642 | Converting the message to a string returns the message text. Instances |
|
642 | Converting the message to a string returns the message text. Instances | |
643 | also have the following attributes: |
|
643 | also have the following attributes: | |
644 |
|
644 | |||
645 | * ``message``: the message text. |
|
645 | * ``message``: the message text. | |
646 | * ``category``: the category specified when the message was created. |
|
646 | * ``category``: the category specified when the message was created. | |
647 | """ |
|
647 | """ | |
648 |
|
648 | |||
649 | def __init__(self, category, message): |
|
649 | def __init__(self, category, message): | |
650 | self.category = category |
|
650 | self.category = category | |
651 | self.message = message |
|
651 | self.message = message | |
652 |
|
652 | |||
653 | def __str__(self): |
|
653 | def __str__(self): | |
654 | return self.message |
|
654 | return self.message | |
655 |
|
655 | |||
656 | __unicode__ = __str__ |
|
656 | __unicode__ = __str__ | |
657 |
|
657 | |||
658 | def __html__(self): |
|
658 | def __html__(self): | |
659 | return escape(safe_unicode(self.message)) |
|
659 | return escape(safe_unicode(self.message)) | |
660 |
|
660 | |||
661 |
|
661 | |||
662 | class Flash(_Flash): |
|
662 | class Flash(_Flash): | |
663 |
|
663 | |||
664 | def pop_messages(self, request=None): |
|
664 | def pop_messages(self, request=None): | |
665 | """Return all accumulated messages and delete them from the session. |
|
665 | """Return all accumulated messages and delete them from the session. | |
666 |
|
666 | |||
667 | The return value is a list of ``Message`` objects. |
|
667 | The return value is a list of ``Message`` objects. | |
668 | """ |
|
668 | """ | |
669 | messages = [] |
|
669 | messages = [] | |
670 |
|
670 | |||
671 | if request: |
|
671 | if request: | |
672 | session = request.session |
|
672 | session = request.session | |
673 | else: |
|
673 | else: | |
674 | from pylons import session |
|
674 | from pylons import session | |
675 |
|
675 | |||
676 | # Pop the 'old' pylons flash messages. They are tuples of the form |
|
676 | # Pop the 'old' pylons flash messages. They are tuples of the form | |
677 | # (category, message) |
|
677 | # (category, message) | |
678 | for cat, msg in session.pop(self.session_key, []): |
|
678 | for cat, msg in session.pop(self.session_key, []): | |
679 | messages.append(_Message(cat, msg)) |
|
679 | messages.append(_Message(cat, msg)) | |
680 |
|
680 | |||
681 | # Pop the 'new' pyramid flash messages for each category as list |
|
681 | # Pop the 'new' pyramid flash messages for each category as list | |
682 | # of strings. |
|
682 | # of strings. | |
683 | for cat in self.categories: |
|
683 | for cat in self.categories: | |
684 | for msg in session.pop_flash(queue=cat): |
|
684 | for msg in session.pop_flash(queue=cat): | |
685 | messages.append(_Message(cat, msg)) |
|
685 | messages.append(_Message(cat, msg)) | |
686 | # Map messages from the default queue to the 'notice' category. |
|
686 | # Map messages from the default queue to the 'notice' category. | |
687 | for msg in session.pop_flash(): |
|
687 | for msg in session.pop_flash(): | |
688 | messages.append(_Message('notice', msg)) |
|
688 | messages.append(_Message('notice', msg)) | |
689 |
|
689 | |||
690 | session.save() |
|
690 | session.save() | |
691 | return messages |
|
691 | return messages | |
692 |
|
692 | |||
693 | def json_alerts(self, request=None): |
|
693 | def json_alerts(self, request=None): | |
694 | payloads = [] |
|
694 | payloads = [] | |
695 | messages = flash.pop_messages(request=request) |
|
695 | messages = flash.pop_messages(request=request) | |
696 | if messages: |
|
696 | if messages: | |
697 | for message in messages: |
|
697 | for message in messages: | |
698 | subdata = {} |
|
698 | subdata = {} | |
699 | if hasattr(message.message, 'rsplit'): |
|
699 | if hasattr(message.message, 'rsplit'): | |
700 | flash_data = message.message.rsplit('|DELIM|', 1) |
|
700 | flash_data = message.message.rsplit('|DELIM|', 1) | |
701 | org_message = flash_data[0] |
|
701 | org_message = flash_data[0] | |
702 | if len(flash_data) > 1: |
|
702 | if len(flash_data) > 1: | |
703 | subdata = json.loads(flash_data[1]) |
|
703 | subdata = json.loads(flash_data[1]) | |
704 | else: |
|
704 | else: | |
705 | org_message = message.message |
|
705 | org_message = message.message | |
706 | payloads.append({ |
|
706 | payloads.append({ | |
707 | 'message': { |
|
707 | 'message': { | |
708 | 'message': u'{}'.format(org_message), |
|
708 | 'message': u'{}'.format(org_message), | |
709 | 'level': message.category, |
|
709 | 'level': message.category, | |
710 | 'force': True, |
|
710 | 'force': True, | |
711 | 'subdata': subdata |
|
711 | 'subdata': subdata | |
712 | } |
|
712 | } | |
713 | }) |
|
713 | }) | |
714 | return json.dumps(payloads) |
|
714 | return json.dumps(payloads) | |
715 |
|
715 | |||
716 | flash = Flash() |
|
716 | flash = Flash() | |
717 |
|
717 | |||
718 | #============================================================================== |
|
718 | #============================================================================== | |
719 | # SCM FILTERS available via h. |
|
719 | # SCM FILTERS available via h. | |
720 | #============================================================================== |
|
720 | #============================================================================== | |
721 | from rhodecode.lib.vcs.utils import author_name, author_email |
|
721 | from rhodecode.lib.vcs.utils import author_name, author_email | |
722 | from rhodecode.lib.utils2 import credentials_filter, age as _age |
|
722 | from rhodecode.lib.utils2 import credentials_filter, age as _age | |
723 | from rhodecode.model.db import User, ChangesetStatus |
|
723 | from rhodecode.model.db import User, ChangesetStatus | |
724 |
|
724 | |||
725 | age = _age |
|
725 | age = _age | |
726 | capitalize = lambda x: x.capitalize() |
|
726 | capitalize = lambda x: x.capitalize() | |
727 | email = author_email |
|
727 | email = author_email | |
728 | short_id = lambda x: x[:12] |
|
728 | short_id = lambda x: x[:12] | |
729 | hide_credentials = lambda x: ''.join(credentials_filter(x)) |
|
729 | hide_credentials = lambda x: ''.join(credentials_filter(x)) | |
730 |
|
730 | |||
731 |
|
731 | |||
732 | def age_component(datetime_iso, value=None, time_is_local=False): |
|
732 | def age_component(datetime_iso, value=None, time_is_local=False): | |
733 | title = value or format_date(datetime_iso) |
|
733 | title = value or format_date(datetime_iso) | |
734 | tzinfo = '+00:00' |
|
734 | tzinfo = '+00:00' | |
735 |
|
735 | |||
736 | # detect if we have a timezone info, otherwise, add it |
|
736 | # detect if we have a timezone info, otherwise, add it | |
737 | if isinstance(datetime_iso, datetime) and not datetime_iso.tzinfo: |
|
737 | if isinstance(datetime_iso, datetime) and not datetime_iso.tzinfo: | |
738 | if time_is_local: |
|
738 | if time_is_local: | |
739 | tzinfo = time.strftime("+%H:%M", |
|
739 | tzinfo = time.strftime("+%H:%M", | |
740 | time.gmtime( |
|
740 | time.gmtime( | |
741 | (datetime.now() - datetime.utcnow()).seconds + 1 |
|
741 | (datetime.now() - datetime.utcnow()).seconds + 1 | |
742 | ) |
|
742 | ) | |
743 | ) |
|
743 | ) | |
744 |
|
744 | |||
745 | return literal( |
|
745 | return literal( | |
746 | '<time class="timeago tooltip" ' |
|
746 | '<time class="timeago tooltip" ' | |
747 | 'title="{1}{2}" datetime="{0}{2}">{1}</time>'.format( |
|
747 | 'title="{1}{2}" datetime="{0}{2}">{1}</time>'.format( | |
748 | datetime_iso, title, tzinfo)) |
|
748 | datetime_iso, title, tzinfo)) | |
749 |
|
749 | |||
750 |
|
750 | |||
751 | def _shorten_commit_id(commit_id): |
|
751 | def _shorten_commit_id(commit_id): | |
752 | from rhodecode import CONFIG |
|
752 | from rhodecode import CONFIG | |
753 | def_len = safe_int(CONFIG.get('rhodecode_show_sha_length', 12)) |
|
753 | def_len = safe_int(CONFIG.get('rhodecode_show_sha_length', 12)) | |
754 | return commit_id[:def_len] |
|
754 | return commit_id[:def_len] | |
755 |
|
755 | |||
756 |
|
756 | |||
757 | def show_id(commit): |
|
757 | def show_id(commit): | |
758 | """ |
|
758 | """ | |
759 | Configurable function that shows ID |
|
759 | Configurable function that shows ID | |
760 | by default it's r123:fffeeefffeee |
|
760 | by default it's r123:fffeeefffeee | |
761 |
|
761 | |||
762 | :param commit: commit instance |
|
762 | :param commit: commit instance | |
763 | """ |
|
763 | """ | |
764 | from rhodecode import CONFIG |
|
764 | from rhodecode import CONFIG | |
765 | show_idx = str2bool(CONFIG.get('rhodecode_show_revision_number', True)) |
|
765 | show_idx = str2bool(CONFIG.get('rhodecode_show_revision_number', True)) | |
766 |
|
766 | |||
767 | raw_id = _shorten_commit_id(commit.raw_id) |
|
767 | raw_id = _shorten_commit_id(commit.raw_id) | |
768 | if show_idx: |
|
768 | if show_idx: | |
769 | return 'r%s:%s' % (commit.idx, raw_id) |
|
769 | return 'r%s:%s' % (commit.idx, raw_id) | |
770 | else: |
|
770 | else: | |
771 | return '%s' % (raw_id, ) |
|
771 | return '%s' % (raw_id, ) | |
772 |
|
772 | |||
773 |
|
773 | |||
774 | def format_date(date): |
|
774 | def format_date(date): | |
775 | """ |
|
775 | """ | |
776 | use a standardized formatting for dates used in RhodeCode |
|
776 | use a standardized formatting for dates used in RhodeCode | |
777 |
|
777 | |||
778 | :param date: date/datetime object |
|
778 | :param date: date/datetime object | |
779 | :return: formatted date |
|
779 | :return: formatted date | |
780 | """ |
|
780 | """ | |
781 |
|
781 | |||
782 | if date: |
|
782 | if date: | |
783 | _fmt = "%a, %d %b %Y %H:%M:%S" |
|
783 | _fmt = "%a, %d %b %Y %H:%M:%S" | |
784 | return safe_unicode(date.strftime(_fmt)) |
|
784 | return safe_unicode(date.strftime(_fmt)) | |
785 |
|
785 | |||
786 | return u"" |
|
786 | return u"" | |
787 |
|
787 | |||
788 |
|
788 | |||
789 | class _RepoChecker(object): |
|
789 | class _RepoChecker(object): | |
790 |
|
790 | |||
791 | def __init__(self, backend_alias): |
|
791 | def __init__(self, backend_alias): | |
792 | self._backend_alias = backend_alias |
|
792 | self._backend_alias = backend_alias | |
793 |
|
793 | |||
794 | def __call__(self, repository): |
|
794 | def __call__(self, repository): | |
795 | if hasattr(repository, 'alias'): |
|
795 | if hasattr(repository, 'alias'): | |
796 | _type = repository.alias |
|
796 | _type = repository.alias | |
797 | elif hasattr(repository, 'repo_type'): |
|
797 | elif hasattr(repository, 'repo_type'): | |
798 | _type = repository.repo_type |
|
798 | _type = repository.repo_type | |
799 | else: |
|
799 | else: | |
800 | _type = repository |
|
800 | _type = repository | |
801 | return _type == self._backend_alias |
|
801 | return _type == self._backend_alias | |
802 |
|
802 | |||
803 | is_git = _RepoChecker('git') |
|
803 | is_git = _RepoChecker('git') | |
804 | is_hg = _RepoChecker('hg') |
|
804 | is_hg = _RepoChecker('hg') | |
805 | is_svn = _RepoChecker('svn') |
|
805 | is_svn = _RepoChecker('svn') | |
806 |
|
806 | |||
807 |
|
807 | |||
808 | def get_repo_type_by_name(repo_name): |
|
808 | def get_repo_type_by_name(repo_name): | |
809 | repo = Repository.get_by_repo_name(repo_name) |
|
809 | repo = Repository.get_by_repo_name(repo_name) | |
810 | return repo.repo_type |
|
810 | return repo.repo_type | |
811 |
|
811 | |||
812 |
|
812 | |||
813 | def is_svn_without_proxy(repository): |
|
813 | def is_svn_without_proxy(repository): | |
814 | if is_svn(repository): |
|
814 | if is_svn(repository): | |
815 | from rhodecode.model.settings import VcsSettingsModel |
|
815 | from rhodecode.model.settings import VcsSettingsModel | |
816 | conf = VcsSettingsModel().get_ui_settings_as_config_obj() |
|
816 | conf = VcsSettingsModel().get_ui_settings_as_config_obj() | |
817 | return not str2bool(conf.get('vcs_svn_proxy', 'http_requests_enabled')) |
|
817 | return not str2bool(conf.get('vcs_svn_proxy', 'http_requests_enabled')) | |
818 | return False |
|
818 | return False | |
819 |
|
819 | |||
820 |
|
820 | |||
821 | def discover_user(author): |
|
821 | def discover_user(author): | |
822 | """ |
|
822 | """ | |
823 | Tries to discover RhodeCode User based on the autho string. Author string |
|
823 | Tries to discover RhodeCode User based on the autho string. Author string | |
824 | is typically `FirstName LastName <email@address.com>` |
|
824 | is typically `FirstName LastName <email@address.com>` | |
825 | """ |
|
825 | """ | |
826 |
|
826 | |||
827 | # if author is already an instance use it for extraction |
|
827 | # if author is already an instance use it for extraction | |
828 | if isinstance(author, User): |
|
828 | if isinstance(author, User): | |
829 | return author |
|
829 | return author | |
830 |
|
830 | |||
831 | # Valid email in the attribute passed, see if they're in the system |
|
831 | # Valid email in the attribute passed, see if they're in the system | |
832 | _email = author_email(author) |
|
832 | _email = author_email(author) | |
833 | if _email != '': |
|
833 | if _email != '': | |
834 | user = User.get_by_email(_email, case_insensitive=True, cache=True) |
|
834 | user = User.get_by_email(_email, case_insensitive=True, cache=True) | |
835 | if user is not None: |
|
835 | if user is not None: | |
836 | return user |
|
836 | return user | |
837 |
|
837 | |||
838 | # Maybe it's a username, we try to extract it and fetch by username ? |
|
838 | # Maybe it's a username, we try to extract it and fetch by username ? | |
839 | _author = author_name(author) |
|
839 | _author = author_name(author) | |
840 | user = User.get_by_username(_author, case_insensitive=True, cache=True) |
|
840 | user = User.get_by_username(_author, case_insensitive=True, cache=True) | |
841 | if user is not None: |
|
841 | if user is not None: | |
842 | return user |
|
842 | return user | |
843 |
|
843 | |||
844 | return None |
|
844 | return None | |
845 |
|
845 | |||
846 |
|
846 | |||
847 | def email_or_none(author): |
|
847 | def email_or_none(author): | |
848 | # extract email from the commit string |
|
848 | # extract email from the commit string | |
849 | _email = author_email(author) |
|
849 | _email = author_email(author) | |
850 |
|
850 | |||
851 | # If we have an email, use it, otherwise |
|
851 | # If we have an email, use it, otherwise | |
852 | # see if it contains a username we can get an email from |
|
852 | # see if it contains a username we can get an email from | |
853 | if _email != '': |
|
853 | if _email != '': | |
854 | return _email |
|
854 | return _email | |
855 | else: |
|
855 | else: | |
856 | user = User.get_by_username( |
|
856 | user = User.get_by_username( | |
857 | author_name(author), case_insensitive=True, cache=True) |
|
857 | author_name(author), case_insensitive=True, cache=True) | |
858 |
|
858 | |||
859 | if user is not None: |
|
859 | if user is not None: | |
860 | return user.email |
|
860 | return user.email | |
861 |
|
861 | |||
862 | # No valid email, not a valid user in the system, none! |
|
862 | # No valid email, not a valid user in the system, none! | |
863 | return None |
|
863 | return None | |
864 |
|
864 | |||
865 |
|
865 | |||
866 | def link_to_user(author, length=0, **kwargs): |
|
866 | def link_to_user(author, length=0, **kwargs): | |
867 | user = discover_user(author) |
|
867 | user = discover_user(author) | |
868 | # user can be None, but if we have it already it means we can re-use it |
|
868 | # user can be None, but if we have it already it means we can re-use it | |
869 | # in the person() function, so we save 1 intensive-query |
|
869 | # in the person() function, so we save 1 intensive-query | |
870 | if user: |
|
870 | if user: | |
871 | author = user |
|
871 | author = user | |
872 |
|
872 | |||
873 | display_person = person(author, 'username_or_name_or_email') |
|
873 | display_person = person(author, 'username_or_name_or_email') | |
874 | if length: |
|
874 | if length: | |
875 | display_person = shorter(display_person, length) |
|
875 | display_person = shorter(display_person, length) | |
876 |
|
876 | |||
877 | if user: |
|
877 | if user: | |
878 | return link_to( |
|
878 | return link_to( | |
879 | escape(display_person), |
|
879 | escape(display_person), | |
880 | route_path('user_profile', username=user.username), |
|
880 | route_path('user_profile', username=user.username), | |
881 | **kwargs) |
|
881 | **kwargs) | |
882 | else: |
|
882 | else: | |
883 | return escape(display_person) |
|
883 | return escape(display_person) | |
884 |
|
884 | |||
885 |
|
885 | |||
886 | def person(author, show_attr="username_and_name"): |
|
886 | def person(author, show_attr="username_and_name"): | |
887 | user = discover_user(author) |
|
887 | user = discover_user(author) | |
888 | if user: |
|
888 | if user: | |
889 | return getattr(user, show_attr) |
|
889 | return getattr(user, show_attr) | |
890 | else: |
|
890 | else: | |
891 | _author = author_name(author) |
|
891 | _author = author_name(author) | |
892 | _email = email(author) |
|
892 | _email = email(author) | |
893 | return _author or _email |
|
893 | return _author or _email | |
894 |
|
894 | |||
895 |
|
895 | |||
896 | def author_string(email): |
|
896 | def author_string(email): | |
897 | if email: |
|
897 | if email: | |
898 | user = User.get_by_email(email, case_insensitive=True, cache=True) |
|
898 | user = User.get_by_email(email, case_insensitive=True, cache=True) | |
899 | if user: |
|
899 | if user: | |
900 | if user.first_name or user.last_name: |
|
900 | if user.first_name or user.last_name: | |
901 | return '%s %s <%s>' % ( |
|
901 | return '%s %s <%s>' % ( | |
902 | user.first_name, user.last_name, email) |
|
902 | user.first_name, user.last_name, email) | |
903 | else: |
|
903 | else: | |
904 | return email |
|
904 | return email | |
905 | else: |
|
905 | else: | |
906 | return email |
|
906 | return email | |
907 | else: |
|
907 | else: | |
908 | return None |
|
908 | return None | |
909 |
|
909 | |||
910 |
|
910 | |||
911 | def person_by_id(id_, show_attr="username_and_name"): |
|
911 | def person_by_id(id_, show_attr="username_and_name"): | |
912 | # attr to return from fetched user |
|
912 | # attr to return from fetched user | |
913 | person_getter = lambda usr: getattr(usr, show_attr) |
|
913 | person_getter = lambda usr: getattr(usr, show_attr) | |
914 |
|
914 | |||
915 | #maybe it's an ID ? |
|
915 | #maybe it's an ID ? | |
916 | if str(id_).isdigit() or isinstance(id_, int): |
|
916 | if str(id_).isdigit() or isinstance(id_, int): | |
917 | id_ = int(id_) |
|
917 | id_ = int(id_) | |
918 | user = User.get(id_) |
|
918 | user = User.get(id_) | |
919 | if user is not None: |
|
919 | if user is not None: | |
920 | return person_getter(user) |
|
920 | return person_getter(user) | |
921 | return id_ |
|
921 | return id_ | |
922 |
|
922 | |||
923 |
|
923 | |||
924 | def gravatar_with_user(author, show_disabled=False): |
|
924 | def gravatar_with_user(author, show_disabled=False): | |
925 | from rhodecode.lib.utils import PartialRenderer |
|
925 | from rhodecode.lib.utils import PartialRenderer | |
926 | _render = PartialRenderer('base/base.mako') |
|
926 | _render = PartialRenderer('base/base.mako') | |
927 | return _render('gravatar_with_user', author, show_disabled=show_disabled) |
|
927 | return _render('gravatar_with_user', author, show_disabled=show_disabled) | |
928 |
|
928 | |||
929 |
|
929 | |||
930 | def desc_stylize(value): |
|
930 | def desc_stylize(value): | |
931 | """ |
|
931 | """ | |
932 | converts tags from value into html equivalent |
|
932 | converts tags from value into html equivalent | |
933 |
|
933 | |||
934 | :param value: |
|
934 | :param value: | |
935 | """ |
|
935 | """ | |
936 | if not value: |
|
936 | if not value: | |
937 | return '' |
|
937 | return '' | |
938 |
|
938 | |||
939 | value = re.sub(r'\[see\ \=\>\ *([a-zA-Z0-9\/\=\?\&\ \:\/\.\-]*)\]', |
|
939 | value = re.sub(r'\[see\ \=\>\ *([a-zA-Z0-9\/\=\?\&\ \:\/\.\-]*)\]', | |
940 | '<div class="metatag" tag="see">see => \\1 </div>', value) |
|
940 | '<div class="metatag" tag="see">see => \\1 </div>', value) | |
941 | value = re.sub(r'\[license\ \=\>\ *([a-zA-Z0-9\/\=\?\&\ \:\/\.\-]*)\]', |
|
941 | value = re.sub(r'\[license\ \=\>\ *([a-zA-Z0-9\/\=\?\&\ \:\/\.\-]*)\]', | |
942 | '<div class="metatag" tag="license"><a href="http:\/\/www.opensource.org/licenses/\\1">\\1</a></div>', value) |
|
942 | '<div class="metatag" tag="license"><a href="http:\/\/www.opensource.org/licenses/\\1">\\1</a></div>', value) | |
943 | value = re.sub(r'\[(requires|recommends|conflicts|base)\ \=\>\ *([a-zA-Z0-9\-\/]*)\]', |
|
943 | value = re.sub(r'\[(requires|recommends|conflicts|base)\ \=\>\ *([a-zA-Z0-9\-\/]*)\]', | |
944 | '<div class="metatag" tag="\\1">\\1 => <a href="/\\2">\\2</a></div>', value) |
|
944 | '<div class="metatag" tag="\\1">\\1 => <a href="/\\2">\\2</a></div>', value) | |
945 | value = re.sub(r'\[(lang|language)\ \=\>\ *([a-zA-Z\-\/\#\+]*)\]', |
|
945 | value = re.sub(r'\[(lang|language)\ \=\>\ *([a-zA-Z\-\/\#\+]*)\]', | |
946 | '<div class="metatag" tag="lang">\\2</div>', value) |
|
946 | '<div class="metatag" tag="lang">\\2</div>', value) | |
947 | value = re.sub(r'\[([a-z]+)\]', |
|
947 | value = re.sub(r'\[([a-z]+)\]', | |
948 | '<div class="metatag" tag="\\1">\\1</div>', value) |
|
948 | '<div class="metatag" tag="\\1">\\1</div>', value) | |
949 |
|
949 | |||
950 | return value |
|
950 | return value | |
951 |
|
951 | |||
952 |
|
952 | |||
953 | def escaped_stylize(value): |
|
953 | def escaped_stylize(value): | |
954 | """ |
|
954 | """ | |
955 | converts tags from value into html equivalent, but escaping its value first |
|
955 | converts tags from value into html equivalent, but escaping its value first | |
956 | """ |
|
956 | """ | |
957 | if not value: |
|
957 | if not value: | |
958 | return '' |
|
958 | return '' | |
959 |
|
959 | |||
960 | # Using default webhelper escape method, but has to force it as a |
|
960 | # Using default webhelper escape method, but has to force it as a | |
961 | # plain unicode instead of a markup tag to be used in regex expressions |
|
961 | # plain unicode instead of a markup tag to be used in regex expressions | |
962 | value = unicode(escape(safe_unicode(value))) |
|
962 | value = unicode(escape(safe_unicode(value))) | |
963 |
|
963 | |||
964 | value = re.sub(r'\[see\ \=\>\ *([a-zA-Z0-9\/\=\?\&\ \:\/\.\-]*)\]', |
|
964 | value = re.sub(r'\[see\ \=\>\ *([a-zA-Z0-9\/\=\?\&\ \:\/\.\-]*)\]', | |
965 | '<div class="metatag" tag="see">see => \\1 </div>', value) |
|
965 | '<div class="metatag" tag="see">see => \\1 </div>', value) | |
966 | value = re.sub(r'\[license\ \=\>\ *([a-zA-Z0-9\/\=\?\&\ \:\/\.\-]*)\]', |
|
966 | value = re.sub(r'\[license\ \=\>\ *([a-zA-Z0-9\/\=\?\&\ \:\/\.\-]*)\]', | |
967 | '<div class="metatag" tag="license"><a href="http:\/\/www.opensource.org/licenses/\\1">\\1</a></div>', value) |
|
967 | '<div class="metatag" tag="license"><a href="http:\/\/www.opensource.org/licenses/\\1">\\1</a></div>', value) | |
968 | value = re.sub(r'\[(requires|recommends|conflicts|base)\ \=\>\ *([a-zA-Z0-9\-\/]*)\]', |
|
968 | value = re.sub(r'\[(requires|recommends|conflicts|base)\ \=\>\ *([a-zA-Z0-9\-\/]*)\]', | |
969 | '<div class="metatag" tag="\\1">\\1 => <a href="/\\2">\\2</a></div>', value) |
|
969 | '<div class="metatag" tag="\\1">\\1 => <a href="/\\2">\\2</a></div>', value) | |
970 | value = re.sub(r'\[(lang|language)\ \=\>\ *([a-zA-Z\-\/\#\+]*)\]', |
|
970 | value = re.sub(r'\[(lang|language)\ \=\>\ *([a-zA-Z\-\/\#\+]*)\]', | |
971 | '<div class="metatag" tag="lang">\\2</div>', value) |
|
971 | '<div class="metatag" tag="lang">\\2</div>', value) | |
972 | value = re.sub(r'\[([a-z]+)\]', |
|
972 | value = re.sub(r'\[([a-z]+)\]', | |
973 | '<div class="metatag" tag="\\1">\\1</div>', value) |
|
973 | '<div class="metatag" tag="\\1">\\1</div>', value) | |
974 |
|
974 | |||
975 | return value |
|
975 | return value | |
976 |
|
976 | |||
977 |
|
977 | |||
978 | def bool2icon(value): |
|
978 | def bool2icon(value): | |
979 | """ |
|
979 | """ | |
980 | Returns boolean value of a given value, represented as html element with |
|
980 | Returns boolean value of a given value, represented as html element with | |
981 | classes that will represent icons |
|
981 | classes that will represent icons | |
982 |
|
982 | |||
983 | :param value: given value to convert to html node |
|
983 | :param value: given value to convert to html node | |
984 | """ |
|
984 | """ | |
985 |
|
985 | |||
986 | if value: # does bool conversion |
|
986 | if value: # does bool conversion | |
987 | return HTML.tag('i', class_="icon-true") |
|
987 | return HTML.tag('i', class_="icon-true") | |
988 | else: # not true as bool |
|
988 | else: # not true as bool | |
989 | return HTML.tag('i', class_="icon-false") |
|
989 | return HTML.tag('i', class_="icon-false") | |
990 |
|
990 | |||
991 |
|
991 | |||
992 | #============================================================================== |
|
992 | #============================================================================== | |
993 | # PERMS |
|
993 | # PERMS | |
994 | #============================================================================== |
|
994 | #============================================================================== | |
995 | from rhodecode.lib.auth import HasPermissionAny, HasPermissionAll, \ |
|
995 | from rhodecode.lib.auth import HasPermissionAny, HasPermissionAll, \ | |
996 | HasRepoPermissionAny, HasRepoPermissionAll, HasRepoGroupPermissionAll, \ |
|
996 | HasRepoPermissionAny, HasRepoPermissionAll, HasRepoGroupPermissionAll, \ | |
997 | HasRepoGroupPermissionAny, HasRepoPermissionAnyApi, get_csrf_token, \ |
|
997 | HasRepoGroupPermissionAny, HasRepoPermissionAnyApi, get_csrf_token, \ | |
998 | csrf_token_key |
|
998 | csrf_token_key | |
999 |
|
999 | |||
1000 |
|
1000 | |||
1001 | #============================================================================== |
|
1001 | #============================================================================== | |
1002 | # GRAVATAR URL |
|
1002 | # GRAVATAR URL | |
1003 | #============================================================================== |
|
1003 | #============================================================================== | |
1004 | class InitialsGravatar(object): |
|
1004 | class InitialsGravatar(object): | |
1005 | def __init__(self, email_address, first_name, last_name, size=30, |
|
1005 | def __init__(self, email_address, first_name, last_name, size=30, | |
1006 | background=None, text_color='#fff'): |
|
1006 | background=None, text_color='#fff'): | |
1007 | self.size = size |
|
1007 | self.size = size | |
1008 | self.first_name = first_name |
|
1008 | self.first_name = first_name | |
1009 | self.last_name = last_name |
|
1009 | self.last_name = last_name | |
1010 | self.email_address = email_address |
|
1010 | self.email_address = email_address | |
1011 | self.background = background or self.str2color(email_address) |
|
1011 | self.background = background or self.str2color(email_address) | |
1012 | self.text_color = text_color |
|
1012 | self.text_color = text_color | |
1013 |
|
1013 | |||
1014 | def get_color_bank(self): |
|
1014 | def get_color_bank(self): | |
1015 | """ |
|
1015 | """ | |
1016 | returns a predefined list of colors that gravatars can use. |
|
1016 | returns a predefined list of colors that gravatars can use. | |
1017 | Those are randomized distinct colors that guarantee readability and |
|
1017 | Those are randomized distinct colors that guarantee readability and | |
1018 | uniqueness. |
|
1018 | uniqueness. | |
1019 |
|
1019 | |||
1020 | generated with: http://phrogz.net/css/distinct-colors.html |
|
1020 | generated with: http://phrogz.net/css/distinct-colors.html | |
1021 | """ |
|
1021 | """ | |
1022 | return [ |
|
1022 | return [ | |
1023 | '#bf3030', '#a67f53', '#00ff00', '#5989b3', '#392040', '#d90000', |
|
1023 | '#bf3030', '#a67f53', '#00ff00', '#5989b3', '#392040', '#d90000', | |
1024 | '#402910', '#204020', '#79baf2', '#a700b3', '#bf6060', '#7f5320', |
|
1024 | '#402910', '#204020', '#79baf2', '#a700b3', '#bf6060', '#7f5320', | |
1025 | '#008000', '#003059', '#ee00ff', '#ff0000', '#8c4b00', '#007300', |
|
1025 | '#008000', '#003059', '#ee00ff', '#ff0000', '#8c4b00', '#007300', | |
1026 | '#005fb3', '#de73e6', '#ff4040', '#ffaa00', '#3df255', '#203140', |
|
1026 | '#005fb3', '#de73e6', '#ff4040', '#ffaa00', '#3df255', '#203140', | |
1027 | '#47004d', '#591616', '#664400', '#59b365', '#0d2133', '#83008c', |
|
1027 | '#47004d', '#591616', '#664400', '#59b365', '#0d2133', '#83008c', | |
1028 | '#592d2d', '#bf9f60', '#73e682', '#1d3f73', '#73006b', '#402020', |
|
1028 | '#592d2d', '#bf9f60', '#73e682', '#1d3f73', '#73006b', '#402020', | |
1029 | '#b2862d', '#397341', '#597db3', '#e600d6', '#a60000', '#736039', |
|
1029 | '#b2862d', '#397341', '#597db3', '#e600d6', '#a60000', '#736039', | |
1030 | '#00b318', '#79aaf2', '#330d30', '#ff8080', '#403010', '#16591f', |
|
1030 | '#00b318', '#79aaf2', '#330d30', '#ff8080', '#403010', '#16591f', | |
1031 | '#002459', '#8c4688', '#e50000', '#ffbf40', '#00732e', '#102340', |
|
1031 | '#002459', '#8c4688', '#e50000', '#ffbf40', '#00732e', '#102340', | |
1032 | '#bf60ac', '#8c4646', '#cc8800', '#00a642', '#1d3473', '#b32d98', |
|
1032 | '#bf60ac', '#8c4646', '#cc8800', '#00a642', '#1d3473', '#b32d98', | |
1033 | '#660e00', '#ffd580', '#80ffb2', '#7391e6', '#733967', '#d97b6c', |
|
1033 | '#660e00', '#ffd580', '#80ffb2', '#7391e6', '#733967', '#d97b6c', | |
1034 | '#8c5e00', '#59b389', '#3967e6', '#590047', '#73281d', '#665200', |
|
1034 | '#8c5e00', '#59b389', '#3967e6', '#590047', '#73281d', '#665200', | |
1035 | '#00e67a', '#2d50b3', '#8c2377', '#734139', '#b2982d', '#16593a', |
|
1035 | '#00e67a', '#2d50b3', '#8c2377', '#734139', '#b2982d', '#16593a', | |
1036 | '#001859', '#ff00aa', '#a65e53', '#ffcc00', '#0d3321', '#2d3959', |
|
1036 | '#001859', '#ff00aa', '#a65e53', '#ffcc00', '#0d3321', '#2d3959', | |
1037 | '#731d56', '#401610', '#4c3d00', '#468c6c', '#002ca6', '#d936a3', |
|
1037 | '#731d56', '#401610', '#4c3d00', '#468c6c', '#002ca6', '#d936a3', | |
1038 | '#d94c36', '#403920', '#36d9a3', '#0d1733', '#592d4a', '#993626', |
|
1038 | '#d94c36', '#403920', '#36d9a3', '#0d1733', '#592d4a', '#993626', | |
1039 | '#cca300', '#00734d', '#46598c', '#8c005e', '#7f1100', '#8c7000', |
|
1039 | '#cca300', '#00734d', '#46598c', '#8c005e', '#7f1100', '#8c7000', | |
1040 | '#00a66f', '#7382e6', '#b32d74', '#d9896c', '#ffe680', '#1d7362', |
|
1040 | '#00a66f', '#7382e6', '#b32d74', '#d9896c', '#ffe680', '#1d7362', | |
1041 | '#364cd9', '#73003d', '#d93a00', '#998a4d', '#59b3a1', '#5965b3', |
|
1041 | '#364cd9', '#73003d', '#d93a00', '#998a4d', '#59b3a1', '#5965b3', | |
1042 | '#e5007a', '#73341d', '#665f00', '#00b38f', '#0018b3', '#59163a', |
|
1042 | '#e5007a', '#73341d', '#665f00', '#00b38f', '#0018b3', '#59163a', | |
1043 | '#b2502d', '#bfb960', '#00ffcc', '#23318c', '#a6537f', '#734939', |
|
1043 | '#b2502d', '#bfb960', '#00ffcc', '#23318c', '#a6537f', '#734939', | |
1044 | '#b2a700', '#104036', '#3d3df2', '#402031', '#e56739', '#736f39', |
|
1044 | '#b2a700', '#104036', '#3d3df2', '#402031', '#e56739', '#736f39', | |
1045 | '#79f2ea', '#000059', '#401029', '#4c1400', '#ffee00', '#005953', |
|
1045 | '#79f2ea', '#000059', '#401029', '#4c1400', '#ffee00', '#005953', | |
1046 | '#101040', '#990052', '#402820', '#403d10', '#00ffee', '#0000d9', |
|
1046 | '#101040', '#990052', '#402820', '#403d10', '#00ffee', '#0000d9', | |
1047 | '#ff80c4', '#a66953', '#eeff00', '#00ccbe', '#8080ff', '#e673a1', |
|
1047 | '#ff80c4', '#a66953', '#eeff00', '#00ccbe', '#8080ff', '#e673a1', | |
1048 | '#a62c00', '#474d00', '#1a3331', '#46468c', '#733950', '#662900', |
|
1048 | '#a62c00', '#474d00', '#1a3331', '#46468c', '#733950', '#662900', | |
1049 | '#858c23', '#238c85', '#0f0073', '#b20047', '#d9986c', '#becc00', |
|
1049 | '#858c23', '#238c85', '#0f0073', '#b20047', '#d9986c', '#becc00', | |
1050 | '#396f73', '#281d73', '#ff0066', '#ff6600', '#dee673', '#59adb3', |
|
1050 | '#396f73', '#281d73', '#ff0066', '#ff6600', '#dee673', '#59adb3', | |
1051 | '#6559b3', '#590024', '#b2622d', '#98b32d', '#36ced9', '#332d59', |
|
1051 | '#6559b3', '#590024', '#b2622d', '#98b32d', '#36ced9', '#332d59', | |
1052 | '#40001a', '#733f1d', '#526600', '#005359', '#242040', '#bf6079', |
|
1052 | '#40001a', '#733f1d', '#526600', '#005359', '#242040', '#bf6079', | |
1053 | '#735039', '#cef23d', '#007780', '#5630bf', '#66001b', '#b24700', |
|
1053 | '#735039', '#cef23d', '#007780', '#5630bf', '#66001b', '#b24700', | |
1054 | '#acbf60', '#1d6273', '#25008c', '#731d34', '#a67453', '#50592d', |
|
1054 | '#acbf60', '#1d6273', '#25008c', '#731d34', '#a67453', '#50592d', | |
1055 | '#00ccff', '#6600ff', '#ff0044', '#4c1f00', '#8a994d', '#79daf2', |
|
1055 | '#00ccff', '#6600ff', '#ff0044', '#4c1f00', '#8a994d', '#79daf2', | |
1056 | '#a173e6', '#d93662', '#402310', '#aaff00', '#2d98b3', '#8c40ff', |
|
1056 | '#a173e6', '#d93662', '#402310', '#aaff00', '#2d98b3', '#8c40ff', | |
1057 | '#592d39', '#ff8c40', '#354020', '#103640', '#1a0040', '#331a20', |
|
1057 | '#592d39', '#ff8c40', '#354020', '#103640', '#1a0040', '#331a20', | |
1058 | '#331400', '#334d00', '#1d5673', '#583973', '#7f0022', '#4c3626', |
|
1058 | '#331400', '#334d00', '#1d5673', '#583973', '#7f0022', '#4c3626', | |
1059 | '#88cc00', '#36a3d9', '#3d0073', '#d9364c', '#33241a', '#698c23', |
|
1059 | '#88cc00', '#36a3d9', '#3d0073', '#d9364c', '#33241a', '#698c23', | |
1060 | '#5995b3', '#300059', '#e57382', '#7f3300', '#366600', '#00aaff', |
|
1060 | '#5995b3', '#300059', '#e57382', '#7f3300', '#366600', '#00aaff', | |
1061 | '#3a1659', '#733941', '#663600', '#74b32d', '#003c59', '#7f53a6', |
|
1061 | '#3a1659', '#733941', '#663600', '#74b32d', '#003c59', '#7f53a6', | |
1062 | '#73000f', '#ff8800', '#baf279', '#79caf2', '#291040', '#a6293a', |
|
1062 | '#73000f', '#ff8800', '#baf279', '#79caf2', '#291040', '#a6293a', | |
1063 | '#b2742d', '#587339', '#0077b3', '#632699', '#400009', '#d9a66c', |
|
1063 | '#b2742d', '#587339', '#0077b3', '#632699', '#400009', '#d9a66c', | |
1064 | '#294010', '#2d4a59', '#aa00ff', '#4c131b', '#b25f00', '#5ce600', |
|
1064 | '#294010', '#2d4a59', '#aa00ff', '#4c131b', '#b25f00', '#5ce600', | |
1065 | '#267399', '#a336d9', '#990014', '#664e33', '#86bf60', '#0088ff', |
|
1065 | '#267399', '#a336d9', '#990014', '#664e33', '#86bf60', '#0088ff', | |
1066 | '#7700b3', '#593a16', '#073300', '#1d4b73', '#ac60bf', '#e59539', |
|
1066 | '#7700b3', '#593a16', '#073300', '#1d4b73', '#ac60bf', '#e59539', | |
1067 | '#4f8c46', '#368dd9', '#5c0073' |
|
1067 | '#4f8c46', '#368dd9', '#5c0073' | |
1068 | ] |
|
1068 | ] | |
1069 |
|
1069 | |||
1070 | def rgb_to_hex_color(self, rgb_tuple): |
|
1070 | def rgb_to_hex_color(self, rgb_tuple): | |
1071 | """ |
|
1071 | """ | |
1072 | Converts an rgb_tuple passed to an hex color. |
|
1072 | Converts an rgb_tuple passed to an hex color. | |
1073 |
|
1073 | |||
1074 | :param rgb_tuple: tuple with 3 ints represents rgb color space |
|
1074 | :param rgb_tuple: tuple with 3 ints represents rgb color space | |
1075 | """ |
|
1075 | """ | |
1076 | return '#' + ("".join(map(chr, rgb_tuple)).encode('hex')) |
|
1076 | return '#' + ("".join(map(chr, rgb_tuple)).encode('hex')) | |
1077 |
|
1077 | |||
1078 | def email_to_int_list(self, email_str): |
|
1078 | def email_to_int_list(self, email_str): | |
1079 | """ |
|
1079 | """ | |
1080 | Get every byte of the hex digest value of email and turn it to integer. |
|
1080 | Get every byte of the hex digest value of email and turn it to integer. | |
1081 | It's going to be always between 0-255 |
|
1081 | It's going to be always between 0-255 | |
1082 | """ |
|
1082 | """ | |
1083 | digest = md5_safe(email_str.lower()) |
|
1083 | digest = md5_safe(email_str.lower()) | |
1084 | return [int(digest[i * 2:i * 2 + 2], 16) for i in range(16)] |
|
1084 | return [int(digest[i * 2:i * 2 + 2], 16) for i in range(16)] | |
1085 |
|
1085 | |||
1086 | def pick_color_bank_index(self, email_str, color_bank): |
|
1086 | def pick_color_bank_index(self, email_str, color_bank): | |
1087 | return self.email_to_int_list(email_str)[0] % len(color_bank) |
|
1087 | return self.email_to_int_list(email_str)[0] % len(color_bank) | |
1088 |
|
1088 | |||
1089 | def str2color(self, email_str): |
|
1089 | def str2color(self, email_str): | |
1090 | """ |
|
1090 | """ | |
1091 | Tries to map in a stable algorithm an email to color |
|
1091 | Tries to map in a stable algorithm an email to color | |
1092 |
|
1092 | |||
1093 | :param email_str: |
|
1093 | :param email_str: | |
1094 | """ |
|
1094 | """ | |
1095 | color_bank = self.get_color_bank() |
|
1095 | color_bank = self.get_color_bank() | |
1096 | # pick position (module it's length so we always find it in the |
|
1096 | # pick position (module it's length so we always find it in the | |
1097 | # bank even if it's smaller than 256 values |
|
1097 | # bank even if it's smaller than 256 values | |
1098 | pos = self.pick_color_bank_index(email_str, color_bank) |
|
1098 | pos = self.pick_color_bank_index(email_str, color_bank) | |
1099 | return color_bank[pos] |
|
1099 | return color_bank[pos] | |
1100 |
|
1100 | |||
1101 | def normalize_email(self, email_address): |
|
1101 | def normalize_email(self, email_address): | |
1102 | import unicodedata |
|
1102 | import unicodedata | |
1103 | # default host used to fill in the fake/missing email |
|
1103 | # default host used to fill in the fake/missing email | |
1104 | default_host = u'localhost' |
|
1104 | default_host = u'localhost' | |
1105 |
|
1105 | |||
1106 | if not email_address: |
|
1106 | if not email_address: | |
1107 | email_address = u'%s@%s' % (User.DEFAULT_USER, default_host) |
|
1107 | email_address = u'%s@%s' % (User.DEFAULT_USER, default_host) | |
1108 |
|
1108 | |||
1109 | email_address = safe_unicode(email_address) |
|
1109 | email_address = safe_unicode(email_address) | |
1110 |
|
1110 | |||
1111 | if u'@' not in email_address: |
|
1111 | if u'@' not in email_address: | |
1112 | email_address = u'%s@%s' % (email_address, default_host) |
|
1112 | email_address = u'%s@%s' % (email_address, default_host) | |
1113 |
|
1113 | |||
1114 | if email_address.endswith(u'@'): |
|
1114 | if email_address.endswith(u'@'): | |
1115 | email_address = u'%s%s' % (email_address, default_host) |
|
1115 | email_address = u'%s%s' % (email_address, default_host) | |
1116 |
|
1116 | |||
1117 | email_address = unicodedata.normalize('NFKD', email_address)\ |
|
1117 | email_address = unicodedata.normalize('NFKD', email_address)\ | |
1118 | .encode('ascii', 'ignore') |
|
1118 | .encode('ascii', 'ignore') | |
1119 | return email_address |
|
1119 | return email_address | |
1120 |
|
1120 | |||
1121 | def get_initials(self): |
|
1121 | def get_initials(self): | |
1122 | """ |
|
1122 | """ | |
1123 | Returns 2 letter initials calculated based on the input. |
|
1123 | Returns 2 letter initials calculated based on the input. | |
1124 | The algorithm picks first given email address, and takes first letter |
|
1124 | The algorithm picks first given email address, and takes first letter | |
1125 | of part before @, and then the first letter of server name. In case |
|
1125 | of part before @, and then the first letter of server name. In case | |
1126 | the part before @ is in a format of `somestring.somestring2` it replaces |
|
1126 | the part before @ is in a format of `somestring.somestring2` it replaces | |
1127 | the server letter with first letter of somestring2 |
|
1127 | the server letter with first letter of somestring2 | |
1128 |
|
1128 | |||
1129 | In case function was initialized with both first and lastname, this |
|
1129 | In case function was initialized with both first and lastname, this | |
1130 | overrides the extraction from email by first letter of the first and |
|
1130 | overrides the extraction from email by first letter of the first and | |
1131 | last name. We add special logic to that functionality, In case Full name |
|
1131 | last name. We add special logic to that functionality, In case Full name | |
1132 | is compound, like Guido Von Rossum, we use last part of the last name |
|
1132 | is compound, like Guido Von Rossum, we use last part of the last name | |
1133 | (Von Rossum) picking `R`. |
|
1133 | (Von Rossum) picking `R`. | |
1134 |
|
1134 | |||
1135 | Function also normalizes the non-ascii characters to they ascii |
|
1135 | Function also normalizes the non-ascii characters to they ascii | |
1136 | representation, eg Δ => A |
|
1136 | representation, eg Δ => A | |
1137 | """ |
|
1137 | """ | |
1138 | import unicodedata |
|
1138 | import unicodedata | |
1139 | # replace non-ascii to ascii |
|
1139 | # replace non-ascii to ascii | |
1140 | first_name = unicodedata.normalize( |
|
1140 | first_name = unicodedata.normalize( | |
1141 | 'NFKD', safe_unicode(self.first_name)).encode('ascii', 'ignore') |
|
1141 | 'NFKD', safe_unicode(self.first_name)).encode('ascii', 'ignore') | |
1142 | last_name = unicodedata.normalize( |
|
1142 | last_name = unicodedata.normalize( | |
1143 | 'NFKD', safe_unicode(self.last_name)).encode('ascii', 'ignore') |
|
1143 | 'NFKD', safe_unicode(self.last_name)).encode('ascii', 'ignore') | |
1144 |
|
1144 | |||
1145 | # do NFKD encoding, and also make sure email has proper format |
|
1145 | # do NFKD encoding, and also make sure email has proper format | |
1146 | email_address = self.normalize_email(self.email_address) |
|
1146 | email_address = self.normalize_email(self.email_address) | |
1147 |
|
1147 | |||
1148 | # first push the email initials |
|
1148 | # first push the email initials | |
1149 | prefix, server = email_address.split('@', 1) |
|
1149 | prefix, server = email_address.split('@', 1) | |
1150 |
|
1150 | |||
1151 | # check if prefix is maybe a 'first_name.last_name' syntax |
|
1151 | # check if prefix is maybe a 'first_name.last_name' syntax | |
1152 | _dot_split = prefix.rsplit('.', 1) |
|
1152 | _dot_split = prefix.rsplit('.', 1) | |
1153 | if len(_dot_split) == 2: |
|
1153 | if len(_dot_split) == 2: | |
1154 | initials = [_dot_split[0][0], _dot_split[1][0]] |
|
1154 | initials = [_dot_split[0][0], _dot_split[1][0]] | |
1155 | else: |
|
1155 | else: | |
1156 | initials = [prefix[0], server[0]] |
|
1156 | initials = [prefix[0], server[0]] | |
1157 |
|
1157 | |||
1158 | # then try to replace either first_name or last_name |
|
1158 | # then try to replace either first_name or last_name | |
1159 | fn_letter = (first_name or " ")[0].strip() |
|
1159 | fn_letter = (first_name or " ")[0].strip() | |
1160 | ln_letter = (last_name.split(' ', 1)[-1] or " ")[0].strip() |
|
1160 | ln_letter = (last_name.split(' ', 1)[-1] or " ")[0].strip() | |
1161 |
|
1161 | |||
1162 | if fn_letter: |
|
1162 | if fn_letter: | |
1163 | initials[0] = fn_letter |
|
1163 | initials[0] = fn_letter | |
1164 |
|
1164 | |||
1165 | if ln_letter: |
|
1165 | if ln_letter: | |
1166 | initials[1] = ln_letter |
|
1166 | initials[1] = ln_letter | |
1167 |
|
1167 | |||
1168 | return ''.join(initials).upper() |
|
1168 | return ''.join(initials).upper() | |
1169 |
|
1169 | |||
1170 | def get_img_data_by_type(self, font_family, img_type): |
|
1170 | def get_img_data_by_type(self, font_family, img_type): | |
1171 | default_user = """ |
|
1171 | default_user = """ | |
1172 | <svg xmlns="http://www.w3.org/2000/svg" |
|
1172 | <svg xmlns="http://www.w3.org/2000/svg" | |
1173 | version="1.1" x="0px" y="0px" width="{size}" height="{size}" |
|
1173 | version="1.1" x="0px" y="0px" width="{size}" height="{size}" | |
1174 | viewBox="-15 -10 439.165 429.164" |
|
1174 | viewBox="-15 -10 439.165 429.164" | |
1175 |
|
1175 | |||
1176 | xml:space="preserve" |
|
1176 | xml:space="preserve" | |
1177 | style="background:{background};" > |
|
1177 | style="background:{background};" > | |
1178 |
|
1178 | |||
1179 | <path d="M204.583,216.671c50.664,0,91.74-48.075, |
|
1179 | <path d="M204.583,216.671c50.664,0,91.74-48.075, | |
1180 | 91.74-107.378c0-82.237-41.074-107.377-91.74-107.377 |
|
1180 | 91.74-107.378c0-82.237-41.074-107.377-91.74-107.377 | |
1181 | c-50.668,0-91.74,25.14-91.74,107.377C112.844, |
|
1181 | c-50.668,0-91.74,25.14-91.74,107.377C112.844, | |
1182 | 168.596,153.916,216.671, |
|
1182 | 168.596,153.916,216.671, | |
1183 | 204.583,216.671z" fill="{text_color}"/> |
|
1183 | 204.583,216.671z" fill="{text_color}"/> | |
1184 | <path d="M407.164,374.717L360.88, |
|
1184 | <path d="M407.164,374.717L360.88, | |
1185 | 270.454c-2.117-4.771-5.836-8.728-10.465-11.138l-71.83-37.392 |
|
1185 | 270.454c-2.117-4.771-5.836-8.728-10.465-11.138l-71.83-37.392 | |
1186 | c-1.584-0.823-3.502-0.663-4.926,0.415c-20.316, |
|
1186 | c-1.584-0.823-3.502-0.663-4.926,0.415c-20.316, | |
1187 | 15.366-44.203,23.488-69.076,23.488c-24.877, |
|
1187 | 15.366-44.203,23.488-69.076,23.488c-24.877, | |
1188 | 0-48.762-8.122-69.078-23.488 |
|
1188 | 0-48.762-8.122-69.078-23.488 | |
1189 | c-1.428-1.078-3.346-1.238-4.93-0.415L58.75, |
|
1189 | c-1.428-1.078-3.346-1.238-4.93-0.415L58.75, | |
1190 | 259.316c-4.631,2.41-8.346,6.365-10.465,11.138L2.001,374.717 |
|
1190 | 259.316c-4.631,2.41-8.346,6.365-10.465,11.138L2.001,374.717 | |
1191 | c-3.191,7.188-2.537,15.412,1.75,22.005c4.285, |
|
1191 | c-3.191,7.188-2.537,15.412,1.75,22.005c4.285, | |
1192 | 6.592,11.537,10.526,19.4,10.526h362.861c7.863,0,15.117-3.936, |
|
1192 | 6.592,11.537,10.526,19.4,10.526h362.861c7.863,0,15.117-3.936, | |
1193 | 19.402-10.527 C409.699,390.129, |
|
1193 | 19.402-10.527 C409.699,390.129, | |
1194 | 410.355,381.902,407.164,374.717z" fill="{text_color}"/> |
|
1194 | 410.355,381.902,407.164,374.717z" fill="{text_color}"/> | |
1195 | </svg>""".format( |
|
1195 | </svg>""".format( | |
1196 | size=self.size, |
|
1196 | size=self.size, | |
1197 | background='#979797', # @grey4 |
|
1197 | background='#979797', # @grey4 | |
1198 | text_color=self.text_color, |
|
1198 | text_color=self.text_color, | |
1199 | font_family=font_family) |
|
1199 | font_family=font_family) | |
1200 |
|
1200 | |||
1201 | return { |
|
1201 | return { | |
1202 | "default_user": default_user |
|
1202 | "default_user": default_user | |
1203 | }[img_type] |
|
1203 | }[img_type] | |
1204 |
|
1204 | |||
1205 | def get_img_data(self, svg_type=None): |
|
1205 | def get_img_data(self, svg_type=None): | |
1206 | """ |
|
1206 | """ | |
1207 | generates the svg metadata for image |
|
1207 | generates the svg metadata for image | |
1208 | """ |
|
1208 | """ | |
1209 |
|
1209 | |||
1210 | font_family = ','.join([ |
|
1210 | font_family = ','.join([ | |
1211 | 'proximanovaregular', |
|
1211 | 'proximanovaregular', | |
1212 | 'Proxima Nova Regular', |
|
1212 | 'Proxima Nova Regular', | |
1213 | 'Proxima Nova', |
|
1213 | 'Proxima Nova', | |
1214 | 'Arial', |
|
1214 | 'Arial', | |
1215 | 'Lucida Grande', |
|
1215 | 'Lucida Grande', | |
1216 | 'sans-serif' |
|
1216 | 'sans-serif' | |
1217 | ]) |
|
1217 | ]) | |
1218 | if svg_type: |
|
1218 | if svg_type: | |
1219 | return self.get_img_data_by_type(font_family, svg_type) |
|
1219 | return self.get_img_data_by_type(font_family, svg_type) | |
1220 |
|
1220 | |||
1221 | initials = self.get_initials() |
|
1221 | initials = self.get_initials() | |
1222 | img_data = """ |
|
1222 | img_data = """ | |
1223 | <svg xmlns="http://www.w3.org/2000/svg" pointer-events="none" |
|
1223 | <svg xmlns="http://www.w3.org/2000/svg" pointer-events="none" | |
1224 | width="{size}" height="{size}" |
|
1224 | width="{size}" height="{size}" | |
1225 | style="width: 100%; height: 100%; background-color: {background}" |
|
1225 | style="width: 100%; height: 100%; background-color: {background}" | |
1226 | viewBox="0 0 {size} {size}"> |
|
1226 | viewBox="0 0 {size} {size}"> | |
1227 | <text text-anchor="middle" y="50%" x="50%" dy="0.35em" |
|
1227 | <text text-anchor="middle" y="50%" x="50%" dy="0.35em" | |
1228 | pointer-events="auto" fill="{text_color}" |
|
1228 | pointer-events="auto" fill="{text_color}" | |
1229 | font-family="{font_family}" |
|
1229 | font-family="{font_family}" | |
1230 | style="font-weight: 400; font-size: {f_size}px;">{text} |
|
1230 | style="font-weight: 400; font-size: {f_size}px;">{text} | |
1231 | </text> |
|
1231 | </text> | |
1232 | </svg>""".format( |
|
1232 | </svg>""".format( | |
1233 | size=self.size, |
|
1233 | size=self.size, | |
1234 | f_size=self.size/1.85, # scale the text inside the box nicely |
|
1234 | f_size=self.size/1.85, # scale the text inside the box nicely | |
1235 | background=self.background, |
|
1235 | background=self.background, | |
1236 | text_color=self.text_color, |
|
1236 | text_color=self.text_color, | |
1237 | text=initials.upper(), |
|
1237 | text=initials.upper(), | |
1238 | font_family=font_family) |
|
1238 | font_family=font_family) | |
1239 |
|
1239 | |||
1240 | return img_data |
|
1240 | return img_data | |
1241 |
|
1241 | |||
1242 | def generate_svg(self, svg_type=None): |
|
1242 | def generate_svg(self, svg_type=None): | |
1243 | img_data = self.get_img_data(svg_type) |
|
1243 | img_data = self.get_img_data(svg_type) | |
1244 | return "data:image/svg+xml;base64,%s" % img_data.encode('base64') |
|
1244 | return "data:image/svg+xml;base64,%s" % img_data.encode('base64') | |
1245 |
|
1245 | |||
1246 |
|
1246 | |||
1247 | def initials_gravatar(email_address, first_name, last_name, size=30): |
|
1247 | def initials_gravatar(email_address, first_name, last_name, size=30): | |
1248 | svg_type = None |
|
1248 | svg_type = None | |
1249 | if email_address == User.DEFAULT_USER_EMAIL: |
|
1249 | if email_address == User.DEFAULT_USER_EMAIL: | |
1250 | svg_type = 'default_user' |
|
1250 | svg_type = 'default_user' | |
1251 | klass = InitialsGravatar(email_address, first_name, last_name, size) |
|
1251 | klass = InitialsGravatar(email_address, first_name, last_name, size) | |
1252 | return klass.generate_svg(svg_type=svg_type) |
|
1252 | return klass.generate_svg(svg_type=svg_type) | |
1253 |
|
1253 | |||
1254 |
|
1254 | |||
1255 | def gravatar_url(email_address, size=30, request=None): |
|
1255 | def gravatar_url(email_address, size=30, request=None): | |
1256 | request = get_current_request() |
|
1256 | request = get_current_request() | |
1257 | if request and hasattr(request, 'call_context'): |
|
1257 | if request and hasattr(request, 'call_context'): | |
1258 | _use_gravatar = request.call_context.visual.use_gravatar |
|
1258 | _use_gravatar = request.call_context.visual.use_gravatar | |
1259 | _gravatar_url = request.call_context.visual.gravatar_url |
|
1259 | _gravatar_url = request.call_context.visual.gravatar_url | |
1260 | else: |
|
1260 | else: | |
1261 | # doh, we need to re-import those to mock it later |
|
1261 | # doh, we need to re-import those to mock it later | |
1262 | from pylons import tmpl_context as c |
|
1262 | from pylons import tmpl_context as c | |
1263 |
|
1263 | |||
1264 | _use_gravatar = c.visual.use_gravatar |
|
1264 | _use_gravatar = c.visual.use_gravatar | |
1265 | _gravatar_url = c.visual.gravatar_url |
|
1265 | _gravatar_url = c.visual.gravatar_url | |
1266 |
|
1266 | |||
1267 | _gravatar_url = _gravatar_url or User.DEFAULT_GRAVATAR_URL |
|
1267 | _gravatar_url = _gravatar_url or User.DEFAULT_GRAVATAR_URL | |
1268 |
|
1268 | |||
1269 | email_address = email_address or User.DEFAULT_USER_EMAIL |
|
1269 | email_address = email_address or User.DEFAULT_USER_EMAIL | |
1270 | if isinstance(email_address, unicode): |
|
1270 | if isinstance(email_address, unicode): | |
1271 | # hashlib crashes on unicode items |
|
1271 | # hashlib crashes on unicode items | |
1272 | email_address = safe_str(email_address) |
|
1272 | email_address = safe_str(email_address) | |
1273 |
|
1273 | |||
1274 | # empty email or default user |
|
1274 | # empty email or default user | |
1275 | if not email_address or email_address == User.DEFAULT_USER_EMAIL: |
|
1275 | if not email_address or email_address == User.DEFAULT_USER_EMAIL: | |
1276 | return initials_gravatar(User.DEFAULT_USER_EMAIL, '', '', size=size) |
|
1276 | return initials_gravatar(User.DEFAULT_USER_EMAIL, '', '', size=size) | |
1277 |
|
1277 | |||
1278 | if _use_gravatar: |
|
1278 | if _use_gravatar: | |
1279 | # TODO: Disuse pyramid thread locals. Think about another solution to |
|
1279 | # TODO: Disuse pyramid thread locals. Think about another solution to | |
1280 | # get the host and schema here. |
|
1280 | # get the host and schema here. | |
1281 | request = get_current_request() |
|
1281 | request = get_current_request() | |
1282 | tmpl = safe_str(_gravatar_url) |
|
1282 | tmpl = safe_str(_gravatar_url) | |
1283 | tmpl = tmpl.replace('{email}', email_address)\ |
|
1283 | tmpl = tmpl.replace('{email}', email_address)\ | |
1284 | .replace('{md5email}', md5_safe(email_address.lower())) \ |
|
1284 | .replace('{md5email}', md5_safe(email_address.lower())) \ | |
1285 | .replace('{netloc}', request.host)\ |
|
1285 | .replace('{netloc}', request.host)\ | |
1286 | .replace('{scheme}', request.scheme)\ |
|
1286 | .replace('{scheme}', request.scheme)\ | |
1287 | .replace('{size}', safe_str(size)) |
|
1287 | .replace('{size}', safe_str(size)) | |
1288 | return tmpl |
|
1288 | return tmpl | |
1289 | else: |
|
1289 | else: | |
1290 | return initials_gravatar(email_address, '', '', size=size) |
|
1290 | return initials_gravatar(email_address, '', '', size=size) | |
1291 |
|
1291 | |||
1292 |
|
1292 | |||
1293 | class Page(_Page): |
|
1293 | class Page(_Page): | |
1294 | """ |
|
1294 | """ | |
1295 | Custom pager to match rendering style with paginator |
|
1295 | Custom pager to match rendering style with paginator | |
1296 | """ |
|
1296 | """ | |
1297 |
|
1297 | |||
1298 | def _get_pos(self, cur_page, max_page, items): |
|
1298 | def _get_pos(self, cur_page, max_page, items): | |
1299 | edge = (items / 2) + 1 |
|
1299 | edge = (items / 2) + 1 | |
1300 | if (cur_page <= edge): |
|
1300 | if (cur_page <= edge): | |
1301 | radius = max(items / 2, items - cur_page) |
|
1301 | radius = max(items / 2, items - cur_page) | |
1302 | elif (max_page - cur_page) < edge: |
|
1302 | elif (max_page - cur_page) < edge: | |
1303 | radius = (items - 1) - (max_page - cur_page) |
|
1303 | radius = (items - 1) - (max_page - cur_page) | |
1304 | else: |
|
1304 | else: | |
1305 | radius = items / 2 |
|
1305 | radius = items / 2 | |
1306 |
|
1306 | |||
1307 | left = max(1, (cur_page - (radius))) |
|
1307 | left = max(1, (cur_page - (radius))) | |
1308 | right = min(max_page, cur_page + (radius)) |
|
1308 | right = min(max_page, cur_page + (radius)) | |
1309 | return left, cur_page, right |
|
1309 | return left, cur_page, right | |
1310 |
|
1310 | |||
1311 | def _range(self, regexp_match): |
|
1311 | def _range(self, regexp_match): | |
1312 | """ |
|
1312 | """ | |
1313 | Return range of linked pages (e.g. '1 2 [3] 4 5 6 7 8'). |
|
1313 | Return range of linked pages (e.g. '1 2 [3] 4 5 6 7 8'). | |
1314 |
|
1314 | |||
1315 | Arguments: |
|
1315 | Arguments: | |
1316 |
|
1316 | |||
1317 | regexp_match |
|
1317 | regexp_match | |
1318 | A "re" (regular expressions) match object containing the |
|
1318 | A "re" (regular expressions) match object containing the | |
1319 | radius of linked pages around the current page in |
|
1319 | radius of linked pages around the current page in | |
1320 | regexp_match.group(1) as a string |
|
1320 | regexp_match.group(1) as a string | |
1321 |
|
1321 | |||
1322 | This function is supposed to be called as a callable in |
|
1322 | This function is supposed to be called as a callable in | |
1323 | re.sub. |
|
1323 | re.sub. | |
1324 |
|
1324 | |||
1325 | """ |
|
1325 | """ | |
1326 | radius = int(regexp_match.group(1)) |
|
1326 | radius = int(regexp_match.group(1)) | |
1327 |
|
1327 | |||
1328 | # Compute the first and last page number within the radius |
|
1328 | # Compute the first and last page number within the radius | |
1329 | # e.g. '1 .. 5 6 [7] 8 9 .. 12' |
|
1329 | # e.g. '1 .. 5 6 [7] 8 9 .. 12' | |
1330 | # -> leftmost_page = 5 |
|
1330 | # -> leftmost_page = 5 | |
1331 | # -> rightmost_page = 9 |
|
1331 | # -> rightmost_page = 9 | |
1332 | leftmost_page, _cur, rightmost_page = self._get_pos(self.page, |
|
1332 | leftmost_page, _cur, rightmost_page = self._get_pos(self.page, | |
1333 | self.last_page, |
|
1333 | self.last_page, | |
1334 | (radius * 2) + 1) |
|
1334 | (radius * 2) + 1) | |
1335 | nav_items = [] |
|
1335 | nav_items = [] | |
1336 |
|
1336 | |||
1337 | # Create a link to the first page (unless we are on the first page |
|
1337 | # Create a link to the first page (unless we are on the first page | |
1338 | # or there would be no need to insert '..' spacers) |
|
1338 | # or there would be no need to insert '..' spacers) | |
1339 | if self.page != self.first_page and self.first_page < leftmost_page: |
|
1339 | if self.page != self.first_page and self.first_page < leftmost_page: | |
1340 | nav_items.append(self._pagerlink(self.first_page, self.first_page)) |
|
1340 | nav_items.append(self._pagerlink(self.first_page, self.first_page)) | |
1341 |
|
1341 | |||
1342 | # Insert dots if there are pages between the first page |
|
1342 | # Insert dots if there are pages between the first page | |
1343 | # and the currently displayed page range |
|
1343 | # and the currently displayed page range | |
1344 | if leftmost_page - self.first_page > 1: |
|
1344 | if leftmost_page - self.first_page > 1: | |
1345 | # Wrap in a SPAN tag if nolink_attr is set |
|
1345 | # Wrap in a SPAN tag if nolink_attr is set | |
1346 | text = '..' |
|
1346 | text = '..' | |
1347 | if self.dotdot_attr: |
|
1347 | if self.dotdot_attr: | |
1348 | text = HTML.span(c=text, **self.dotdot_attr) |
|
1348 | text = HTML.span(c=text, **self.dotdot_attr) | |
1349 | nav_items.append(text) |
|
1349 | nav_items.append(text) | |
1350 |
|
1350 | |||
1351 | for thispage in xrange(leftmost_page, rightmost_page + 1): |
|
1351 | for thispage in xrange(leftmost_page, rightmost_page + 1): | |
1352 | # Hilight the current page number and do not use a link |
|
1352 | # Hilight the current page number and do not use a link | |
1353 | if thispage == self.page: |
|
1353 | if thispage == self.page: | |
1354 | text = '%s' % (thispage,) |
|
1354 | text = '%s' % (thispage,) | |
1355 | # Wrap in a SPAN tag if nolink_attr is set |
|
1355 | # Wrap in a SPAN tag if nolink_attr is set | |
1356 | if self.curpage_attr: |
|
1356 | if self.curpage_attr: | |
1357 | text = HTML.span(c=text, **self.curpage_attr) |
|
1357 | text = HTML.span(c=text, **self.curpage_attr) | |
1358 | nav_items.append(text) |
|
1358 | nav_items.append(text) | |
1359 | # Otherwise create just a link to that page |
|
1359 | # Otherwise create just a link to that page | |
1360 | else: |
|
1360 | else: | |
1361 | text = '%s' % (thispage,) |
|
1361 | text = '%s' % (thispage,) | |
1362 | nav_items.append(self._pagerlink(thispage, text)) |
|
1362 | nav_items.append(self._pagerlink(thispage, text)) | |
1363 |
|
1363 | |||
1364 | # Insert dots if there are pages between the displayed |
|
1364 | # Insert dots if there are pages between the displayed | |
1365 | # page numbers and the end of the page range |
|
1365 | # page numbers and the end of the page range | |
1366 | if self.last_page - rightmost_page > 1: |
|
1366 | if self.last_page - rightmost_page > 1: | |
1367 | text = '..' |
|
1367 | text = '..' | |
1368 | # Wrap in a SPAN tag if nolink_attr is set |
|
1368 | # Wrap in a SPAN tag if nolink_attr is set | |
1369 | if self.dotdot_attr: |
|
1369 | if self.dotdot_attr: | |
1370 | text = HTML.span(c=text, **self.dotdot_attr) |
|
1370 | text = HTML.span(c=text, **self.dotdot_attr) | |
1371 | nav_items.append(text) |
|
1371 | nav_items.append(text) | |
1372 |
|
1372 | |||
1373 | # Create a link to the very last page (unless we are on the last |
|
1373 | # Create a link to the very last page (unless we are on the last | |
1374 | # page or there would be no need to insert '..' spacers) |
|
1374 | # page or there would be no need to insert '..' spacers) | |
1375 | if self.page != self.last_page and rightmost_page < self.last_page: |
|
1375 | if self.page != self.last_page and rightmost_page < self.last_page: | |
1376 | nav_items.append(self._pagerlink(self.last_page, self.last_page)) |
|
1376 | nav_items.append(self._pagerlink(self.last_page, self.last_page)) | |
1377 |
|
1377 | |||
1378 | ## prerender links |
|
1378 | ## prerender links | |
1379 | #_page_link = url.current() |
|
1379 | #_page_link = url.current() | |
1380 | #nav_items.append(literal('<link rel="prerender" href="%s?page=%s">' % (_page_link, str(int(self.page)+1)))) |
|
1380 | #nav_items.append(literal('<link rel="prerender" href="%s?page=%s">' % (_page_link, str(int(self.page)+1)))) | |
1381 | #nav_items.append(literal('<link rel="prefetch" href="%s?page=%s">' % (_page_link, str(int(self.page)+1)))) |
|
1381 | #nav_items.append(literal('<link rel="prefetch" href="%s?page=%s">' % (_page_link, str(int(self.page)+1)))) | |
1382 | return self.separator.join(nav_items) |
|
1382 | return self.separator.join(nav_items) | |
1383 |
|
1383 | |||
1384 | def pager(self, format='~2~', page_param='page', partial_param='partial', |
|
1384 | def pager(self, format='~2~', page_param='page', partial_param='partial', | |
1385 | show_if_single_page=False, separator=' ', onclick=None, |
|
1385 | show_if_single_page=False, separator=' ', onclick=None, | |
1386 | symbol_first='<<', symbol_last='>>', |
|
1386 | symbol_first='<<', symbol_last='>>', | |
1387 | symbol_previous='<', symbol_next='>', |
|
1387 | symbol_previous='<', symbol_next='>', | |
1388 | link_attr={'class': 'pager_link', 'rel': 'prerender'}, |
|
1388 | link_attr={'class': 'pager_link', 'rel': 'prerender'}, | |
1389 | curpage_attr={'class': 'pager_curpage'}, |
|
1389 | curpage_attr={'class': 'pager_curpage'}, | |
1390 | dotdot_attr={'class': 'pager_dotdot'}, **kwargs): |
|
1390 | dotdot_attr={'class': 'pager_dotdot'}, **kwargs): | |
1391 |
|
1391 | |||
1392 | self.curpage_attr = curpage_attr |
|
1392 | self.curpage_attr = curpage_attr | |
1393 | self.separator = separator |
|
1393 | self.separator = separator | |
1394 | self.pager_kwargs = kwargs |
|
1394 | self.pager_kwargs = kwargs | |
1395 | self.page_param = page_param |
|
1395 | self.page_param = page_param | |
1396 | self.partial_param = partial_param |
|
1396 | self.partial_param = partial_param | |
1397 | self.onclick = onclick |
|
1397 | self.onclick = onclick | |
1398 | self.link_attr = link_attr |
|
1398 | self.link_attr = link_attr | |
1399 | self.dotdot_attr = dotdot_attr |
|
1399 | self.dotdot_attr = dotdot_attr | |
1400 |
|
1400 | |||
1401 | # Don't show navigator if there is no more than one page |
|
1401 | # Don't show navigator if there is no more than one page | |
1402 | if self.page_count == 0 or (self.page_count == 1 and not show_if_single_page): |
|
1402 | if self.page_count == 0 or (self.page_count == 1 and not show_if_single_page): | |
1403 | return '' |
|
1403 | return '' | |
1404 |
|
1404 | |||
1405 | from string import Template |
|
1405 | from string import Template | |
1406 | # Replace ~...~ in token format by range of pages |
|
1406 | # Replace ~...~ in token format by range of pages | |
1407 | result = re.sub(r'~(\d+)~', self._range, format) |
|
1407 | result = re.sub(r'~(\d+)~', self._range, format) | |
1408 |
|
1408 | |||
1409 | # Interpolate '%' variables |
|
1409 | # Interpolate '%' variables | |
1410 | result = Template(result).safe_substitute({ |
|
1410 | result = Template(result).safe_substitute({ | |
1411 | 'first_page': self.first_page, |
|
1411 | 'first_page': self.first_page, | |
1412 | 'last_page': self.last_page, |
|
1412 | 'last_page': self.last_page, | |
1413 | 'page': self.page, |
|
1413 | 'page': self.page, | |
1414 | 'page_count': self.page_count, |
|
1414 | 'page_count': self.page_count, | |
1415 | 'items_per_page': self.items_per_page, |
|
1415 | 'items_per_page': self.items_per_page, | |
1416 | 'first_item': self.first_item, |
|
1416 | 'first_item': self.first_item, | |
1417 | 'last_item': self.last_item, |
|
1417 | 'last_item': self.last_item, | |
1418 | 'item_count': self.item_count, |
|
1418 | 'item_count': self.item_count, | |
1419 | 'link_first': self.page > self.first_page and \ |
|
1419 | 'link_first': self.page > self.first_page and \ | |
1420 | self._pagerlink(self.first_page, symbol_first) or '', |
|
1420 | self._pagerlink(self.first_page, symbol_first) or '', | |
1421 | 'link_last': self.page < self.last_page and \ |
|
1421 | 'link_last': self.page < self.last_page and \ | |
1422 | self._pagerlink(self.last_page, symbol_last) or '', |
|
1422 | self._pagerlink(self.last_page, symbol_last) or '', | |
1423 | 'link_previous': self.previous_page and \ |
|
1423 | 'link_previous': self.previous_page and \ | |
1424 | self._pagerlink(self.previous_page, symbol_previous) \ |
|
1424 | self._pagerlink(self.previous_page, symbol_previous) \ | |
1425 | or HTML.span(symbol_previous, class_="pg-previous disabled"), |
|
1425 | or HTML.span(symbol_previous, class_="pg-previous disabled"), | |
1426 | 'link_next': self.next_page and \ |
|
1426 | 'link_next': self.next_page and \ | |
1427 | self._pagerlink(self.next_page, symbol_next) \ |
|
1427 | self._pagerlink(self.next_page, symbol_next) \ | |
1428 | or HTML.span(symbol_next, class_="pg-next disabled") |
|
1428 | or HTML.span(symbol_next, class_="pg-next disabled") | |
1429 | }) |
|
1429 | }) | |
1430 |
|
1430 | |||
1431 | return literal(result) |
|
1431 | return literal(result) | |
1432 |
|
1432 | |||
1433 |
|
1433 | |||
1434 | #============================================================================== |
|
1434 | #============================================================================== | |
1435 | # REPO PAGER, PAGER FOR REPOSITORY |
|
1435 | # REPO PAGER, PAGER FOR REPOSITORY | |
1436 | #============================================================================== |
|
1436 | #============================================================================== | |
1437 | class RepoPage(Page): |
|
1437 | class RepoPage(Page): | |
1438 |
|
1438 | |||
1439 | def __init__(self, collection, page=1, items_per_page=20, |
|
1439 | def __init__(self, collection, page=1, items_per_page=20, | |
1440 | item_count=None, url=None, **kwargs): |
|
1440 | item_count=None, url=None, **kwargs): | |
1441 |
|
1441 | |||
1442 | """Create a "RepoPage" instance. special pager for paging |
|
1442 | """Create a "RepoPage" instance. special pager for paging | |
1443 | repository |
|
1443 | repository | |
1444 | """ |
|
1444 | """ | |
1445 | self._url_generator = url |
|
1445 | self._url_generator = url | |
1446 |
|
1446 | |||
1447 | # Safe the kwargs class-wide so they can be used in the pager() method |
|
1447 | # Safe the kwargs class-wide so they can be used in the pager() method | |
1448 | self.kwargs = kwargs |
|
1448 | self.kwargs = kwargs | |
1449 |
|
1449 | |||
1450 | # Save a reference to the collection |
|
1450 | # Save a reference to the collection | |
1451 | self.original_collection = collection |
|
1451 | self.original_collection = collection | |
1452 |
|
1452 | |||
1453 | self.collection = collection |
|
1453 | self.collection = collection | |
1454 |
|
1454 | |||
1455 | # The self.page is the number of the current page. |
|
1455 | # The self.page is the number of the current page. | |
1456 | # The first page has the number 1! |
|
1456 | # The first page has the number 1! | |
1457 | try: |
|
1457 | try: | |
1458 | self.page = int(page) # make it int() if we get it as a string |
|
1458 | self.page = int(page) # make it int() if we get it as a string | |
1459 | except (ValueError, TypeError): |
|
1459 | except (ValueError, TypeError): | |
1460 | self.page = 1 |
|
1460 | self.page = 1 | |
1461 |
|
1461 | |||
1462 | self.items_per_page = items_per_page |
|
1462 | self.items_per_page = items_per_page | |
1463 |
|
1463 | |||
1464 | # Unless the user tells us how many items the collections has |
|
1464 | # Unless the user tells us how many items the collections has | |
1465 | # we calculate that ourselves. |
|
1465 | # we calculate that ourselves. | |
1466 | if item_count is not None: |
|
1466 | if item_count is not None: | |
1467 | self.item_count = item_count |
|
1467 | self.item_count = item_count | |
1468 | else: |
|
1468 | else: | |
1469 | self.item_count = len(self.collection) |
|
1469 | self.item_count = len(self.collection) | |
1470 |
|
1470 | |||
1471 | # Compute the number of the first and last available page |
|
1471 | # Compute the number of the first and last available page | |
1472 | if self.item_count > 0: |
|
1472 | if self.item_count > 0: | |
1473 | self.first_page = 1 |
|
1473 | self.first_page = 1 | |
1474 | self.page_count = int(math.ceil(float(self.item_count) / |
|
1474 | self.page_count = int(math.ceil(float(self.item_count) / | |
1475 | self.items_per_page)) |
|
1475 | self.items_per_page)) | |
1476 | self.last_page = self.first_page + self.page_count - 1 |
|
1476 | self.last_page = self.first_page + self.page_count - 1 | |
1477 |
|
1477 | |||
1478 | # Make sure that the requested page number is the range of |
|
1478 | # Make sure that the requested page number is the range of | |
1479 | # valid pages |
|
1479 | # valid pages | |
1480 | if self.page > self.last_page: |
|
1480 | if self.page > self.last_page: | |
1481 | self.page = self.last_page |
|
1481 | self.page = self.last_page | |
1482 | elif self.page < self.first_page: |
|
1482 | elif self.page < self.first_page: | |
1483 | self.page = self.first_page |
|
1483 | self.page = self.first_page | |
1484 |
|
1484 | |||
1485 | # Note: the number of items on this page can be less than |
|
1485 | # Note: the number of items on this page can be less than | |
1486 | # items_per_page if the last page is not full |
|
1486 | # items_per_page if the last page is not full | |
1487 | self.first_item = max(0, (self.item_count) - (self.page * |
|
1487 | self.first_item = max(0, (self.item_count) - (self.page * | |
1488 | items_per_page)) |
|
1488 | items_per_page)) | |
1489 | self.last_item = ((self.item_count - 1) - items_per_page * |
|
1489 | self.last_item = ((self.item_count - 1) - items_per_page * | |
1490 | (self.page - 1)) |
|
1490 | (self.page - 1)) | |
1491 |
|
1491 | |||
1492 | self.items = list(self.collection[self.first_item:self.last_item + 1]) |
|
1492 | self.items = list(self.collection[self.first_item:self.last_item + 1]) | |
1493 |
|
1493 | |||
1494 | # Links to previous and next page |
|
1494 | # Links to previous and next page | |
1495 | if self.page > self.first_page: |
|
1495 | if self.page > self.first_page: | |
1496 | self.previous_page = self.page - 1 |
|
1496 | self.previous_page = self.page - 1 | |
1497 | else: |
|
1497 | else: | |
1498 | self.previous_page = None |
|
1498 | self.previous_page = None | |
1499 |
|
1499 | |||
1500 | if self.page < self.last_page: |
|
1500 | if self.page < self.last_page: | |
1501 | self.next_page = self.page + 1 |
|
1501 | self.next_page = self.page + 1 | |
1502 | else: |
|
1502 | else: | |
1503 | self.next_page = None |
|
1503 | self.next_page = None | |
1504 |
|
1504 | |||
1505 | # No items available |
|
1505 | # No items available | |
1506 | else: |
|
1506 | else: | |
1507 | self.first_page = None |
|
1507 | self.first_page = None | |
1508 | self.page_count = 0 |
|
1508 | self.page_count = 0 | |
1509 | self.last_page = None |
|
1509 | self.last_page = None | |
1510 | self.first_item = None |
|
1510 | self.first_item = None | |
1511 | self.last_item = None |
|
1511 | self.last_item = None | |
1512 | self.previous_page = None |
|
1512 | self.previous_page = None | |
1513 | self.next_page = None |
|
1513 | self.next_page = None | |
1514 | self.items = [] |
|
1514 | self.items = [] | |
1515 |
|
1515 | |||
1516 | # This is a subclass of the 'list' type. Initialise the list now. |
|
1516 | # This is a subclass of the 'list' type. Initialise the list now. | |
1517 | list.__init__(self, reversed(self.items)) |
|
1517 | list.__init__(self, reversed(self.items)) | |
1518 |
|
1518 | |||
1519 |
|
1519 | |||
1520 | def breadcrumb_repo_link(repo): |
|
1520 | def breadcrumb_repo_link(repo): | |
1521 | """ |
|
1521 | """ | |
1522 | Makes a breadcrumbs path link to repo |
|
1522 | Makes a breadcrumbs path link to repo | |
1523 |
|
1523 | |||
1524 | ex:: |
|
1524 | ex:: | |
1525 | group >> subgroup >> repo |
|
1525 | group >> subgroup >> repo | |
1526 |
|
1526 | |||
1527 | :param repo: a Repository instance |
|
1527 | :param repo: a Repository instance | |
1528 | """ |
|
1528 | """ | |
1529 |
|
1529 | |||
1530 | path = [ |
|
1530 | path = [ | |
1531 | link_to(group.name, route_path('repo_group_home', repo_group_name=group.group_name)) |
|
1531 | link_to(group.name, route_path('repo_group_home', repo_group_name=group.group_name)) | |
1532 | for group in repo.groups_with_parents |
|
1532 | for group in repo.groups_with_parents | |
1533 | ] + [ |
|
1533 | ] + [ | |
1534 | link_to(repo.just_name, route_path('repo_summary', repo_name=repo.repo_name)) |
|
1534 | link_to(repo.just_name, route_path('repo_summary', repo_name=repo.repo_name)) | |
1535 | ] |
|
1535 | ] | |
1536 |
|
1536 | |||
1537 | return literal(' » '.join(path)) |
|
1537 | return literal(' » '.join(path)) | |
1538 |
|
1538 | |||
1539 |
|
1539 | |||
1540 | def format_byte_size_binary(file_size): |
|
1540 | def format_byte_size_binary(file_size): | |
1541 | """ |
|
1541 | """ | |
1542 | Formats file/folder sizes to standard. |
|
1542 | Formats file/folder sizes to standard. | |
1543 | """ |
|
1543 | """ | |
1544 | formatted_size = format_byte_size(file_size, binary=True) |
|
1544 | formatted_size = format_byte_size(file_size, binary=True) | |
1545 | return formatted_size |
|
1545 | return formatted_size | |
1546 |
|
1546 | |||
1547 |
|
1547 | |||
1548 | def urlify_text(text_, safe=True): |
|
1548 | def urlify_text(text_, safe=True): | |
1549 | """ |
|
1549 | """ | |
1550 | Extrac urls from text and make html links out of them |
|
1550 | Extrac urls from text and make html links out of them | |
1551 |
|
1551 | |||
1552 | :param text_: |
|
1552 | :param text_: | |
1553 | """ |
|
1553 | """ | |
1554 |
|
1554 | |||
1555 | url_pat = re.compile(r'''(http[s]?://(?:[a-zA-Z]|[0-9]|[$-_@#.&+]''' |
|
1555 | url_pat = re.compile(r'''(http[s]?://(?:[a-zA-Z]|[0-9]|[$-_@#.&+]''' | |
1556 | '''|[!*\(\),]|(?:%[0-9a-fA-F][0-9a-fA-F]))+)''') |
|
1556 | '''|[!*\(\),]|(?:%[0-9a-fA-F][0-9a-fA-F]))+)''') | |
1557 |
|
1557 | |||
1558 | def url_func(match_obj): |
|
1558 | def url_func(match_obj): | |
1559 | url_full = match_obj.groups()[0] |
|
1559 | url_full = match_obj.groups()[0] | |
1560 | return '<a href="%(url)s">%(url)s</a>' % ({'url': url_full}) |
|
1560 | return '<a href="%(url)s">%(url)s</a>' % ({'url': url_full}) | |
1561 | _newtext = url_pat.sub(url_func, text_) |
|
1561 | _newtext = url_pat.sub(url_func, text_) | |
1562 | if safe: |
|
1562 | if safe: | |
1563 | return literal(_newtext) |
|
1563 | return literal(_newtext) | |
1564 | return _newtext |
|
1564 | return _newtext | |
1565 |
|
1565 | |||
1566 |
|
1566 | |||
1567 | def urlify_commits(text_, repository): |
|
1567 | def urlify_commits(text_, repository): | |
1568 | """ |
|
1568 | """ | |
1569 | Extract commit ids from text and make link from them |
|
1569 | Extract commit ids from text and make link from them | |
1570 |
|
1570 | |||
1571 | :param text_: |
|
1571 | :param text_: | |
1572 | :param repository: repo name to build the URL with |
|
1572 | :param repository: repo name to build the URL with | |
1573 | """ |
|
1573 | """ | |
1574 | from pylons import url # doh, we need to re-import url to mock it later |
|
1574 | from pylons import url # doh, we need to re-import url to mock it later | |
1575 | URL_PAT = re.compile(r'(^|\s)([0-9a-fA-F]{12,40})($|\s)') |
|
1575 | URL_PAT = re.compile(r'(^|\s)([0-9a-fA-F]{12,40})($|\s)') | |
1576 |
|
1576 | |||
1577 | def url_func(match_obj): |
|
1577 | def url_func(match_obj): | |
1578 | commit_id = match_obj.groups()[1] |
|
1578 | commit_id = match_obj.groups()[1] | |
1579 | pref = match_obj.groups()[0] |
|
1579 | pref = match_obj.groups()[0] | |
1580 | suf = match_obj.groups()[2] |
|
1580 | suf = match_obj.groups()[2] | |
1581 |
|
1581 | |||
1582 | tmpl = ( |
|
1582 | tmpl = ( | |
1583 | '%(pref)s<a class="%(cls)s" href="%(url)s">' |
|
1583 | '%(pref)s<a class="%(cls)s" href="%(url)s">' | |
1584 | '%(commit_id)s</a>%(suf)s' |
|
1584 | '%(commit_id)s</a>%(suf)s' | |
1585 | ) |
|
1585 | ) | |
1586 | return tmpl % { |
|
1586 | return tmpl % { | |
1587 | 'pref': pref, |
|
1587 | 'pref': pref, | |
1588 | 'cls': 'revision-link', |
|
1588 | 'cls': 'revision-link', | |
1589 | 'url': url('changeset_home', repo_name=repository, |
|
1589 | 'url': url('changeset_home', repo_name=repository, | |
1590 | revision=commit_id, qualified=True), |
|
1590 | revision=commit_id, qualified=True), | |
1591 | 'commit_id': commit_id, |
|
1591 | 'commit_id': commit_id, | |
1592 | 'suf': suf |
|
1592 | 'suf': suf | |
1593 | } |
|
1593 | } | |
1594 |
|
1594 | |||
1595 | newtext = URL_PAT.sub(url_func, text_) |
|
1595 | newtext = URL_PAT.sub(url_func, text_) | |
1596 |
|
1596 | |||
1597 | return newtext |
|
1597 | return newtext | |
1598 |
|
1598 | |||
1599 |
|
1599 | |||
1600 | def _process_url_func(match_obj, repo_name, uid, entry, |
|
1600 | def _process_url_func(match_obj, repo_name, uid, entry, | |
1601 | return_raw_data=False, link_format='html'): |
|
1601 | return_raw_data=False, link_format='html'): | |
1602 | pref = '' |
|
1602 | pref = '' | |
1603 | if match_obj.group().startswith(' '): |
|
1603 | if match_obj.group().startswith(' '): | |
1604 | pref = ' ' |
|
1604 | pref = ' ' | |
1605 |
|
1605 | |||
1606 | issue_id = ''.join(match_obj.groups()) |
|
1606 | issue_id = ''.join(match_obj.groups()) | |
1607 |
|
1607 | |||
1608 | if link_format == 'html': |
|
1608 | if link_format == 'html': | |
1609 | tmpl = ( |
|
1609 | tmpl = ( | |
1610 | '%(pref)s<a class="%(cls)s" href="%(url)s">' |
|
1610 | '%(pref)s<a class="%(cls)s" href="%(url)s">' | |
1611 | '%(issue-prefix)s%(id-repr)s' |
|
1611 | '%(issue-prefix)s%(id-repr)s' | |
1612 | '</a>') |
|
1612 | '</a>') | |
1613 | elif link_format == 'rst': |
|
1613 | elif link_format == 'rst': | |
1614 | tmpl = '`%(issue-prefix)s%(id-repr)s <%(url)s>`_' |
|
1614 | tmpl = '`%(issue-prefix)s%(id-repr)s <%(url)s>`_' | |
1615 | elif link_format == 'markdown': |
|
1615 | elif link_format == 'markdown': | |
1616 | tmpl = '[%(issue-prefix)s%(id-repr)s](%(url)s)' |
|
1616 | tmpl = '[%(issue-prefix)s%(id-repr)s](%(url)s)' | |
1617 | else: |
|
1617 | else: | |
1618 | raise ValueError('Bad link_format:{}'.format(link_format)) |
|
1618 | raise ValueError('Bad link_format:{}'.format(link_format)) | |
1619 |
|
1619 | |||
1620 | (repo_name_cleaned, |
|
1620 | (repo_name_cleaned, | |
1621 | parent_group_name) = RepoGroupModel().\ |
|
1621 | parent_group_name) = RepoGroupModel().\ | |
1622 | _get_group_name_and_parent(repo_name) |
|
1622 | _get_group_name_and_parent(repo_name) | |
1623 |
|
1623 | |||
1624 | # variables replacement |
|
1624 | # variables replacement | |
1625 | named_vars = { |
|
1625 | named_vars = { | |
1626 | 'id': issue_id, |
|
1626 | 'id': issue_id, | |
1627 | 'repo': repo_name, |
|
1627 | 'repo': repo_name, | |
1628 | 'repo_name': repo_name_cleaned, |
|
1628 | 'repo_name': repo_name_cleaned, | |
1629 | 'group_name': parent_group_name |
|
1629 | 'group_name': parent_group_name | |
1630 | } |
|
1630 | } | |
1631 | # named regex variables |
|
1631 | # named regex variables | |
1632 | named_vars.update(match_obj.groupdict()) |
|
1632 | named_vars.update(match_obj.groupdict()) | |
1633 | _url = string.Template(entry['url']).safe_substitute(**named_vars) |
|
1633 | _url = string.Template(entry['url']).safe_substitute(**named_vars) | |
1634 |
|
1634 | |||
1635 | data = { |
|
1635 | data = { | |
1636 | 'pref': pref, |
|
1636 | 'pref': pref, | |
1637 | 'cls': 'issue-tracker-link', |
|
1637 | 'cls': 'issue-tracker-link', | |
1638 | 'url': _url, |
|
1638 | 'url': _url, | |
1639 | 'id-repr': issue_id, |
|
1639 | 'id-repr': issue_id, | |
1640 | 'issue-prefix': entry['pref'], |
|
1640 | 'issue-prefix': entry['pref'], | |
1641 | 'serv': entry['url'], |
|
1641 | 'serv': entry['url'], | |
1642 | } |
|
1642 | } | |
1643 | if return_raw_data: |
|
1643 | if return_raw_data: | |
1644 | return { |
|
1644 | return { | |
1645 | 'id': issue_id, |
|
1645 | 'id': issue_id, | |
1646 | 'url': _url |
|
1646 | 'url': _url | |
1647 | } |
|
1647 | } | |
1648 | return tmpl % data |
|
1648 | return tmpl % data | |
1649 |
|
1649 | |||
1650 |
|
1650 | |||
1651 | def process_patterns(text_string, repo_name, link_format='html'): |
|
1651 | def process_patterns(text_string, repo_name, link_format='html'): | |
1652 | allowed_formats = ['html', 'rst', 'markdown'] |
|
1652 | allowed_formats = ['html', 'rst', 'markdown'] | |
1653 | if link_format not in allowed_formats: |
|
1653 | if link_format not in allowed_formats: | |
1654 | raise ValueError('Link format can be only one of:{} got {}'.format( |
|
1654 | raise ValueError('Link format can be only one of:{} got {}'.format( | |
1655 | allowed_formats, link_format)) |
|
1655 | allowed_formats, link_format)) | |
1656 |
|
1656 | |||
1657 | repo = None |
|
1657 | repo = None | |
1658 | if repo_name: |
|
1658 | if repo_name: | |
1659 | # Retrieving repo_name to avoid invalid repo_name to explode on |
|
1659 | # Retrieving repo_name to avoid invalid repo_name to explode on | |
1660 | # IssueTrackerSettingsModel but still passing invalid name further down |
|
1660 | # IssueTrackerSettingsModel but still passing invalid name further down | |
1661 | repo = Repository.get_by_repo_name(repo_name, cache=True) |
|
1661 | repo = Repository.get_by_repo_name(repo_name, cache=True) | |
1662 |
|
1662 | |||
1663 | settings_model = IssueTrackerSettingsModel(repo=repo) |
|
1663 | settings_model = IssueTrackerSettingsModel(repo=repo) | |
1664 | active_entries = settings_model.get_settings(cache=True) |
|
1664 | active_entries = settings_model.get_settings(cache=True) | |
1665 |
|
1665 | |||
1666 | issues_data = [] |
|
1666 | issues_data = [] | |
1667 | newtext = text_string |
|
1667 | newtext = text_string | |
1668 |
|
1668 | |||
1669 | for uid, entry in active_entries.items(): |
|
1669 | for uid, entry in active_entries.items(): | |
1670 | log.debug('found issue tracker entry with uid %s' % (uid,)) |
|
1670 | log.debug('found issue tracker entry with uid %s' % (uid,)) | |
1671 |
|
1671 | |||
1672 | if not (entry['pat'] and entry['url']): |
|
1672 | if not (entry['pat'] and entry['url']): | |
1673 | log.debug('skipping due to missing data') |
|
1673 | log.debug('skipping due to missing data') | |
1674 | continue |
|
1674 | continue | |
1675 |
|
1675 | |||
1676 | log.debug('issue tracker entry: uid: `%s` PAT:%s URL:%s PREFIX:%s' |
|
1676 | log.debug('issue tracker entry: uid: `%s` PAT:%s URL:%s PREFIX:%s' | |
1677 | % (uid, entry['pat'], entry['url'], entry['pref'])) |
|
1677 | % (uid, entry['pat'], entry['url'], entry['pref'])) | |
1678 |
|
1678 | |||
1679 | try: |
|
1679 | try: | |
1680 | pattern = re.compile(r'%s' % entry['pat']) |
|
1680 | pattern = re.compile(r'%s' % entry['pat']) | |
1681 | except re.error: |
|
1681 | except re.error: | |
1682 | log.exception( |
|
1682 | log.exception( | |
1683 | 'issue tracker pattern: `%s` failed to compile', |
|
1683 | 'issue tracker pattern: `%s` failed to compile', | |
1684 | entry['pat']) |
|
1684 | entry['pat']) | |
1685 | continue |
|
1685 | continue | |
1686 |
|
1686 | |||
1687 | data_func = partial( |
|
1687 | data_func = partial( | |
1688 | _process_url_func, repo_name=repo_name, entry=entry, uid=uid, |
|
1688 | _process_url_func, repo_name=repo_name, entry=entry, uid=uid, | |
1689 | return_raw_data=True) |
|
1689 | return_raw_data=True) | |
1690 |
|
1690 | |||
1691 | for match_obj in pattern.finditer(text_string): |
|
1691 | for match_obj in pattern.finditer(text_string): | |
1692 | issues_data.append(data_func(match_obj)) |
|
1692 | issues_data.append(data_func(match_obj)) | |
1693 |
|
1693 | |||
1694 | url_func = partial( |
|
1694 | url_func = partial( | |
1695 | _process_url_func, repo_name=repo_name, entry=entry, uid=uid, |
|
1695 | _process_url_func, repo_name=repo_name, entry=entry, uid=uid, | |
1696 | link_format=link_format) |
|
1696 | link_format=link_format) | |
1697 |
|
1697 | |||
1698 | newtext = pattern.sub(url_func, newtext) |
|
1698 | newtext = pattern.sub(url_func, newtext) | |
1699 | log.debug('processed prefix:uid `%s`' % (uid,)) |
|
1699 | log.debug('processed prefix:uid `%s`' % (uid,)) | |
1700 |
|
1700 | |||
1701 | return newtext, issues_data |
|
1701 | return newtext, issues_data | |
1702 |
|
1702 | |||
1703 |
|
1703 | |||
1704 | def urlify_commit_message(commit_text, repository=None): |
|
1704 | def urlify_commit_message(commit_text, repository=None): | |
1705 | """ |
|
1705 | """ | |
1706 | Parses given text message and makes proper links. |
|
1706 | Parses given text message and makes proper links. | |
1707 | issues are linked to given issue-server, and rest is a commit link |
|
1707 | issues are linked to given issue-server, and rest is a commit link | |
1708 |
|
1708 | |||
1709 | :param commit_text: |
|
1709 | :param commit_text: | |
1710 | :param repository: |
|
1710 | :param repository: | |
1711 | """ |
|
1711 | """ | |
1712 | from pylons import url # doh, we need to re-import url to mock it later |
|
1712 | from pylons import url # doh, we need to re-import url to mock it later | |
1713 |
|
1713 | |||
1714 | def escaper(string): |
|
1714 | def escaper(string): | |
1715 | return string.replace('<', '<').replace('>', '>') |
|
1715 | return string.replace('<', '<').replace('>', '>') | |
1716 |
|
1716 | |||
1717 | newtext = escaper(commit_text) |
|
1717 | newtext = escaper(commit_text) | |
1718 |
|
1718 | |||
1719 | # extract http/https links and make them real urls |
|
1719 | # extract http/https links and make them real urls | |
1720 | newtext = urlify_text(newtext, safe=False) |
|
1720 | newtext = urlify_text(newtext, safe=False) | |
1721 |
|
1721 | |||
1722 | # urlify commits - extract commit ids and make link out of them, if we have |
|
1722 | # urlify commits - extract commit ids and make link out of them, if we have | |
1723 | # the scope of repository present. |
|
1723 | # the scope of repository present. | |
1724 | if repository: |
|
1724 | if repository: | |
1725 | newtext = urlify_commits(newtext, repository) |
|
1725 | newtext = urlify_commits(newtext, repository) | |
1726 |
|
1726 | |||
1727 | # process issue tracker patterns |
|
1727 | # process issue tracker patterns | |
1728 | newtext, issues = process_patterns(newtext, repository or '') |
|
1728 | newtext, issues = process_patterns(newtext, repository or '') | |
1729 |
|
1729 | |||
1730 | return literal(newtext) |
|
1730 | return literal(newtext) | |
1731 |
|
1731 | |||
1732 |
|
1732 | |||
1733 | def render_binary(repo_name, file_obj): |
|
1733 | def render_binary(repo_name, file_obj): | |
1734 | """ |
|
1734 | """ | |
1735 | Choose how to render a binary file |
|
1735 | Choose how to render a binary file | |
1736 | """ |
|
1736 | """ | |
1737 | filename = file_obj.name |
|
1737 | filename = file_obj.name | |
1738 |
|
1738 | |||
1739 | # images |
|
1739 | # images | |
1740 | for ext in ['*.png', '*.jpg', '*.ico', '*.gif']: |
|
1740 | for ext in ['*.png', '*.jpg', '*.ico', '*.gif']: | |
1741 | if fnmatch.fnmatch(filename, pat=ext): |
|
1741 | if fnmatch.fnmatch(filename, pat=ext): | |
1742 | alt = filename |
|
1742 | alt = filename | |
1743 | src = url('files_raw_home', repo_name=repo_name, |
|
1743 | src = url('files_raw_home', repo_name=repo_name, | |
1744 | revision=file_obj.commit.raw_id, f_path=file_obj.path) |
|
1744 | revision=file_obj.commit.raw_id, f_path=file_obj.path) | |
1745 | return literal('<img class="rendered-binary" alt="{}" src="{}">'.format(alt, src)) |
|
1745 | return literal('<img class="rendered-binary" alt="{}" src="{}">'.format(alt, src)) | |
1746 |
|
1746 | |||
1747 |
|
1747 | |||
1748 | def renderer_from_filename(filename, exclude=None): |
|
1748 | def renderer_from_filename(filename, exclude=None): | |
1749 | """ |
|
1749 | """ | |
1750 | choose a renderer based on filename, this works only for text based files |
|
1750 | choose a renderer based on filename, this works only for text based files | |
1751 | """ |
|
1751 | """ | |
1752 |
|
1752 | |||
1753 | # ipython |
|
1753 | # ipython | |
1754 | for ext in ['*.ipynb']: |
|
1754 | for ext in ['*.ipynb']: | |
1755 | if fnmatch.fnmatch(filename, pat=ext): |
|
1755 | if fnmatch.fnmatch(filename, pat=ext): | |
1756 | return 'jupyter' |
|
1756 | return 'jupyter' | |
1757 |
|
1757 | |||
1758 | is_markup = MarkupRenderer.renderer_from_filename(filename, exclude=exclude) |
|
1758 | is_markup = MarkupRenderer.renderer_from_filename(filename, exclude=exclude) | |
1759 | if is_markup: |
|
1759 | if is_markup: | |
1760 | return is_markup |
|
1760 | return is_markup | |
1761 | return None |
|
1761 | return None | |
1762 |
|
1762 | |||
1763 |
|
1763 | |||
1764 | def render(source, renderer='rst', mentions=False, relative_url=None, |
|
1764 | def render(source, renderer='rst', mentions=False, relative_url=None, | |
1765 | repo_name=None): |
|
1765 | repo_name=None): | |
1766 |
|
1766 | |||
1767 | def maybe_convert_relative_links(html_source): |
|
1767 | def maybe_convert_relative_links(html_source): | |
1768 | if relative_url: |
|
1768 | if relative_url: | |
1769 | return relative_links(html_source, relative_url) |
|
1769 | return relative_links(html_source, relative_url) | |
1770 | return html_source |
|
1770 | return html_source | |
1771 |
|
1771 | |||
1772 | if renderer == 'rst': |
|
1772 | if renderer == 'rst': | |
1773 | if repo_name: |
|
1773 | if repo_name: | |
1774 | # process patterns on comments if we pass in repo name |
|
1774 | # process patterns on comments if we pass in repo name | |
1775 | source, issues = process_patterns( |
|
1775 | source, issues = process_patterns( | |
1776 | source, repo_name, link_format='rst') |
|
1776 | source, repo_name, link_format='rst') | |
1777 |
|
1777 | |||
1778 | return literal( |
|
1778 | return literal( | |
1779 | '<div class="rst-block">%s</div>' % |
|
1779 | '<div class="rst-block">%s</div>' % | |
1780 | maybe_convert_relative_links( |
|
1780 | maybe_convert_relative_links( | |
1781 | MarkupRenderer.rst(source, mentions=mentions))) |
|
1781 | MarkupRenderer.rst(source, mentions=mentions))) | |
1782 | elif renderer == 'markdown': |
|
1782 | elif renderer == 'markdown': | |
1783 | if repo_name: |
|
1783 | if repo_name: | |
1784 | # process patterns on comments if we pass in repo name |
|
1784 | # process patterns on comments if we pass in repo name | |
1785 | source, issues = process_patterns( |
|
1785 | source, issues = process_patterns( | |
1786 | source, repo_name, link_format='markdown') |
|
1786 | source, repo_name, link_format='markdown') | |
1787 |
|
1787 | |||
1788 | return literal( |
|
1788 | return literal( | |
1789 | '<div class="markdown-block">%s</div>' % |
|
1789 | '<div class="markdown-block">%s</div>' % | |
1790 | maybe_convert_relative_links( |
|
1790 | maybe_convert_relative_links( | |
1791 | MarkupRenderer.markdown(source, flavored=True, |
|
1791 | MarkupRenderer.markdown(source, flavored=True, | |
1792 | mentions=mentions))) |
|
1792 | mentions=mentions))) | |
1793 | elif renderer == 'jupyter': |
|
1793 | elif renderer == 'jupyter': | |
1794 | return literal( |
|
1794 | return literal( | |
1795 | '<div class="ipynb">%s</div>' % |
|
1795 | '<div class="ipynb">%s</div>' % | |
1796 | maybe_convert_relative_links( |
|
1796 | maybe_convert_relative_links( | |
1797 | MarkupRenderer.jupyter(source))) |
|
1797 | MarkupRenderer.jupyter(source))) | |
1798 |
|
1798 | |||
1799 | # None means just show the file-source |
|
1799 | # None means just show the file-source | |
1800 | return None |
|
1800 | return None | |
1801 |
|
1801 | |||
1802 |
|
1802 | |||
1803 | def commit_status(repo, commit_id): |
|
1803 | def commit_status(repo, commit_id): | |
1804 | return ChangesetStatusModel().get_status(repo, commit_id) |
|
1804 | return ChangesetStatusModel().get_status(repo, commit_id) | |
1805 |
|
1805 | |||
1806 |
|
1806 | |||
1807 | def commit_status_lbl(commit_status): |
|
1807 | def commit_status_lbl(commit_status): | |
1808 | return dict(ChangesetStatus.STATUSES).get(commit_status) |
|
1808 | return dict(ChangesetStatus.STATUSES).get(commit_status) | |
1809 |
|
1809 | |||
1810 |
|
1810 | |||
1811 | def commit_time(repo_name, commit_id): |
|
1811 | def commit_time(repo_name, commit_id): | |
1812 | repo = Repository.get_by_repo_name(repo_name) |
|
1812 | repo = Repository.get_by_repo_name(repo_name) | |
1813 | commit = repo.get_commit(commit_id=commit_id) |
|
1813 | commit = repo.get_commit(commit_id=commit_id) | |
1814 | return commit.date |
|
1814 | return commit.date | |
1815 |
|
1815 | |||
1816 |
|
1816 | |||
1817 | def get_permission_name(key): |
|
1817 | def get_permission_name(key): | |
1818 | return dict(Permission.PERMS).get(key) |
|
1818 | return dict(Permission.PERMS).get(key) | |
1819 |
|
1819 | |||
1820 |
|
1820 | |||
1821 | def journal_filter_help(request): |
|
1821 | def journal_filter_help(request): | |
1822 | _ = request.translate |
|
1822 | _ = request.translate | |
1823 |
|
1823 | |||
1824 | return _( |
|
1824 | return _( | |
1825 | 'Example filter terms:\n' + |
|
1825 | 'Example filter terms:\n' + | |
1826 | ' repository:vcs\n' + |
|
1826 | ' repository:vcs\n' + | |
1827 | ' username:marcin\n' + |
|
1827 | ' username:marcin\n' + | |
1828 | ' username:(NOT marcin)\n' + |
|
1828 | ' username:(NOT marcin)\n' + | |
1829 | ' action:*push*\n' + |
|
1829 | ' action:*push*\n' + | |
1830 | ' ip:127.0.0.1\n' + |
|
1830 | ' ip:127.0.0.1\n' + | |
1831 | ' date:20120101\n' + |
|
1831 | ' date:20120101\n' + | |
1832 | ' date:[20120101100000 TO 20120102]\n' + |
|
1832 | ' date:[20120101100000 TO 20120102]\n' + | |
1833 | '\n' + |
|
1833 | '\n' + | |
1834 | 'Generate wildcards using \'*\' character:\n' + |
|
1834 | 'Generate wildcards using \'*\' character:\n' + | |
1835 | ' "repository:vcs*" - search everything starting with \'vcs\'\n' + |
|
1835 | ' "repository:vcs*" - search everything starting with \'vcs\'\n' + | |
1836 | ' "repository:*vcs*" - search for repository containing \'vcs\'\n' + |
|
1836 | ' "repository:*vcs*" - search for repository containing \'vcs\'\n' + | |
1837 | '\n' + |
|
1837 | '\n' + | |
1838 | 'Optional AND / OR operators in queries\n' + |
|
1838 | 'Optional AND / OR operators in queries\n' + | |
1839 | ' "repository:vcs OR repository:test"\n' + |
|
1839 | ' "repository:vcs OR repository:test"\n' + | |
1840 | ' "username:test AND repository:test*"\n' |
|
1840 | ' "username:test AND repository:test*"\n' | |
1841 | ) |
|
1841 | ) | |
1842 |
|
1842 | |||
1843 |
|
1843 | |||
1844 | def search_filter_help(searcher, request): |
|
1844 | def search_filter_help(searcher, request): | |
1845 | _ = request.translate |
|
1845 | _ = request.translate | |
1846 |
|
1846 | |||
1847 | terms = '' |
|
1847 | terms = '' | |
1848 | return _( |
|
1848 | return _( | |
1849 | 'Example filter terms for `{searcher}` search:\n' + |
|
1849 | 'Example filter terms for `{searcher}` search:\n' + | |
1850 | '{terms}\n' + |
|
1850 | '{terms}\n' + | |
1851 | 'Generate wildcards using \'*\' character:\n' + |
|
1851 | 'Generate wildcards using \'*\' character:\n' + | |
1852 | ' "repo_name:vcs*" - search everything starting with \'vcs\'\n' + |
|
1852 | ' "repo_name:vcs*" - search everything starting with \'vcs\'\n' + | |
1853 | ' "repo_name:*vcs*" - search for repository containing \'vcs\'\n' + |
|
1853 | ' "repo_name:*vcs*" - search for repository containing \'vcs\'\n' + | |
1854 | '\n' + |
|
1854 | '\n' + | |
1855 | 'Optional AND / OR operators in queries\n' + |
|
1855 | 'Optional AND / OR operators in queries\n' + | |
1856 | ' "repo_name:vcs OR repo_name:test"\n' + |
|
1856 | ' "repo_name:vcs OR repo_name:test"\n' + | |
1857 | ' "owner:test AND repo_name:test*"\n' + |
|
1857 | ' "owner:test AND repo_name:test*"\n' + | |
1858 | 'More: {search_doc}' |
|
1858 | 'More: {search_doc}' | |
1859 | ).format(searcher=searcher.name, |
|
1859 | ).format(searcher=searcher.name, | |
1860 | terms=terms, search_doc=searcher.query_lang_doc) |
|
1860 | terms=terms, search_doc=searcher.query_lang_doc) | |
1861 |
|
1861 | |||
1862 |
|
1862 | |||
1863 | def not_mapped_error(repo_name): |
|
1863 | def not_mapped_error(repo_name): | |
1864 | from rhodecode.translation import _ |
|
1864 | from rhodecode.translation import _ | |
1865 | flash(_('%s repository is not mapped to db perhaps' |
|
1865 | flash(_('%s repository is not mapped to db perhaps' | |
1866 | ' it was created or renamed from the filesystem' |
|
1866 | ' it was created or renamed from the filesystem' | |
1867 | ' please run the application again' |
|
1867 | ' please run the application again' | |
1868 | ' in order to rescan repositories') % repo_name, category='error') |
|
1868 | ' in order to rescan repositories') % repo_name, category='error') | |
1869 |
|
1869 | |||
1870 |
|
1870 | |||
1871 | def ip_range(ip_addr): |
|
1871 | def ip_range(ip_addr): | |
1872 | from rhodecode.model.db import UserIpMap |
|
1872 | from rhodecode.model.db import UserIpMap | |
1873 | s, e = UserIpMap._get_ip_range(ip_addr) |
|
1873 | s, e = UserIpMap._get_ip_range(ip_addr) | |
1874 | return '%s - %s' % (s, e) |
|
1874 | return '%s - %s' % (s, e) | |
1875 |
|
1875 | |||
1876 |
|
1876 | |||
1877 | def form(url, method='post', needs_csrf_token=True, **attrs): |
|
1877 | def form(url, method='post', needs_csrf_token=True, **attrs): | |
1878 | """Wrapper around webhelpers.tags.form to prevent CSRF attacks.""" |
|
1878 | """Wrapper around webhelpers.tags.form to prevent CSRF attacks.""" | |
1879 | if method.lower() != 'get' and needs_csrf_token: |
|
1879 | if method.lower() != 'get' and needs_csrf_token: | |
1880 | raise Exception( |
|
1880 | raise Exception( | |
1881 | 'Forms to POST/PUT/DELETE endpoints should have (in general) a ' + |
|
1881 | 'Forms to POST/PUT/DELETE endpoints should have (in general) a ' + | |
1882 | 'CSRF token. If the endpoint does not require such token you can ' + |
|
1882 | 'CSRF token. If the endpoint does not require such token you can ' + | |
1883 | 'explicitly set the parameter needs_csrf_token to false.') |
|
1883 | 'explicitly set the parameter needs_csrf_token to false.') | |
1884 |
|
1884 | |||
1885 | return wh_form(url, method=method, **attrs) |
|
1885 | return wh_form(url, method=method, **attrs) | |
1886 |
|
1886 | |||
1887 |
|
1887 | |||
1888 | def secure_form(url, method="POST", multipart=False, **attrs): |
|
1888 | def secure_form(url, method="POST", multipart=False, **attrs): | |
1889 | """Start a form tag that points the action to an url. This |
|
1889 | """Start a form tag that points the action to an url. This | |
1890 | form tag will also include the hidden field containing |
|
1890 | form tag will also include the hidden field containing | |
1891 | the auth token. |
|
1891 | the auth token. | |
1892 |
|
1892 | |||
1893 | The url options should be given either as a string, or as a |
|
1893 | The url options should be given either as a string, or as a | |
1894 | ``url()`` function. The method for the form defaults to POST. |
|
1894 | ``url()`` function. The method for the form defaults to POST. | |
1895 |
|
1895 | |||
1896 | Options: |
|
1896 | Options: | |
1897 |
|
1897 | |||
1898 | ``multipart`` |
|
1898 | ``multipart`` | |
1899 | If set to True, the enctype is set to "multipart/form-data". |
|
1899 | If set to True, the enctype is set to "multipart/form-data". | |
1900 | ``method`` |
|
1900 | ``method`` | |
1901 | The method to use when submitting the form, usually either |
|
1901 | The method to use when submitting the form, usually either | |
1902 | "GET" or "POST". If "PUT", "DELETE", or another verb is used, a |
|
1902 | "GET" or "POST". If "PUT", "DELETE", or another verb is used, a | |
1903 | hidden input with name _method is added to simulate the verb |
|
1903 | hidden input with name _method is added to simulate the verb | |
1904 | over POST. |
|
1904 | over POST. | |
1905 |
|
1905 | |||
1906 | """ |
|
1906 | """ | |
1907 | from webhelpers.pylonslib.secure_form import insecure_form |
|
1907 | from webhelpers.pylonslib.secure_form import insecure_form | |
1908 | form = insecure_form(url, method, multipart, **attrs) |
|
1908 | form = insecure_form(url, method, multipart, **attrs) | |
1909 | token = csrf_input() |
|
1909 | ||
|
1910 | session = None | |||
|
1911 | # TODO(marcink): after pyramid migration require request variable ALWAYS | |||
|
1912 | if 'request' in attrs: | |||
|
1913 | session = attrs['request'].session | |||
|
1914 | ||||
|
1915 | token = literal( | |||
|
1916 | '<input type="hidden" id="{}" name="{}" value="{}">'.format( | |||
|
1917 | csrf_token_key, csrf_token_key, get_csrf_token(session))) | |||
|
1918 | ||||
1910 | return literal("%s\n%s" % (form, token)) |
|
1919 | return literal("%s\n%s" % (form, token)) | |
1911 |
|
1920 | |||
1912 | def csrf_input(): |
|
|||
1913 | return literal( |
|
|||
1914 | '<input type="hidden" id="{}" name="{}" value="{}">'.format( |
|
|||
1915 | csrf_token_key, csrf_token_key, get_csrf_token())) |
|
|||
1916 |
|
1921 | |||
1917 | def dropdownmenu(name, selected, options, enable_filter=False, **attrs): |
|
1922 | def dropdownmenu(name, selected, options, enable_filter=False, **attrs): | |
1918 | select_html = select(name, selected, options, **attrs) |
|
1923 | select_html = select(name, selected, options, **attrs) | |
1919 | select2 = """ |
|
1924 | select2 = """ | |
1920 | <script> |
|
1925 | <script> | |
1921 | $(document).ready(function() { |
|
1926 | $(document).ready(function() { | |
1922 | $('#%s').select2({ |
|
1927 | $('#%s').select2({ | |
1923 | containerCssClass: 'drop-menu', |
|
1928 | containerCssClass: 'drop-menu', | |
1924 | dropdownCssClass: 'drop-menu-dropdown', |
|
1929 | dropdownCssClass: 'drop-menu-dropdown', | |
1925 | dropdownAutoWidth: true%s |
|
1930 | dropdownAutoWidth: true%s | |
1926 | }); |
|
1931 | }); | |
1927 | }); |
|
1932 | }); | |
1928 | </script> |
|
1933 | </script> | |
1929 | """ |
|
1934 | """ | |
1930 | filter_option = """, |
|
1935 | filter_option = """, | |
1931 | minimumResultsForSearch: -1 |
|
1936 | minimumResultsForSearch: -1 | |
1932 | """ |
|
1937 | """ | |
1933 | input_id = attrs.get('id') or name |
|
1938 | input_id = attrs.get('id') or name | |
1934 | filter_enabled = "" if enable_filter else filter_option |
|
1939 | filter_enabled = "" if enable_filter else filter_option | |
1935 | select_script = literal(select2 % (input_id, filter_enabled)) |
|
1940 | select_script = literal(select2 % (input_id, filter_enabled)) | |
1936 |
|
1941 | |||
1937 | return literal(select_html+select_script) |
|
1942 | return literal(select_html+select_script) | |
1938 |
|
1943 | |||
1939 |
|
1944 | |||
1940 | def get_visual_attr(tmpl_context_var, attr_name): |
|
1945 | def get_visual_attr(tmpl_context_var, attr_name): | |
1941 | """ |
|
1946 | """ | |
1942 | A safe way to get a variable from visual variable of template context |
|
1947 | A safe way to get a variable from visual variable of template context | |
1943 |
|
1948 | |||
1944 | :param tmpl_context_var: instance of tmpl_context, usually present as `c` |
|
1949 | :param tmpl_context_var: instance of tmpl_context, usually present as `c` | |
1945 | :param attr_name: name of the attribute we fetch from the c.visual |
|
1950 | :param attr_name: name of the attribute we fetch from the c.visual | |
1946 | """ |
|
1951 | """ | |
1947 | visual = getattr(tmpl_context_var, 'visual', None) |
|
1952 | visual = getattr(tmpl_context_var, 'visual', None) | |
1948 | if not visual: |
|
1953 | if not visual: | |
1949 | return |
|
1954 | return | |
1950 | else: |
|
1955 | else: | |
1951 | return getattr(visual, attr_name, None) |
|
1956 | return getattr(visual, attr_name, None) | |
1952 |
|
1957 | |||
1953 |
|
1958 | |||
1954 | def get_last_path_part(file_node): |
|
1959 | def get_last_path_part(file_node): | |
1955 | if not file_node.path: |
|
1960 | if not file_node.path: | |
1956 | return u'' |
|
1961 | return u'' | |
1957 |
|
1962 | |||
1958 | path = safe_unicode(file_node.path.split('/')[-1]) |
|
1963 | path = safe_unicode(file_node.path.split('/')[-1]) | |
1959 | return u'../' + path |
|
1964 | return u'../' + path | |
1960 |
|
1965 | |||
1961 |
|
1966 | |||
1962 | def route_url(*args, **kwargs): |
|
1967 | def route_url(*args, **kwargs): | |
1963 | """ |
|
1968 | """ | |
1964 | Wrapper around pyramids `route_url` (fully qualified url) function. |
|
1969 | Wrapper around pyramids `route_url` (fully qualified url) function. | |
1965 | It is used to generate URLs from within pylons views or templates. |
|
1970 | It is used to generate URLs from within pylons views or templates. | |
1966 | This will be removed when pyramid migration if finished. |
|
1971 | This will be removed when pyramid migration if finished. | |
1967 | """ |
|
1972 | """ | |
1968 | req = get_current_request() |
|
1973 | req = get_current_request() | |
1969 | return req.route_url(*args, **kwargs) |
|
1974 | return req.route_url(*args, **kwargs) | |
1970 |
|
1975 | |||
1971 |
|
1976 | |||
1972 | def route_path(*args, **kwargs): |
|
1977 | def route_path(*args, **kwargs): | |
1973 | """ |
|
1978 | """ | |
1974 | Wrapper around pyramids `route_path` function. It is used to generate |
|
1979 | Wrapper around pyramids `route_path` function. It is used to generate | |
1975 | URLs from within pylons views or templates. This will be removed when |
|
1980 | URLs from within pylons views or templates. This will be removed when | |
1976 | pyramid migration if finished. |
|
1981 | pyramid migration if finished. | |
1977 | """ |
|
1982 | """ | |
1978 | req = get_current_request() |
|
1983 | req = get_current_request() | |
1979 | return req.route_path(*args, **kwargs) |
|
1984 | return req.route_path(*args, **kwargs) | |
1980 |
|
1985 | |||
1981 |
|
1986 | |||
1982 | def route_path_or_none(*args, **kwargs): |
|
1987 | def route_path_or_none(*args, **kwargs): | |
1983 | try: |
|
1988 | try: | |
1984 | return route_path(*args, **kwargs) |
|
1989 | return route_path(*args, **kwargs) | |
1985 | except KeyError: |
|
1990 | except KeyError: | |
1986 | return None |
|
1991 | return None | |
1987 |
|
1992 | |||
1988 |
|
1993 | |||
1989 | def static_url(*args, **kwds): |
|
1994 | def static_url(*args, **kwds): | |
1990 | """ |
|
1995 | """ | |
1991 | Wrapper around pyramids `route_path` function. It is used to generate |
|
1996 | Wrapper around pyramids `route_path` function. It is used to generate | |
1992 | URLs from within pylons views or templates. This will be removed when |
|
1997 | URLs from within pylons views or templates. This will be removed when | |
1993 | pyramid migration if finished. |
|
1998 | pyramid migration if finished. | |
1994 | """ |
|
1999 | """ | |
1995 | req = get_current_request() |
|
2000 | req = get_current_request() | |
1996 | return req.static_url(*args, **kwds) |
|
2001 | return req.static_url(*args, **kwds) | |
1997 |
|
2002 | |||
1998 |
|
2003 | |||
1999 | def resource_path(*args, **kwds): |
|
2004 | def resource_path(*args, **kwds): | |
2000 | """ |
|
2005 | """ | |
2001 | Wrapper around pyramids `route_path` function. It is used to generate |
|
2006 | Wrapper around pyramids `route_path` function. It is used to generate | |
2002 | URLs from within pylons views or templates. This will be removed when |
|
2007 | URLs from within pylons views or templates. This will be removed when | |
2003 | pyramid migration if finished. |
|
2008 | pyramid migration if finished. | |
2004 | """ |
|
2009 | """ | |
2005 | req = get_current_request() |
|
2010 | req = get_current_request() | |
2006 | return req.resource_path(*args, **kwds) |
|
2011 | return req.resource_path(*args, **kwds) | |
2007 |
|
2012 | |||
2008 |
|
2013 | |||
2009 | def api_call_example(method, args): |
|
2014 | def api_call_example(method, args): | |
2010 | """ |
|
2015 | """ | |
2011 | Generates an API call example via CURL |
|
2016 | Generates an API call example via CURL | |
2012 | """ |
|
2017 | """ | |
2013 | args_json = json.dumps(OrderedDict([ |
|
2018 | args_json = json.dumps(OrderedDict([ | |
2014 | ('id', 1), |
|
2019 | ('id', 1), | |
2015 | ('auth_token', 'SECRET'), |
|
2020 | ('auth_token', 'SECRET'), | |
2016 | ('method', method), |
|
2021 | ('method', method), | |
2017 | ('args', args) |
|
2022 | ('args', args) | |
2018 | ])) |
|
2023 | ])) | |
2019 | return literal( |
|
2024 | return literal( | |
2020 | "curl {api_url} -X POST -H 'content-type:text/plain' --data-binary '{data}'" |
|
2025 | "curl {api_url} -X POST -H 'content-type:text/plain' --data-binary '{data}'" | |
2021 | "<br/><br/>SECRET can be found in <a href=\"{token_url}\">auth-tokens</a> page, " |
|
2026 | "<br/><br/>SECRET can be found in <a href=\"{token_url}\">auth-tokens</a> page, " | |
2022 | "and needs to be of `api calls` role." |
|
2027 | "and needs to be of `api calls` role." | |
2023 | .format( |
|
2028 | .format( | |
2024 | api_url=route_url('apiv2'), |
|
2029 | api_url=route_url('apiv2'), | |
2025 | token_url=route_url('my_account_auth_tokens'), |
|
2030 | token_url=route_url('my_account_auth_tokens'), | |
2026 | data=args_json)) |
|
2031 | data=args_json)) |
@@ -1,563 +1,565 b'' | |||||
1 | # -*- coding: utf-8 -*- |
|
1 | # -*- coding: utf-8 -*- | |
2 |
|
2 | |||
3 | # Copyright (C) 2010-2017 RhodeCode GmbH |
|
3 | # Copyright (C) 2010-2017 RhodeCode GmbH | |
4 | # |
|
4 | # | |
5 | # This program is free software: you can redistribute it and/or modify |
|
5 | # This program is free software: you can redistribute it and/or modify | |
6 | # it under the terms of the GNU Affero General Public License, version 3 |
|
6 | # it under the terms of the GNU Affero General Public License, version 3 | |
7 | # (only), as published by the Free Software Foundation. |
|
7 | # (only), as published by the Free Software Foundation. | |
8 | # |
|
8 | # | |
9 | # This program is distributed in the hope that it will be useful, |
|
9 | # This program is distributed in the hope that it will be useful, | |
10 | # but WITHOUT ANY WARRANTY; without even the implied warranty of |
|
10 | # but WITHOUT ANY WARRANTY; without even the implied warranty of | |
11 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
|
11 | # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | |
12 | # GNU General Public License for more details. |
|
12 | # GNU General Public License for more details. | |
13 | # |
|
13 | # | |
14 | # You should have received a copy of the GNU Affero General Public License |
|
14 | # You should have received a copy of the GNU Affero General Public License | |
15 | # along with this program. If not, see <http://www.gnu.org/licenses/>. |
|
15 | # along with this program. If not, see <http://www.gnu.org/licenses/>. | |
16 | # |
|
16 | # | |
17 | # This program is dual-licensed. If you wish to learn more about the |
|
17 | # This program is dual-licensed. If you wish to learn more about the | |
18 | # RhodeCode Enterprise Edition, including its added features, Support services, |
|
18 | # RhodeCode Enterprise Edition, including its added features, Support services, | |
19 | # and proprietary license terms, please see https://rhodecode.com/licenses/ |
|
19 | # and proprietary license terms, please see https://rhodecode.com/licenses/ | |
20 |
|
20 | |||
21 | """ |
|
21 | """ | |
22 | this is forms validation classes |
|
22 | this is forms validation classes | |
23 | http://formencode.org/module-formencode.validators.html |
|
23 | http://formencode.org/module-formencode.validators.html | |
24 | for list off all availible validators |
|
24 | for list off all availible validators | |
25 |
|
25 | |||
26 | we can create our own validators |
|
26 | we can create our own validators | |
27 |
|
27 | |||
28 | The table below outlines the options which can be used in a schema in addition to the validators themselves |
|
28 | The table below outlines the options which can be used in a schema in addition to the validators themselves | |
29 | pre_validators [] These validators will be applied before the schema |
|
29 | pre_validators [] These validators will be applied before the schema | |
30 | chained_validators [] These validators will be applied after the schema |
|
30 | chained_validators [] These validators will be applied after the schema | |
31 | allow_extra_fields False If True, then it is not an error when keys that aren't associated with a validator are present |
|
31 | allow_extra_fields False If True, then it is not an error when keys that aren't associated with a validator are present | |
32 | filter_extra_fields False If True, then keys that aren't associated with a validator are removed |
|
32 | filter_extra_fields False If True, then keys that aren't associated with a validator are removed | |
33 | if_key_missing NoDefault If this is given, then any keys that aren't available but are expected will be replaced with this value (and then validated). This does not override a present .if_missing attribute on validators. NoDefault is a special FormEncode class to mean that no default values has been specified and therefore missing keys shouldn't take a default value. |
|
33 | if_key_missing NoDefault If this is given, then any keys that aren't available but are expected will be replaced with this value (and then validated). This does not override a present .if_missing attribute on validators. NoDefault is a special FormEncode class to mean that no default values has been specified and therefore missing keys shouldn't take a default value. | |
34 | ignore_key_missing False If True, then missing keys will be missing in the result, if the validator doesn't have .if_missing on it already |
|
34 | ignore_key_missing False If True, then missing keys will be missing in the result, if the validator doesn't have .if_missing on it already | |
35 |
|
35 | |||
36 |
|
36 | |||
37 | <name> = formencode.validators.<name of validator> |
|
37 | <name> = formencode.validators.<name of validator> | |
38 | <name> must equal form name |
|
38 | <name> must equal form name | |
39 | list=[1,2,3,4,5] |
|
39 | list=[1,2,3,4,5] | |
40 | for SELECT use formencode.All(OneOf(list), Int()) |
|
40 | for SELECT use formencode.All(OneOf(list), Int()) | |
41 |
|
41 | |||
42 | """ |
|
42 | """ | |
43 |
|
43 | |||
44 | import deform |
|
44 | import deform | |
45 | import logging |
|
45 | import logging | |
46 | import formencode |
|
46 | import formencode | |
47 |
|
47 | |||
48 | from pkg_resources import resource_filename |
|
48 | from pkg_resources import resource_filename | |
49 | from formencode import All, Pipe |
|
49 | from formencode import All, Pipe | |
50 |
|
50 | |||
51 | from pylons.i18n.translation import _ |
|
51 | from pylons.i18n.translation import _ | |
|
52 | from pyramid.threadlocal import get_current_request | |||
52 |
|
53 | |||
53 | from rhodecode import BACKENDS |
|
54 | from rhodecode import BACKENDS | |
54 | from rhodecode.lib import helpers |
|
55 | from rhodecode.lib import helpers | |
55 | from rhodecode.model import validators as v |
|
56 | from rhodecode.model import validators as v | |
56 |
|
57 | |||
57 | log = logging.getLogger(__name__) |
|
58 | log = logging.getLogger(__name__) | |
58 |
|
59 | |||
59 |
|
60 | |||
60 | deform_templates = resource_filename('deform', 'templates') |
|
61 | deform_templates = resource_filename('deform', 'templates') | |
61 | rhodecode_templates = resource_filename('rhodecode', 'templates/forms') |
|
62 | rhodecode_templates = resource_filename('rhodecode', 'templates/forms') | |
62 | search_path = (rhodecode_templates, deform_templates) |
|
63 | search_path = (rhodecode_templates, deform_templates) | |
63 |
|
64 | |||
64 |
|
65 | |||
65 | class RhodecodeFormZPTRendererFactory(deform.ZPTRendererFactory): |
|
66 | class RhodecodeFormZPTRendererFactory(deform.ZPTRendererFactory): | |
66 | """ Subclass of ZPTRendererFactory to add rhodecode context variables """ |
|
67 | """ Subclass of ZPTRendererFactory to add rhodecode context variables """ | |
67 | def __call__(self, template_name, **kw): |
|
68 | def __call__(self, template_name, **kw): | |
68 | kw['h'] = helpers |
|
69 | kw['h'] = helpers | |
|
70 | kw['request'] = get_current_request() | |||
69 | return self.load(template_name)(**kw) |
|
71 | return self.load(template_name)(**kw) | |
70 |
|
72 | |||
71 |
|
73 | |||
72 | form_renderer = RhodecodeFormZPTRendererFactory(search_path) |
|
74 | form_renderer = RhodecodeFormZPTRendererFactory(search_path) | |
73 | deform.Form.set_default_renderer(form_renderer) |
|
75 | deform.Form.set_default_renderer(form_renderer) | |
74 |
|
76 | |||
75 |
|
77 | |||
76 | def LoginForm(): |
|
78 | def LoginForm(): | |
77 | class _LoginForm(formencode.Schema): |
|
79 | class _LoginForm(formencode.Schema): | |
78 | allow_extra_fields = True |
|
80 | allow_extra_fields = True | |
79 | filter_extra_fields = True |
|
81 | filter_extra_fields = True | |
80 | username = v.UnicodeString( |
|
82 | username = v.UnicodeString( | |
81 | strip=True, |
|
83 | strip=True, | |
82 | min=1, |
|
84 | min=1, | |
83 | not_empty=True, |
|
85 | not_empty=True, | |
84 | messages={ |
|
86 | messages={ | |
85 | 'empty': _(u'Please enter a login'), |
|
87 | 'empty': _(u'Please enter a login'), | |
86 | 'tooShort': _(u'Enter a value %(min)i characters long or more') |
|
88 | 'tooShort': _(u'Enter a value %(min)i characters long or more') | |
87 | } |
|
89 | } | |
88 | ) |
|
90 | ) | |
89 |
|
91 | |||
90 | password = v.UnicodeString( |
|
92 | password = v.UnicodeString( | |
91 | strip=False, |
|
93 | strip=False, | |
92 | min=3, |
|
94 | min=3, | |
93 | not_empty=True, |
|
95 | not_empty=True, | |
94 | messages={ |
|
96 | messages={ | |
95 | 'empty': _(u'Please enter a password'), |
|
97 | 'empty': _(u'Please enter a password'), | |
96 | 'tooShort': _(u'Enter %(min)i characters or more')} |
|
98 | 'tooShort': _(u'Enter %(min)i characters or more')} | |
97 | ) |
|
99 | ) | |
98 |
|
100 | |||
99 | remember = v.StringBoolean(if_missing=False) |
|
101 | remember = v.StringBoolean(if_missing=False) | |
100 |
|
102 | |||
101 | chained_validators = [v.ValidAuth()] |
|
103 | chained_validators = [v.ValidAuth()] | |
102 | return _LoginForm |
|
104 | return _LoginForm | |
103 |
|
105 | |||
104 |
|
106 | |||
105 | def UserForm(edit=False, available_languages=[], old_data={}): |
|
107 | def UserForm(edit=False, available_languages=[], old_data={}): | |
106 | class _UserForm(formencode.Schema): |
|
108 | class _UserForm(formencode.Schema): | |
107 | allow_extra_fields = True |
|
109 | allow_extra_fields = True | |
108 | filter_extra_fields = True |
|
110 | filter_extra_fields = True | |
109 | username = All(v.UnicodeString(strip=True, min=1, not_empty=True), |
|
111 | username = All(v.UnicodeString(strip=True, min=1, not_empty=True), | |
110 | v.ValidUsername(edit, old_data)) |
|
112 | v.ValidUsername(edit, old_data)) | |
111 | if edit: |
|
113 | if edit: | |
112 | new_password = All( |
|
114 | new_password = All( | |
113 | v.ValidPassword(), |
|
115 | v.ValidPassword(), | |
114 | v.UnicodeString(strip=False, min=6, not_empty=False) |
|
116 | v.UnicodeString(strip=False, min=6, not_empty=False) | |
115 | ) |
|
117 | ) | |
116 | password_confirmation = All( |
|
118 | password_confirmation = All( | |
117 | v.ValidPassword(), |
|
119 | v.ValidPassword(), | |
118 | v.UnicodeString(strip=False, min=6, not_empty=False), |
|
120 | v.UnicodeString(strip=False, min=6, not_empty=False), | |
119 | ) |
|
121 | ) | |
120 | admin = v.StringBoolean(if_missing=False) |
|
122 | admin = v.StringBoolean(if_missing=False) | |
121 | else: |
|
123 | else: | |
122 | password = All( |
|
124 | password = All( | |
123 | v.ValidPassword(), |
|
125 | v.ValidPassword(), | |
124 | v.UnicodeString(strip=False, min=6, not_empty=True) |
|
126 | v.UnicodeString(strip=False, min=6, not_empty=True) | |
125 | ) |
|
127 | ) | |
126 | password_confirmation = All( |
|
128 | password_confirmation = All( | |
127 | v.ValidPassword(), |
|
129 | v.ValidPassword(), | |
128 | v.UnicodeString(strip=False, min=6, not_empty=False) |
|
130 | v.UnicodeString(strip=False, min=6, not_empty=False) | |
129 | ) |
|
131 | ) | |
130 |
|
132 | |||
131 | password_change = v.StringBoolean(if_missing=False) |
|
133 | password_change = v.StringBoolean(if_missing=False) | |
132 | create_repo_group = v.StringBoolean(if_missing=False) |
|
134 | create_repo_group = v.StringBoolean(if_missing=False) | |
133 |
|
135 | |||
134 | active = v.StringBoolean(if_missing=False) |
|
136 | active = v.StringBoolean(if_missing=False) | |
135 | firstname = v.UnicodeString(strip=True, min=1, not_empty=False) |
|
137 | firstname = v.UnicodeString(strip=True, min=1, not_empty=False) | |
136 | lastname = v.UnicodeString(strip=True, min=1, not_empty=False) |
|
138 | lastname = v.UnicodeString(strip=True, min=1, not_empty=False) | |
137 | email = All(v.Email(not_empty=True), v.UniqSystemEmail(old_data)) |
|
139 | email = All(v.Email(not_empty=True), v.UniqSystemEmail(old_data)) | |
138 | extern_name = v.UnicodeString(strip=True) |
|
140 | extern_name = v.UnicodeString(strip=True) | |
139 | extern_type = v.UnicodeString(strip=True) |
|
141 | extern_type = v.UnicodeString(strip=True) | |
140 | language = v.OneOf(available_languages, hideList=False, |
|
142 | language = v.OneOf(available_languages, hideList=False, | |
141 | testValueList=True, if_missing=None) |
|
143 | testValueList=True, if_missing=None) | |
142 | chained_validators = [v.ValidPasswordsMatch()] |
|
144 | chained_validators = [v.ValidPasswordsMatch()] | |
143 | return _UserForm |
|
145 | return _UserForm | |
144 |
|
146 | |||
145 |
|
147 | |||
146 | def UserGroupForm(edit=False, old_data=None, allow_disabled=False): |
|
148 | def UserGroupForm(edit=False, old_data=None, allow_disabled=False): | |
147 | old_data = old_data or {} |
|
149 | old_data = old_data or {} | |
148 |
|
150 | |||
149 | class _UserGroupForm(formencode.Schema): |
|
151 | class _UserGroupForm(formencode.Schema): | |
150 | allow_extra_fields = True |
|
152 | allow_extra_fields = True | |
151 | filter_extra_fields = True |
|
153 | filter_extra_fields = True | |
152 |
|
154 | |||
153 | users_group_name = All( |
|
155 | users_group_name = All( | |
154 | v.UnicodeString(strip=True, min=1, not_empty=True), |
|
156 | v.UnicodeString(strip=True, min=1, not_empty=True), | |
155 | v.ValidUserGroup(edit, old_data) |
|
157 | v.ValidUserGroup(edit, old_data) | |
156 | ) |
|
158 | ) | |
157 | user_group_description = v.UnicodeString(strip=True, min=1, |
|
159 | user_group_description = v.UnicodeString(strip=True, min=1, | |
158 | not_empty=False) |
|
160 | not_empty=False) | |
159 |
|
161 | |||
160 | users_group_active = v.StringBoolean(if_missing=False) |
|
162 | users_group_active = v.StringBoolean(if_missing=False) | |
161 |
|
163 | |||
162 | if edit: |
|
164 | if edit: | |
163 | # this is user group owner |
|
165 | # this is user group owner | |
164 | user = All( |
|
166 | user = All( | |
165 | v.UnicodeString(not_empty=True), |
|
167 | v.UnicodeString(not_empty=True), | |
166 | v.ValidRepoUser(allow_disabled)) |
|
168 | v.ValidRepoUser(allow_disabled)) | |
167 | return _UserGroupForm |
|
169 | return _UserGroupForm | |
168 |
|
170 | |||
169 |
|
171 | |||
170 | def RepoGroupForm(edit=False, old_data=None, available_groups=None, |
|
172 | def RepoGroupForm(edit=False, old_data=None, available_groups=None, | |
171 | can_create_in_root=False, allow_disabled=False): |
|
173 | can_create_in_root=False, allow_disabled=False): | |
172 | old_data = old_data or {} |
|
174 | old_data = old_data or {} | |
173 | available_groups = available_groups or [] |
|
175 | available_groups = available_groups or [] | |
174 |
|
176 | |||
175 | class _RepoGroupForm(formencode.Schema): |
|
177 | class _RepoGroupForm(formencode.Schema): | |
176 | allow_extra_fields = True |
|
178 | allow_extra_fields = True | |
177 | filter_extra_fields = False |
|
179 | filter_extra_fields = False | |
178 |
|
180 | |||
179 | group_name = All(v.UnicodeString(strip=True, min=1, not_empty=True), |
|
181 | group_name = All(v.UnicodeString(strip=True, min=1, not_empty=True), | |
180 | v.SlugifyName(),) |
|
182 | v.SlugifyName(),) | |
181 | group_description = v.UnicodeString(strip=True, min=1, |
|
183 | group_description = v.UnicodeString(strip=True, min=1, | |
182 | not_empty=False) |
|
184 | not_empty=False) | |
183 | group_copy_permissions = v.StringBoolean(if_missing=False) |
|
185 | group_copy_permissions = v.StringBoolean(if_missing=False) | |
184 |
|
186 | |||
185 | group_parent_id = v.OneOf(available_groups, hideList=False, |
|
187 | group_parent_id = v.OneOf(available_groups, hideList=False, | |
186 | testValueList=True, not_empty=True) |
|
188 | testValueList=True, not_empty=True) | |
187 | enable_locking = v.StringBoolean(if_missing=False) |
|
189 | enable_locking = v.StringBoolean(if_missing=False) | |
188 | chained_validators = [ |
|
190 | chained_validators = [ | |
189 | v.ValidRepoGroup(edit, old_data, can_create_in_root)] |
|
191 | v.ValidRepoGroup(edit, old_data, can_create_in_root)] | |
190 |
|
192 | |||
191 | if edit: |
|
193 | if edit: | |
192 | # this is repo group owner |
|
194 | # this is repo group owner | |
193 | user = All( |
|
195 | user = All( | |
194 | v.UnicodeString(not_empty=True), |
|
196 | v.UnicodeString(not_empty=True), | |
195 | v.ValidRepoUser(allow_disabled)) |
|
197 | v.ValidRepoUser(allow_disabled)) | |
196 |
|
198 | |||
197 | return _RepoGroupForm |
|
199 | return _RepoGroupForm | |
198 |
|
200 | |||
199 |
|
201 | |||
200 | def RegisterForm(edit=False, old_data={}): |
|
202 | def RegisterForm(edit=False, old_data={}): | |
201 | class _RegisterForm(formencode.Schema): |
|
203 | class _RegisterForm(formencode.Schema): | |
202 | allow_extra_fields = True |
|
204 | allow_extra_fields = True | |
203 | filter_extra_fields = True |
|
205 | filter_extra_fields = True | |
204 | username = All( |
|
206 | username = All( | |
205 | v.ValidUsername(edit, old_data), |
|
207 | v.ValidUsername(edit, old_data), | |
206 | v.UnicodeString(strip=True, min=1, not_empty=True) |
|
208 | v.UnicodeString(strip=True, min=1, not_empty=True) | |
207 | ) |
|
209 | ) | |
208 | password = All( |
|
210 | password = All( | |
209 | v.ValidPassword(), |
|
211 | v.ValidPassword(), | |
210 | v.UnicodeString(strip=False, min=6, not_empty=True) |
|
212 | v.UnicodeString(strip=False, min=6, not_empty=True) | |
211 | ) |
|
213 | ) | |
212 | password_confirmation = All( |
|
214 | password_confirmation = All( | |
213 | v.ValidPassword(), |
|
215 | v.ValidPassword(), | |
214 | v.UnicodeString(strip=False, min=6, not_empty=True) |
|
216 | v.UnicodeString(strip=False, min=6, not_empty=True) | |
215 | ) |
|
217 | ) | |
216 | active = v.StringBoolean(if_missing=False) |
|
218 | active = v.StringBoolean(if_missing=False) | |
217 | firstname = v.UnicodeString(strip=True, min=1, not_empty=False) |
|
219 | firstname = v.UnicodeString(strip=True, min=1, not_empty=False) | |
218 | lastname = v.UnicodeString(strip=True, min=1, not_empty=False) |
|
220 | lastname = v.UnicodeString(strip=True, min=1, not_empty=False) | |
219 | email = All(v.Email(not_empty=True), v.UniqSystemEmail(old_data)) |
|
221 | email = All(v.Email(not_empty=True), v.UniqSystemEmail(old_data)) | |
220 |
|
222 | |||
221 | chained_validators = [v.ValidPasswordsMatch()] |
|
223 | chained_validators = [v.ValidPasswordsMatch()] | |
222 |
|
224 | |||
223 | return _RegisterForm |
|
225 | return _RegisterForm | |
224 |
|
226 | |||
225 |
|
227 | |||
226 | def PasswordResetForm(): |
|
228 | def PasswordResetForm(): | |
227 | class _PasswordResetForm(formencode.Schema): |
|
229 | class _PasswordResetForm(formencode.Schema): | |
228 | allow_extra_fields = True |
|
230 | allow_extra_fields = True | |
229 | filter_extra_fields = True |
|
231 | filter_extra_fields = True | |
230 | email = All(v.ValidSystemEmail(), v.Email(not_empty=True)) |
|
232 | email = All(v.ValidSystemEmail(), v.Email(not_empty=True)) | |
231 | return _PasswordResetForm |
|
233 | return _PasswordResetForm | |
232 |
|
234 | |||
233 |
|
235 | |||
234 | def RepoForm(edit=False, old_data=None, repo_groups=None, landing_revs=None, |
|
236 | def RepoForm(edit=False, old_data=None, repo_groups=None, landing_revs=None, | |
235 | allow_disabled=False): |
|
237 | allow_disabled=False): | |
236 | old_data = old_data or {} |
|
238 | old_data = old_data or {} | |
237 | repo_groups = repo_groups or [] |
|
239 | repo_groups = repo_groups or [] | |
238 | landing_revs = landing_revs or [] |
|
240 | landing_revs = landing_revs or [] | |
239 | supported_backends = BACKENDS.keys() |
|
241 | supported_backends = BACKENDS.keys() | |
240 |
|
242 | |||
241 | class _RepoForm(formencode.Schema): |
|
243 | class _RepoForm(formencode.Schema): | |
242 | allow_extra_fields = True |
|
244 | allow_extra_fields = True | |
243 | filter_extra_fields = False |
|
245 | filter_extra_fields = False | |
244 | repo_name = All(v.UnicodeString(strip=True, min=1, not_empty=True), |
|
246 | repo_name = All(v.UnicodeString(strip=True, min=1, not_empty=True), | |
245 | v.SlugifyName(), v.CannotHaveGitSuffix()) |
|
247 | v.SlugifyName(), v.CannotHaveGitSuffix()) | |
246 | repo_group = All(v.CanWriteGroup(old_data), |
|
248 | repo_group = All(v.CanWriteGroup(old_data), | |
247 | v.OneOf(repo_groups, hideList=True)) |
|
249 | v.OneOf(repo_groups, hideList=True)) | |
248 | repo_type = v.OneOf(supported_backends, required=False, |
|
250 | repo_type = v.OneOf(supported_backends, required=False, | |
249 | if_missing=old_data.get('repo_type')) |
|
251 | if_missing=old_data.get('repo_type')) | |
250 | repo_description = v.UnicodeString(strip=True, min=1, not_empty=False) |
|
252 | repo_description = v.UnicodeString(strip=True, min=1, not_empty=False) | |
251 | repo_private = v.StringBoolean(if_missing=False) |
|
253 | repo_private = v.StringBoolean(if_missing=False) | |
252 | repo_landing_rev = v.OneOf(landing_revs, hideList=True) |
|
254 | repo_landing_rev = v.OneOf(landing_revs, hideList=True) | |
253 | repo_copy_permissions = v.StringBoolean(if_missing=False) |
|
255 | repo_copy_permissions = v.StringBoolean(if_missing=False) | |
254 | clone_uri = All(v.UnicodeString(strip=True, min=1, not_empty=False)) |
|
256 | clone_uri = All(v.UnicodeString(strip=True, min=1, not_empty=False)) | |
255 |
|
257 | |||
256 | repo_enable_statistics = v.StringBoolean(if_missing=False) |
|
258 | repo_enable_statistics = v.StringBoolean(if_missing=False) | |
257 | repo_enable_downloads = v.StringBoolean(if_missing=False) |
|
259 | repo_enable_downloads = v.StringBoolean(if_missing=False) | |
258 | repo_enable_locking = v.StringBoolean(if_missing=False) |
|
260 | repo_enable_locking = v.StringBoolean(if_missing=False) | |
259 |
|
261 | |||
260 | if edit: |
|
262 | if edit: | |
261 | # this is repo owner |
|
263 | # this is repo owner | |
262 | user = All( |
|
264 | user = All( | |
263 | v.UnicodeString(not_empty=True), |
|
265 | v.UnicodeString(not_empty=True), | |
264 | v.ValidRepoUser(allow_disabled)) |
|
266 | v.ValidRepoUser(allow_disabled)) | |
265 | clone_uri_change = v.UnicodeString( |
|
267 | clone_uri_change = v.UnicodeString( | |
266 | not_empty=False, if_missing=v.Missing) |
|
268 | not_empty=False, if_missing=v.Missing) | |
267 |
|
269 | |||
268 | chained_validators = [v.ValidCloneUri(), |
|
270 | chained_validators = [v.ValidCloneUri(), | |
269 | v.ValidRepoName(edit, old_data)] |
|
271 | v.ValidRepoName(edit, old_data)] | |
270 | return _RepoForm |
|
272 | return _RepoForm | |
271 |
|
273 | |||
272 |
|
274 | |||
273 | def RepoPermsForm(): |
|
275 | def RepoPermsForm(): | |
274 | class _RepoPermsForm(formencode.Schema): |
|
276 | class _RepoPermsForm(formencode.Schema): | |
275 | allow_extra_fields = True |
|
277 | allow_extra_fields = True | |
276 | filter_extra_fields = False |
|
278 | filter_extra_fields = False | |
277 | chained_validators = [v.ValidPerms(type_='repo')] |
|
279 | chained_validators = [v.ValidPerms(type_='repo')] | |
278 | return _RepoPermsForm |
|
280 | return _RepoPermsForm | |
279 |
|
281 | |||
280 |
|
282 | |||
281 | def RepoGroupPermsForm(valid_recursive_choices): |
|
283 | def RepoGroupPermsForm(valid_recursive_choices): | |
282 | class _RepoGroupPermsForm(formencode.Schema): |
|
284 | class _RepoGroupPermsForm(formencode.Schema): | |
283 | allow_extra_fields = True |
|
285 | allow_extra_fields = True | |
284 | filter_extra_fields = False |
|
286 | filter_extra_fields = False | |
285 | recursive = v.OneOf(valid_recursive_choices) |
|
287 | recursive = v.OneOf(valid_recursive_choices) | |
286 | chained_validators = [v.ValidPerms(type_='repo_group')] |
|
288 | chained_validators = [v.ValidPerms(type_='repo_group')] | |
287 | return _RepoGroupPermsForm |
|
289 | return _RepoGroupPermsForm | |
288 |
|
290 | |||
289 |
|
291 | |||
290 | def UserGroupPermsForm(): |
|
292 | def UserGroupPermsForm(): | |
291 | class _UserPermsForm(formencode.Schema): |
|
293 | class _UserPermsForm(formencode.Schema): | |
292 | allow_extra_fields = True |
|
294 | allow_extra_fields = True | |
293 | filter_extra_fields = False |
|
295 | filter_extra_fields = False | |
294 | chained_validators = [v.ValidPerms(type_='user_group')] |
|
296 | chained_validators = [v.ValidPerms(type_='user_group')] | |
295 | return _UserPermsForm |
|
297 | return _UserPermsForm | |
296 |
|
298 | |||
297 |
|
299 | |||
298 | def RepoFieldForm(): |
|
300 | def RepoFieldForm(): | |
299 | class _RepoFieldForm(formencode.Schema): |
|
301 | class _RepoFieldForm(formencode.Schema): | |
300 | filter_extra_fields = True |
|
302 | filter_extra_fields = True | |
301 | allow_extra_fields = True |
|
303 | allow_extra_fields = True | |
302 |
|
304 | |||
303 | new_field_key = All(v.FieldKey(), |
|
305 | new_field_key = All(v.FieldKey(), | |
304 | v.UnicodeString(strip=True, min=3, not_empty=True)) |
|
306 | v.UnicodeString(strip=True, min=3, not_empty=True)) | |
305 | new_field_value = v.UnicodeString(not_empty=False, if_missing=u'') |
|
307 | new_field_value = v.UnicodeString(not_empty=False, if_missing=u'') | |
306 | new_field_type = v.OneOf(['str', 'unicode', 'list', 'tuple'], |
|
308 | new_field_type = v.OneOf(['str', 'unicode', 'list', 'tuple'], | |
307 | if_missing='str') |
|
309 | if_missing='str') | |
308 | new_field_label = v.UnicodeString(not_empty=False) |
|
310 | new_field_label = v.UnicodeString(not_empty=False) | |
309 | new_field_desc = v.UnicodeString(not_empty=False) |
|
311 | new_field_desc = v.UnicodeString(not_empty=False) | |
310 |
|
312 | |||
311 | return _RepoFieldForm |
|
313 | return _RepoFieldForm | |
312 |
|
314 | |||
313 |
|
315 | |||
314 | def RepoForkForm(edit=False, old_data={}, supported_backends=BACKENDS.keys(), |
|
316 | def RepoForkForm(edit=False, old_data={}, supported_backends=BACKENDS.keys(), | |
315 | repo_groups=[], landing_revs=[]): |
|
317 | repo_groups=[], landing_revs=[]): | |
316 | class _RepoForkForm(formencode.Schema): |
|
318 | class _RepoForkForm(formencode.Schema): | |
317 | allow_extra_fields = True |
|
319 | allow_extra_fields = True | |
318 | filter_extra_fields = False |
|
320 | filter_extra_fields = False | |
319 | repo_name = All(v.UnicodeString(strip=True, min=1, not_empty=True), |
|
321 | repo_name = All(v.UnicodeString(strip=True, min=1, not_empty=True), | |
320 | v.SlugifyName()) |
|
322 | v.SlugifyName()) | |
321 | repo_group = All(v.CanWriteGroup(), |
|
323 | repo_group = All(v.CanWriteGroup(), | |
322 | v.OneOf(repo_groups, hideList=True)) |
|
324 | v.OneOf(repo_groups, hideList=True)) | |
323 | repo_type = All(v.ValidForkType(old_data), v.OneOf(supported_backends)) |
|
325 | repo_type = All(v.ValidForkType(old_data), v.OneOf(supported_backends)) | |
324 | description = v.UnicodeString(strip=True, min=1, not_empty=True) |
|
326 | description = v.UnicodeString(strip=True, min=1, not_empty=True) | |
325 | private = v.StringBoolean(if_missing=False) |
|
327 | private = v.StringBoolean(if_missing=False) | |
326 | copy_permissions = v.StringBoolean(if_missing=False) |
|
328 | copy_permissions = v.StringBoolean(if_missing=False) | |
327 | fork_parent_id = v.UnicodeString() |
|
329 | fork_parent_id = v.UnicodeString() | |
328 | chained_validators = [v.ValidForkName(edit, old_data)] |
|
330 | chained_validators = [v.ValidForkName(edit, old_data)] | |
329 | landing_rev = v.OneOf(landing_revs, hideList=True) |
|
331 | landing_rev = v.OneOf(landing_revs, hideList=True) | |
330 |
|
332 | |||
331 | return _RepoForkForm |
|
333 | return _RepoForkForm | |
332 |
|
334 | |||
333 |
|
335 | |||
334 | def ApplicationSettingsForm(): |
|
336 | def ApplicationSettingsForm(): | |
335 | class _ApplicationSettingsForm(formencode.Schema): |
|
337 | class _ApplicationSettingsForm(formencode.Schema): | |
336 | allow_extra_fields = True |
|
338 | allow_extra_fields = True | |
337 | filter_extra_fields = False |
|
339 | filter_extra_fields = False | |
338 | rhodecode_title = v.UnicodeString(strip=True, max=40, not_empty=False) |
|
340 | rhodecode_title = v.UnicodeString(strip=True, max=40, not_empty=False) | |
339 | rhodecode_realm = v.UnicodeString(strip=True, min=1, not_empty=True) |
|
341 | rhodecode_realm = v.UnicodeString(strip=True, min=1, not_empty=True) | |
340 | rhodecode_pre_code = v.UnicodeString(strip=True, min=1, not_empty=False) |
|
342 | rhodecode_pre_code = v.UnicodeString(strip=True, min=1, not_empty=False) | |
341 | rhodecode_post_code = v.UnicodeString(strip=True, min=1, not_empty=False) |
|
343 | rhodecode_post_code = v.UnicodeString(strip=True, min=1, not_empty=False) | |
342 | rhodecode_captcha_public_key = v.UnicodeString(strip=True, min=1, not_empty=False) |
|
344 | rhodecode_captcha_public_key = v.UnicodeString(strip=True, min=1, not_empty=False) | |
343 | rhodecode_captcha_private_key = v.UnicodeString(strip=True, min=1, not_empty=False) |
|
345 | rhodecode_captcha_private_key = v.UnicodeString(strip=True, min=1, not_empty=False) | |
344 | rhodecode_create_personal_repo_group = v.StringBoolean(if_missing=False) |
|
346 | rhodecode_create_personal_repo_group = v.StringBoolean(if_missing=False) | |
345 | rhodecode_personal_repo_group_pattern = v.UnicodeString(strip=True, min=1, not_empty=False) |
|
347 | rhodecode_personal_repo_group_pattern = v.UnicodeString(strip=True, min=1, not_empty=False) | |
346 |
|
348 | |||
347 | return _ApplicationSettingsForm |
|
349 | return _ApplicationSettingsForm | |
348 |
|
350 | |||
349 |
|
351 | |||
350 | def ApplicationVisualisationForm(): |
|
352 | def ApplicationVisualisationForm(): | |
351 | class _ApplicationVisualisationForm(formencode.Schema): |
|
353 | class _ApplicationVisualisationForm(formencode.Schema): | |
352 | allow_extra_fields = True |
|
354 | allow_extra_fields = True | |
353 | filter_extra_fields = False |
|
355 | filter_extra_fields = False | |
354 | rhodecode_show_public_icon = v.StringBoolean(if_missing=False) |
|
356 | rhodecode_show_public_icon = v.StringBoolean(if_missing=False) | |
355 | rhodecode_show_private_icon = v.StringBoolean(if_missing=False) |
|
357 | rhodecode_show_private_icon = v.StringBoolean(if_missing=False) | |
356 | rhodecode_stylify_metatags = v.StringBoolean(if_missing=False) |
|
358 | rhodecode_stylify_metatags = v.StringBoolean(if_missing=False) | |
357 |
|
359 | |||
358 | rhodecode_repository_fields = v.StringBoolean(if_missing=False) |
|
360 | rhodecode_repository_fields = v.StringBoolean(if_missing=False) | |
359 | rhodecode_lightweight_journal = v.StringBoolean(if_missing=False) |
|
361 | rhodecode_lightweight_journal = v.StringBoolean(if_missing=False) | |
360 | rhodecode_dashboard_items = v.Int(min=5, not_empty=True) |
|
362 | rhodecode_dashboard_items = v.Int(min=5, not_empty=True) | |
361 | rhodecode_admin_grid_items = v.Int(min=5, not_empty=True) |
|
363 | rhodecode_admin_grid_items = v.Int(min=5, not_empty=True) | |
362 | rhodecode_show_version = v.StringBoolean(if_missing=False) |
|
364 | rhodecode_show_version = v.StringBoolean(if_missing=False) | |
363 | rhodecode_use_gravatar = v.StringBoolean(if_missing=False) |
|
365 | rhodecode_use_gravatar = v.StringBoolean(if_missing=False) | |
364 | rhodecode_markup_renderer = v.OneOf(['markdown', 'rst']) |
|
366 | rhodecode_markup_renderer = v.OneOf(['markdown', 'rst']) | |
365 | rhodecode_gravatar_url = v.UnicodeString(min=3) |
|
367 | rhodecode_gravatar_url = v.UnicodeString(min=3) | |
366 | rhodecode_clone_uri_tmpl = v.UnicodeString(min=3) |
|
368 | rhodecode_clone_uri_tmpl = v.UnicodeString(min=3) | |
367 | rhodecode_support_url = v.UnicodeString() |
|
369 | rhodecode_support_url = v.UnicodeString() | |
368 | rhodecode_show_revision_number = v.StringBoolean(if_missing=False) |
|
370 | rhodecode_show_revision_number = v.StringBoolean(if_missing=False) | |
369 | rhodecode_show_sha_length = v.Int(min=4, not_empty=True) |
|
371 | rhodecode_show_sha_length = v.Int(min=4, not_empty=True) | |
370 |
|
372 | |||
371 | return _ApplicationVisualisationForm |
|
373 | return _ApplicationVisualisationForm | |
372 |
|
374 | |||
373 |
|
375 | |||
374 | class _BaseVcsSettingsForm(formencode.Schema): |
|
376 | class _BaseVcsSettingsForm(formencode.Schema): | |
375 | allow_extra_fields = True |
|
377 | allow_extra_fields = True | |
376 | filter_extra_fields = False |
|
378 | filter_extra_fields = False | |
377 | hooks_changegroup_repo_size = v.StringBoolean(if_missing=False) |
|
379 | hooks_changegroup_repo_size = v.StringBoolean(if_missing=False) | |
378 | hooks_changegroup_push_logger = v.StringBoolean(if_missing=False) |
|
380 | hooks_changegroup_push_logger = v.StringBoolean(if_missing=False) | |
379 | hooks_outgoing_pull_logger = v.StringBoolean(if_missing=False) |
|
381 | hooks_outgoing_pull_logger = v.StringBoolean(if_missing=False) | |
380 |
|
382 | |||
381 | # PR/Code-review |
|
383 | # PR/Code-review | |
382 | rhodecode_pr_merge_enabled = v.StringBoolean(if_missing=False) |
|
384 | rhodecode_pr_merge_enabled = v.StringBoolean(if_missing=False) | |
383 | rhodecode_use_outdated_comments = v.StringBoolean(if_missing=False) |
|
385 | rhodecode_use_outdated_comments = v.StringBoolean(if_missing=False) | |
384 |
|
386 | |||
385 | # hg |
|
387 | # hg | |
386 | extensions_largefiles = v.StringBoolean(if_missing=False) |
|
388 | extensions_largefiles = v.StringBoolean(if_missing=False) | |
387 | extensions_evolve = v.StringBoolean(if_missing=False) |
|
389 | extensions_evolve = v.StringBoolean(if_missing=False) | |
388 | phases_publish = v.StringBoolean(if_missing=False) |
|
390 | phases_publish = v.StringBoolean(if_missing=False) | |
389 | rhodecode_hg_use_rebase_for_merging = v.StringBoolean(if_missing=False) |
|
391 | rhodecode_hg_use_rebase_for_merging = v.StringBoolean(if_missing=False) | |
390 |
|
392 | |||
391 | # git |
|
393 | # git | |
392 | vcs_git_lfs_enabled = v.StringBoolean(if_missing=False) |
|
394 | vcs_git_lfs_enabled = v.StringBoolean(if_missing=False) | |
393 |
|
395 | |||
394 | # svn |
|
396 | # svn | |
395 | vcs_svn_proxy_http_requests_enabled = v.StringBoolean(if_missing=False) |
|
397 | vcs_svn_proxy_http_requests_enabled = v.StringBoolean(if_missing=False) | |
396 | vcs_svn_proxy_http_server_url = v.UnicodeString(strip=True, if_missing=None) |
|
398 | vcs_svn_proxy_http_server_url = v.UnicodeString(strip=True, if_missing=None) | |
397 |
|
399 | |||
398 |
|
400 | |||
399 | def ApplicationUiSettingsForm(): |
|
401 | def ApplicationUiSettingsForm(): | |
400 | class _ApplicationUiSettingsForm(_BaseVcsSettingsForm): |
|
402 | class _ApplicationUiSettingsForm(_BaseVcsSettingsForm): | |
401 | web_push_ssl = v.StringBoolean(if_missing=False) |
|
403 | web_push_ssl = v.StringBoolean(if_missing=False) | |
402 | paths_root_path = All( |
|
404 | paths_root_path = All( | |
403 | v.ValidPath(), |
|
405 | v.ValidPath(), | |
404 | v.UnicodeString(strip=True, min=1, not_empty=True) |
|
406 | v.UnicodeString(strip=True, min=1, not_empty=True) | |
405 | ) |
|
407 | ) | |
406 | largefiles_usercache = All( |
|
408 | largefiles_usercache = All( | |
407 | v.ValidPath(), |
|
409 | v.ValidPath(), | |
408 | v.UnicodeString(strip=True, min=2, not_empty=True)) |
|
410 | v.UnicodeString(strip=True, min=2, not_empty=True)) | |
409 | vcs_git_lfs_store_location = All( |
|
411 | vcs_git_lfs_store_location = All( | |
410 | v.ValidPath(), |
|
412 | v.ValidPath(), | |
411 | v.UnicodeString(strip=True, min=2, not_empty=True)) |
|
413 | v.UnicodeString(strip=True, min=2, not_empty=True)) | |
412 | extensions_hgsubversion = v.StringBoolean(if_missing=False) |
|
414 | extensions_hgsubversion = v.StringBoolean(if_missing=False) | |
413 | extensions_hggit = v.StringBoolean(if_missing=False) |
|
415 | extensions_hggit = v.StringBoolean(if_missing=False) | |
414 | new_svn_branch = v.ValidSvnPattern(section='vcs_svn_branch') |
|
416 | new_svn_branch = v.ValidSvnPattern(section='vcs_svn_branch') | |
415 | new_svn_tag = v.ValidSvnPattern(section='vcs_svn_tag') |
|
417 | new_svn_tag = v.ValidSvnPattern(section='vcs_svn_tag') | |
416 |
|
418 | |||
417 | return _ApplicationUiSettingsForm |
|
419 | return _ApplicationUiSettingsForm | |
418 |
|
420 | |||
419 |
|
421 | |||
420 | def RepoVcsSettingsForm(repo_name): |
|
422 | def RepoVcsSettingsForm(repo_name): | |
421 | class _RepoVcsSettingsForm(_BaseVcsSettingsForm): |
|
423 | class _RepoVcsSettingsForm(_BaseVcsSettingsForm): | |
422 | inherit_global_settings = v.StringBoolean(if_missing=False) |
|
424 | inherit_global_settings = v.StringBoolean(if_missing=False) | |
423 | new_svn_branch = v.ValidSvnPattern( |
|
425 | new_svn_branch = v.ValidSvnPattern( | |
424 | section='vcs_svn_branch', repo_name=repo_name) |
|
426 | section='vcs_svn_branch', repo_name=repo_name) | |
425 | new_svn_tag = v.ValidSvnPattern( |
|
427 | new_svn_tag = v.ValidSvnPattern( | |
426 | section='vcs_svn_tag', repo_name=repo_name) |
|
428 | section='vcs_svn_tag', repo_name=repo_name) | |
427 |
|
429 | |||
428 | return _RepoVcsSettingsForm |
|
430 | return _RepoVcsSettingsForm | |
429 |
|
431 | |||
430 |
|
432 | |||
431 | def LabsSettingsForm(): |
|
433 | def LabsSettingsForm(): | |
432 | class _LabSettingsForm(formencode.Schema): |
|
434 | class _LabSettingsForm(formencode.Schema): | |
433 | allow_extra_fields = True |
|
435 | allow_extra_fields = True | |
434 | filter_extra_fields = False |
|
436 | filter_extra_fields = False | |
435 |
|
437 | |||
436 | return _LabSettingsForm |
|
438 | return _LabSettingsForm | |
437 |
|
439 | |||
438 |
|
440 | |||
439 | def ApplicationPermissionsForm( |
|
441 | def ApplicationPermissionsForm( | |
440 | register_choices, password_reset_choices, extern_activate_choices): |
|
442 | register_choices, password_reset_choices, extern_activate_choices): | |
441 | class _DefaultPermissionsForm(formencode.Schema): |
|
443 | class _DefaultPermissionsForm(formencode.Schema): | |
442 | allow_extra_fields = True |
|
444 | allow_extra_fields = True | |
443 | filter_extra_fields = True |
|
445 | filter_extra_fields = True | |
444 |
|
446 | |||
445 | anonymous = v.StringBoolean(if_missing=False) |
|
447 | anonymous = v.StringBoolean(if_missing=False) | |
446 | default_register = v.OneOf(register_choices) |
|
448 | default_register = v.OneOf(register_choices) | |
447 | default_register_message = v.UnicodeString() |
|
449 | default_register_message = v.UnicodeString() | |
448 | default_password_reset = v.OneOf(password_reset_choices) |
|
450 | default_password_reset = v.OneOf(password_reset_choices) | |
449 | default_extern_activate = v.OneOf(extern_activate_choices) |
|
451 | default_extern_activate = v.OneOf(extern_activate_choices) | |
450 |
|
452 | |||
451 | return _DefaultPermissionsForm |
|
453 | return _DefaultPermissionsForm | |
452 |
|
454 | |||
453 |
|
455 | |||
454 | def ObjectPermissionsForm(repo_perms_choices, group_perms_choices, |
|
456 | def ObjectPermissionsForm(repo_perms_choices, group_perms_choices, | |
455 | user_group_perms_choices): |
|
457 | user_group_perms_choices): | |
456 | class _ObjectPermissionsForm(formencode.Schema): |
|
458 | class _ObjectPermissionsForm(formencode.Schema): | |
457 | allow_extra_fields = True |
|
459 | allow_extra_fields = True | |
458 | filter_extra_fields = True |
|
460 | filter_extra_fields = True | |
459 | overwrite_default_repo = v.StringBoolean(if_missing=False) |
|
461 | overwrite_default_repo = v.StringBoolean(if_missing=False) | |
460 | overwrite_default_group = v.StringBoolean(if_missing=False) |
|
462 | overwrite_default_group = v.StringBoolean(if_missing=False) | |
461 | overwrite_default_user_group = v.StringBoolean(if_missing=False) |
|
463 | overwrite_default_user_group = v.StringBoolean(if_missing=False) | |
462 | default_repo_perm = v.OneOf(repo_perms_choices) |
|
464 | default_repo_perm = v.OneOf(repo_perms_choices) | |
463 | default_group_perm = v.OneOf(group_perms_choices) |
|
465 | default_group_perm = v.OneOf(group_perms_choices) | |
464 | default_user_group_perm = v.OneOf(user_group_perms_choices) |
|
466 | default_user_group_perm = v.OneOf(user_group_perms_choices) | |
465 |
|
467 | |||
466 | return _ObjectPermissionsForm |
|
468 | return _ObjectPermissionsForm | |
467 |
|
469 | |||
468 |
|
470 | |||
469 | def UserPermissionsForm(create_choices, create_on_write_choices, |
|
471 | def UserPermissionsForm(create_choices, create_on_write_choices, | |
470 | repo_group_create_choices, user_group_create_choices, |
|
472 | repo_group_create_choices, user_group_create_choices, | |
471 | fork_choices, inherit_default_permissions_choices): |
|
473 | fork_choices, inherit_default_permissions_choices): | |
472 | class _DefaultPermissionsForm(formencode.Schema): |
|
474 | class _DefaultPermissionsForm(formencode.Schema): | |
473 | allow_extra_fields = True |
|
475 | allow_extra_fields = True | |
474 | filter_extra_fields = True |
|
476 | filter_extra_fields = True | |
475 |
|
477 | |||
476 | anonymous = v.StringBoolean(if_missing=False) |
|
478 | anonymous = v.StringBoolean(if_missing=False) | |
477 |
|
479 | |||
478 | default_repo_create = v.OneOf(create_choices) |
|
480 | default_repo_create = v.OneOf(create_choices) | |
479 | default_repo_create_on_write = v.OneOf(create_on_write_choices) |
|
481 | default_repo_create_on_write = v.OneOf(create_on_write_choices) | |
480 | default_user_group_create = v.OneOf(user_group_create_choices) |
|
482 | default_user_group_create = v.OneOf(user_group_create_choices) | |
481 | default_repo_group_create = v.OneOf(repo_group_create_choices) |
|
483 | default_repo_group_create = v.OneOf(repo_group_create_choices) | |
482 | default_fork_create = v.OneOf(fork_choices) |
|
484 | default_fork_create = v.OneOf(fork_choices) | |
483 | default_inherit_default_permissions = v.OneOf(inherit_default_permissions_choices) |
|
485 | default_inherit_default_permissions = v.OneOf(inherit_default_permissions_choices) | |
484 |
|
486 | |||
485 | return _DefaultPermissionsForm |
|
487 | return _DefaultPermissionsForm | |
486 |
|
488 | |||
487 |
|
489 | |||
488 | def UserIndividualPermissionsForm(): |
|
490 | def UserIndividualPermissionsForm(): | |
489 | class _DefaultPermissionsForm(formencode.Schema): |
|
491 | class _DefaultPermissionsForm(formencode.Schema): | |
490 | allow_extra_fields = True |
|
492 | allow_extra_fields = True | |
491 | filter_extra_fields = True |
|
493 | filter_extra_fields = True | |
492 |
|
494 | |||
493 | inherit_default_permissions = v.StringBoolean(if_missing=False) |
|
495 | inherit_default_permissions = v.StringBoolean(if_missing=False) | |
494 |
|
496 | |||
495 | return _DefaultPermissionsForm |
|
497 | return _DefaultPermissionsForm | |
496 |
|
498 | |||
497 |
|
499 | |||
498 | def DefaultsForm(edit=False, old_data={}, supported_backends=BACKENDS.keys()): |
|
500 | def DefaultsForm(edit=False, old_data={}, supported_backends=BACKENDS.keys()): | |
499 | class _DefaultsForm(formencode.Schema): |
|
501 | class _DefaultsForm(formencode.Schema): | |
500 | allow_extra_fields = True |
|
502 | allow_extra_fields = True | |
501 | filter_extra_fields = True |
|
503 | filter_extra_fields = True | |
502 | default_repo_type = v.OneOf(supported_backends) |
|
504 | default_repo_type = v.OneOf(supported_backends) | |
503 | default_repo_private = v.StringBoolean(if_missing=False) |
|
505 | default_repo_private = v.StringBoolean(if_missing=False) | |
504 | default_repo_enable_statistics = v.StringBoolean(if_missing=False) |
|
506 | default_repo_enable_statistics = v.StringBoolean(if_missing=False) | |
505 | default_repo_enable_downloads = v.StringBoolean(if_missing=False) |
|
507 | default_repo_enable_downloads = v.StringBoolean(if_missing=False) | |
506 | default_repo_enable_locking = v.StringBoolean(if_missing=False) |
|
508 | default_repo_enable_locking = v.StringBoolean(if_missing=False) | |
507 |
|
509 | |||
508 | return _DefaultsForm |
|
510 | return _DefaultsForm | |
509 |
|
511 | |||
510 |
|
512 | |||
511 | def AuthSettingsForm(): |
|
513 | def AuthSettingsForm(): | |
512 | class _AuthSettingsForm(formencode.Schema): |
|
514 | class _AuthSettingsForm(formencode.Schema): | |
513 | allow_extra_fields = True |
|
515 | allow_extra_fields = True | |
514 | filter_extra_fields = True |
|
516 | filter_extra_fields = True | |
515 | auth_plugins = All(v.ValidAuthPlugins(), |
|
517 | auth_plugins = All(v.ValidAuthPlugins(), | |
516 | v.UniqueListFromString()(not_empty=True)) |
|
518 | v.UniqueListFromString()(not_empty=True)) | |
517 |
|
519 | |||
518 | return _AuthSettingsForm |
|
520 | return _AuthSettingsForm | |
519 |
|
521 | |||
520 |
|
522 | |||
521 | def UserExtraEmailForm(): |
|
523 | def UserExtraEmailForm(): | |
522 | class _UserExtraEmailForm(formencode.Schema): |
|
524 | class _UserExtraEmailForm(formencode.Schema): | |
523 | email = All(v.UniqSystemEmail(), v.Email(not_empty=True)) |
|
525 | email = All(v.UniqSystemEmail(), v.Email(not_empty=True)) | |
524 | return _UserExtraEmailForm |
|
526 | return _UserExtraEmailForm | |
525 |
|
527 | |||
526 |
|
528 | |||
527 | def UserExtraIpForm(): |
|
529 | def UserExtraIpForm(): | |
528 | class _UserExtraIpForm(formencode.Schema): |
|
530 | class _UserExtraIpForm(formencode.Schema): | |
529 | ip = v.ValidIp()(not_empty=True) |
|
531 | ip = v.ValidIp()(not_empty=True) | |
530 | return _UserExtraIpForm |
|
532 | return _UserExtraIpForm | |
531 |
|
533 | |||
532 |
|
534 | |||
533 |
|
535 | |||
534 | def PullRequestForm(repo_id): |
|
536 | def PullRequestForm(repo_id): | |
535 | class ReviewerForm(formencode.Schema): |
|
537 | class ReviewerForm(formencode.Schema): | |
536 | user_id = v.Int(not_empty=True) |
|
538 | user_id = v.Int(not_empty=True) | |
537 | reasons = All() |
|
539 | reasons = All() | |
538 | mandatory = v.StringBoolean() |
|
540 | mandatory = v.StringBoolean() | |
539 |
|
541 | |||
540 | class _PullRequestForm(formencode.Schema): |
|
542 | class _PullRequestForm(formencode.Schema): | |
541 | allow_extra_fields = True |
|
543 | allow_extra_fields = True | |
542 | filter_extra_fields = True |
|
544 | filter_extra_fields = True | |
543 |
|
545 | |||
544 | common_ancestor = v.UnicodeString(strip=True, required=True) |
|
546 | common_ancestor = v.UnicodeString(strip=True, required=True) | |
545 | source_repo = v.UnicodeString(strip=True, required=True) |
|
547 | source_repo = v.UnicodeString(strip=True, required=True) | |
546 | source_ref = v.UnicodeString(strip=True, required=True) |
|
548 | source_ref = v.UnicodeString(strip=True, required=True) | |
547 | target_repo = v.UnicodeString(strip=True, required=True) |
|
549 | target_repo = v.UnicodeString(strip=True, required=True) | |
548 | target_ref = v.UnicodeString(strip=True, required=True) |
|
550 | target_ref = v.UnicodeString(strip=True, required=True) | |
549 | revisions = All(#v.NotReviewedRevisions(repo_id)(), |
|
551 | revisions = All(#v.NotReviewedRevisions(repo_id)(), | |
550 | v.UniqueList()(not_empty=True)) |
|
552 | v.UniqueList()(not_empty=True)) | |
551 | review_members = formencode.ForEach(ReviewerForm()) |
|
553 | review_members = formencode.ForEach(ReviewerForm()) | |
552 | pullrequest_title = v.UnicodeString(strip=True, required=True) |
|
554 | pullrequest_title = v.UnicodeString(strip=True, required=True) | |
553 | pullrequest_desc = v.UnicodeString(strip=True, required=False) |
|
555 | pullrequest_desc = v.UnicodeString(strip=True, required=False) | |
554 |
|
556 | |||
555 | return _PullRequestForm |
|
557 | return _PullRequestForm | |
556 |
|
558 | |||
557 |
|
559 | |||
558 | def IssueTrackerPatternsForm(): |
|
560 | def IssueTrackerPatternsForm(): | |
559 | class _IssueTrackerPatternsForm(formencode.Schema): |
|
561 | class _IssueTrackerPatternsForm(formencode.Schema): | |
560 | allow_extra_fields = True |
|
562 | allow_extra_fields = True | |
561 | filter_extra_fields = False |
|
563 | filter_extra_fields = False | |
562 | chained_validators = [v.ValidPattern()] |
|
564 | chained_validators = [v.ValidPattern()] | |
563 | return _IssueTrackerPatternsForm |
|
565 | return _IssueTrackerPatternsForm |
@@ -1,116 +1,116 b'' | |||||
1 | ## -*- coding: utf-8 -*- |
|
1 | ## -*- coding: utf-8 -*- | |
2 | <%inherit file="/base/base.mako"/> |
|
2 | <%inherit file="/base/base.mako"/> | |
3 |
|
3 | |||
4 | <%def name="title()"> |
|
4 | <%def name="title()"> | |
5 | ${_('Authentication Settings')} |
|
5 | ${_('Authentication Settings')} | |
6 | %if c.rhodecode_name: |
|
6 | %if c.rhodecode_name: | |
7 | · ${h.branding(c.rhodecode_name)}} |
|
7 | · ${h.branding(c.rhodecode_name)}} | |
8 | %endif |
|
8 | %endif | |
9 | </%def> |
|
9 | </%def> | |
10 |
|
10 | |||
11 | <%def name="breadcrumbs_links()"> |
|
11 | <%def name="breadcrumbs_links()"> | |
12 | ${h.link_to(_('Admin'),h.route_path('admin_home'))} |
|
12 | ${h.link_to(_('Admin'),h.route_path('admin_home'))} | |
13 | » |
|
13 | » | |
14 | ${_('Authentication Plugins')} |
|
14 | ${_('Authentication Plugins')} | |
15 | </%def> |
|
15 | </%def> | |
16 |
|
16 | |||
17 | <%def name="menu_bar_nav()"> |
|
17 | <%def name="menu_bar_nav()"> | |
18 | ${self.menu_items(active='admin')} |
|
18 | ${self.menu_items(active='admin')} | |
19 | </%def> |
|
19 | </%def> | |
20 |
|
20 | |||
21 | <%def name="main()"> |
|
21 | <%def name="main()"> | |
22 |
|
22 | |||
23 | <div class="box"> |
|
23 | <div class="box"> | |
24 | <div class="title"> |
|
24 | <div class="title"> | |
25 | ${self.breadcrumbs()} |
|
25 | ${self.breadcrumbs()} | |
26 | </div> |
|
26 | </div> | |
27 |
|
27 | |||
28 | <div class='sidebar-col-wrapper'> |
|
28 | <div class='sidebar-col-wrapper'> | |
29 |
|
29 | |||
30 | <div class="sidebar"> |
|
30 | <div class="sidebar"> | |
31 | <ul class="nav nav-pills nav-stacked"> |
|
31 | <ul class="nav nav-pills nav-stacked"> | |
32 | % for item in resource.get_root().get_nav_list(): |
|
32 | % for item in resource.get_root().get_nav_list(): | |
33 | <li ${'class=active' if item == resource else ''}> |
|
33 | <li ${'class=active' if item == resource else ''}> | |
34 | <a href="${request.resource_path(item, route_name='auth_home')}">${item.display_name}</a> |
|
34 | <a href="${request.resource_path(item, route_name='auth_home')}">${item.display_name}</a> | |
35 | </li> |
|
35 | </li> | |
36 | % endfor |
|
36 | % endfor | |
37 | </ul> |
|
37 | </ul> | |
38 | </div> |
|
38 | </div> | |
39 |
|
39 | |||
40 | <div class="main-content-full-width"> |
|
40 | <div class="main-content-full-width"> | |
41 | ${h.secure_form(request.resource_path(resource, route_name='auth_home'))} |
|
41 | ${h.secure_form(request.resource_path(resource, route_name='auth_home'), request=request)} | |
42 | <div class="form"> |
|
42 | <div class="form"> | |
43 |
|
43 | |||
44 | <div class="panel panel-default"> |
|
44 | <div class="panel panel-default"> | |
45 |
|
45 | |||
46 | <div class="panel-heading"> |
|
46 | <div class="panel-heading"> | |
47 | <h3 class="panel-title">${_("Enabled and Available Plugins")}</h3> |
|
47 | <h3 class="panel-title">${_("Enabled and Available Plugins")}</h3> | |
48 | </div> |
|
48 | </div> | |
49 |
|
49 | |||
50 | <div class="fields panel-body"> |
|
50 | <div class="fields panel-body"> | |
51 |
|
51 | |||
52 | <div class="field"> |
|
52 | <div class="field"> | |
53 | <div class="label">${_("Enabled Plugins")}</div> |
|
53 | <div class="label">${_("Enabled Plugins")}</div> | |
54 | <div class="textarea text-area editor"> |
|
54 | <div class="textarea text-area editor"> | |
55 | ${h.textarea('auth_plugins',cols=23,rows=5,class_="medium")} |
|
55 | ${h.textarea('auth_plugins',cols=23,rows=5,class_="medium")} | |
56 | </div> |
|
56 | </div> | |
57 | <p class="help-block"> |
|
57 | <p class="help-block"> | |
58 | ${_('Add a list of plugins, separated by commas. ' |
|
58 | ${_('Add a list of plugins, separated by commas. ' | |
59 | 'The order of the plugins is also the order in which ' |
|
59 | 'The order of the plugins is also the order in which ' | |
60 | 'RhodeCode Enterprise will try to authenticate a user.')} |
|
60 | 'RhodeCode Enterprise will try to authenticate a user.')} | |
61 | </p> |
|
61 | </p> | |
62 | </div> |
|
62 | </div> | |
63 |
|
63 | |||
64 | <div class="field"> |
|
64 | <div class="field"> | |
65 | <div class="label">${_('Available Built-in Plugins')}</div> |
|
65 | <div class="label">${_('Available Built-in Plugins')}</div> | |
66 | <ul class="auth_plugins"> |
|
66 | <ul class="auth_plugins"> | |
67 | %for plugin in available_plugins: |
|
67 | %for plugin in available_plugins: | |
68 | <li> |
|
68 | <li> | |
69 | <div class="auth_buttons"> |
|
69 | <div class="auth_buttons"> | |
70 | <span plugin_id="${plugin.get_id()}" class="toggle-plugin btn ${'btn-success' if plugin.get_id() in enabled_plugins else ''}"> |
|
70 | <span plugin_id="${plugin.get_id()}" class="toggle-plugin btn ${'btn-success' if plugin.get_id() in enabled_plugins else ''}"> | |
71 | ${_('enabled') if plugin.get_id() in enabled_plugins else _('disabled')} |
|
71 | ${_('enabled') if plugin.get_id() in enabled_plugins else _('disabled')} | |
72 | </span> |
|
72 | </span> | |
73 | ${plugin.get_display_name()} (${plugin.get_id()}) |
|
73 | ${plugin.get_display_name()} (${plugin.get_id()}) | |
74 | </div> |
|
74 | </div> | |
75 | </li> |
|
75 | </li> | |
76 | %endfor |
|
76 | %endfor | |
77 | </ul> |
|
77 | </ul> | |
78 | </div> |
|
78 | </div> | |
79 |
|
79 | |||
80 | <div class="buttons"> |
|
80 | <div class="buttons"> | |
81 | ${h.submit('save',_('Save'),class_="btn")} |
|
81 | ${h.submit('save',_('Save'),class_="btn")} | |
82 | </div> |
|
82 | </div> | |
83 | </div> |
|
83 | </div> | |
84 | </div> |
|
84 | </div> | |
85 | </div> |
|
85 | </div> | |
86 | ${h.end_form()} |
|
86 | ${h.end_form()} | |
87 | </div> |
|
87 | </div> | |
88 | </div> |
|
88 | </div> | |
89 | </div> |
|
89 | </div> | |
90 |
|
90 | |||
91 | <script> |
|
91 | <script> | |
92 | $('.toggle-plugin').click(function(e){ |
|
92 | $('.toggle-plugin').click(function(e){ | |
93 | var auth_plugins_input = $('#auth_plugins'); |
|
93 | var auth_plugins_input = $('#auth_plugins'); | |
94 | var notEmpty = function(element, index, array) { |
|
94 | var notEmpty = function(element, index, array) { | |
95 | return (element != ""); |
|
95 | return (element != ""); | |
96 | }; |
|
96 | }; | |
97 | var elems = auth_plugins_input.val().split(',').filter(notEmpty); |
|
97 | var elems = auth_plugins_input.val().split(',').filter(notEmpty); | |
98 | var cur_button = e.currentTarget; |
|
98 | var cur_button = e.currentTarget; | |
99 | var plugin_id = $(cur_button).attr('plugin_id'); |
|
99 | var plugin_id = $(cur_button).attr('plugin_id'); | |
100 | if($(cur_button).hasClass('btn-success')){ |
|
100 | if($(cur_button).hasClass('btn-success')){ | |
101 | elems.splice(elems.indexOf(plugin_id), 1); |
|
101 | elems.splice(elems.indexOf(plugin_id), 1); | |
102 | auth_plugins_input.val(elems.join(',')); |
|
102 | auth_plugins_input.val(elems.join(',')); | |
103 | $(cur_button).removeClass('btn-success'); |
|
103 | $(cur_button).removeClass('btn-success'); | |
104 | cur_button.innerHTML = _gettext('disabled'); |
|
104 | cur_button.innerHTML = _gettext('disabled'); | |
105 | } |
|
105 | } | |
106 | else{ |
|
106 | else{ | |
107 | if(elems.indexOf(plugin_id) == -1){ |
|
107 | if(elems.indexOf(plugin_id) == -1){ | |
108 | elems.push(plugin_id); |
|
108 | elems.push(plugin_id); | |
109 | } |
|
109 | } | |
110 | auth_plugins_input.val(elems.join(',')); |
|
110 | auth_plugins_input.val(elems.join(',')); | |
111 | $(cur_button).addClass('btn-success'); |
|
111 | $(cur_button).addClass('btn-success'); | |
112 | cur_button.innerHTML = _gettext('enabled'); |
|
112 | cur_button.innerHTML = _gettext('enabled'); | |
113 | } |
|
113 | } | |
114 | }); |
|
114 | }); | |
115 | </script> |
|
115 | </script> | |
116 | </%def> |
|
116 | </%def> |
@@ -1,118 +1,118 b'' | |||||
1 | ## -*- coding: utf-8 -*- |
|
1 | ## -*- coding: utf-8 -*- | |
2 | <%inherit file="/base/base.mako"/> |
|
2 | <%inherit file="/base/base.mako"/> | |
3 |
|
3 | |||
4 | <%def name="title()"> |
|
4 | <%def name="title()"> | |
5 | ${_('Authentication Settings')} |
|
5 | ${_('Authentication Settings')} | |
6 | %if c.rhodecode_name: |
|
6 | %if c.rhodecode_name: | |
7 | · ${h.branding(c.rhodecode_name)}} |
|
7 | · ${h.branding(c.rhodecode_name)}} | |
8 | %endif |
|
8 | %endif | |
9 | </%def> |
|
9 | </%def> | |
10 |
|
10 | |||
11 | <%def name="breadcrumbs_links()"> |
|
11 | <%def name="breadcrumbs_links()"> | |
12 | ${h.link_to(_('Admin'),h.route_path('admin_home'))} |
|
12 | ${h.link_to(_('Admin'),h.route_path('admin_home'))} | |
13 | » |
|
13 | » | |
14 | ${h.link_to(_('Authentication Plugins'),request.resource_path(resource.__parent__, route_name='auth_home'))} |
|
14 | ${h.link_to(_('Authentication Plugins'),request.resource_path(resource.__parent__, route_name='auth_home'))} | |
15 | » |
|
15 | » | |
16 | ${resource.display_name} |
|
16 | ${resource.display_name} | |
17 | </%def> |
|
17 | </%def> | |
18 |
|
18 | |||
19 | <%def name="menu_bar_nav()"> |
|
19 | <%def name="menu_bar_nav()"> | |
20 | ${self.menu_items(active='admin')} |
|
20 | ${self.menu_items(active='admin')} | |
21 | </%def> |
|
21 | </%def> | |
22 |
|
22 | |||
23 | <%def name="main()"> |
|
23 | <%def name="main()"> | |
24 | <div class="box"> |
|
24 | <div class="box"> | |
25 | <div class="title"> |
|
25 | <div class="title"> | |
26 | ${self.breadcrumbs()} |
|
26 | ${self.breadcrumbs()} | |
27 | </div> |
|
27 | </div> | |
28 | <div class='sidebar-col-wrapper'> |
|
28 | <div class='sidebar-col-wrapper'> | |
29 |
|
29 | |||
30 | ## TODO: This is repeated in the auth root template and should be merged |
|
30 | ## TODO: This is repeated in the auth root template and should be merged | |
31 | ## into a single solution. |
|
31 | ## into a single solution. | |
32 | <div class="sidebar"> |
|
32 | <div class="sidebar"> | |
33 | <ul class="nav nav-pills nav-stacked"> |
|
33 | <ul class="nav nav-pills nav-stacked"> | |
34 | % for item in resource.get_root().get_nav_list(): |
|
34 | % for item in resource.get_root().get_nav_list(): | |
35 | <li ${'class=active' if item == resource else ''}> |
|
35 | <li ${'class=active' if item == resource else ''}> | |
36 | <a href="${request.resource_path(item, route_name='auth_home')}">${item.display_name}</a> |
|
36 | <a href="${request.resource_path(item, route_name='auth_home')}">${item.display_name}</a> | |
37 | </li> |
|
37 | </li> | |
38 | % endfor |
|
38 | % endfor | |
39 | </ul> |
|
39 | </ul> | |
40 | </div> |
|
40 | </div> | |
41 |
|
41 | |||
42 | <div class="main-content-full-width"> |
|
42 | <div class="main-content-full-width"> | |
43 | <div class="panel panel-default"> |
|
43 | <div class="panel panel-default"> | |
44 | <div class="panel-heading"> |
|
44 | <div class="panel-heading"> | |
45 | <h3 class="panel-title">${_('Plugin')}: ${resource.display_name}</h3> |
|
45 | <h3 class="panel-title">${_('Plugin')}: ${resource.display_name}</h3> | |
46 | </div> |
|
46 | </div> | |
47 | <div class="panel-body"> |
|
47 | <div class="panel-body"> | |
48 | <div class="plugin_form"> |
|
48 | <div class="plugin_form"> | |
49 | <div class="fields"> |
|
49 | <div class="fields"> | |
50 | ${h.secure_form(request.resource_path(resource, route_name='auth_home'))} |
|
50 | ${h.secure_form(request.resource_path(resource, route_name='auth_home'), request=request)} | |
51 | <div class="form"> |
|
51 | <div class="form"> | |
52 |
|
52 | |||
53 | %for node in plugin.get_settings_schema(): |
|
53 | %for node in plugin.get_settings_schema(): | |
54 | <% label_css_class = ("label-checkbox" if (node.widget == "bool") else "") %> |
|
54 | <% label_css_class = ("label-checkbox" if (node.widget == "bool") else "") %> | |
55 | <div class="field"> |
|
55 | <div class="field"> | |
56 | <div class="label ${label_css_class}"><label for="${node.name}">${node.title}</label></div> |
|
56 | <div class="label ${label_css_class}"><label for="${node.name}">${node.title}</label></div> | |
57 | <div class="input"> |
|
57 | <div class="input"> | |
58 | %if node.widget in ["string", "int", "unicode"]: |
|
58 | %if node.widget in ["string", "int", "unicode"]: | |
59 | ${h.text(node.name, defaults.get(node.name), class_="medium")} |
|
59 | ${h.text(node.name, defaults.get(node.name), class_="medium")} | |
60 | %elif node.widget == "password": |
|
60 | %elif node.widget == "password": | |
61 | ${h.password(node.name, defaults.get(node.name), class_="medium")} |
|
61 | ${h.password(node.name, defaults.get(node.name), class_="medium")} | |
62 | %elif node.widget == "bool": |
|
62 | %elif node.widget == "bool": | |
63 | <div class="checkbox">${h.checkbox(node.name, True, checked=defaults.get(node.name))}</div> |
|
63 | <div class="checkbox">${h.checkbox(node.name, True, checked=defaults.get(node.name))}</div> | |
64 | %elif node.widget == "select": |
|
64 | %elif node.widget == "select": | |
65 | ${h.select(node.name, defaults.get(node.name), node.validator.choices)} |
|
65 | ${h.select(node.name, defaults.get(node.name), node.validator.choices)} | |
66 | %elif node.widget == "readonly": |
|
66 | %elif node.widget == "readonly": | |
67 | ${node.default} |
|
67 | ${node.default} | |
68 | %else: |
|
68 | %else: | |
69 | This field is of type ${node.typ}, which cannot be displayed. Must be one of [string|int|bool|select]. |
|
69 | This field is of type ${node.typ}, which cannot be displayed. Must be one of [string|int|bool|select]. | |
70 | %endif |
|
70 | %endif | |
71 | %if node.name in errors: |
|
71 | %if node.name in errors: | |
72 | <span class="error-message">${errors.get(node.name)}</span> |
|
72 | <span class="error-message">${errors.get(node.name)}</span> | |
73 | <br /> |
|
73 | <br /> | |
74 | %endif |
|
74 | %endif | |
75 | <p class="help-block pre-formatting">${node.description}</p> |
|
75 | <p class="help-block pre-formatting">${node.description}</p> | |
76 | </div> |
|
76 | </div> | |
77 | </div> |
|
77 | </div> | |
78 | %endfor |
|
78 | %endfor | |
79 |
|
79 | |||
80 | ## Allow derived templates to add something below the form |
|
80 | ## Allow derived templates to add something below the form | |
81 | ## input fields |
|
81 | ## input fields | |
82 | %if hasattr(next, 'below_form_fields'): |
|
82 | %if hasattr(next, 'below_form_fields'): | |
83 | ${next.below_form_fields()} |
|
83 | ${next.below_form_fields()} | |
84 | %endif |
|
84 | %endif | |
85 |
|
85 | |||
86 | <div class="buttons"> |
|
86 | <div class="buttons"> | |
87 | ${h.submit('save',_('Save'),class_="btn")} |
|
87 | ${h.submit('save',_('Save'),class_="btn")} | |
88 | </div> |
|
88 | </div> | |
89 |
|
89 | |||
90 | </div> |
|
90 | </div> | |
91 | ${h.end_form()} |
|
91 | ${h.end_form()} | |
92 | </div> |
|
92 | </div> | |
93 | </div> |
|
93 | </div> | |
94 | </div> |
|
94 | </div> | |
95 | </div> |
|
95 | </div> | |
96 | </div> |
|
96 | </div> | |
97 |
|
97 | |||
98 | </div> |
|
98 | </div> | |
99 | </div> |
|
99 | </div> | |
100 |
|
100 | |||
101 | ## TODO: Ugly hack to get ldap select elements to work. |
|
101 | ## TODO: Ugly hack to get ldap select elements to work. | |
102 | ## Find a solution to integrate this nicely. |
|
102 | ## Find a solution to integrate this nicely. | |
103 | <script> |
|
103 | <script> | |
104 | $(document).ready(function() { |
|
104 | $(document).ready(function() { | |
105 | var select2Options = { |
|
105 | var select2Options = { | |
106 | containerCssClass: 'drop-menu', |
|
106 | containerCssClass: 'drop-menu', | |
107 | dropdownCssClass: 'drop-menu-dropdown', |
|
107 | dropdownCssClass: 'drop-menu-dropdown', | |
108 | dropdownAutoWidth: true, |
|
108 | dropdownAutoWidth: true, | |
109 | minimumResultsForSearch: -1 |
|
109 | minimumResultsForSearch: -1 | |
110 | }; |
|
110 | }; | |
111 | $("#tls_kind").select2(select2Options); |
|
111 | $("#tls_kind").select2(select2Options); | |
112 | $("#tls_reqcert").select2(select2Options); |
|
112 | $("#tls_reqcert").select2(select2Options); | |
113 | $("#search_scope").select2(select2Options); |
|
113 | $("#search_scope").select2(select2Options); | |
114 | $("#group_extraction_type").select2(select2Options); |
|
114 | $("#group_extraction_type").select2(select2Options); | |
115 | $("#admin_groups_sync").select2(select2Options); |
|
115 | $("#admin_groups_sync").select2(select2Options); | |
116 | }); |
|
116 | }); | |
117 | </script> |
|
117 | </script> | |
118 | </%def> |
|
118 | </%def> |
@@ -1,139 +1,139 b'' | |||||
1 | ## -*- coding: utf-8 -*- |
|
1 | ## -*- coding: utf-8 -*- | |
2 | <%inherit file="/base/base.mako"/> |
|
2 | <%inherit file="/base/base.mako"/> | |
3 |
|
3 | |||
4 | <%def name="title()"> |
|
4 | <%def name="title()"> | |
5 | ${_('Edit Gist')} · ${c.gist.gist_access_id} |
|
5 | ${_('Edit Gist')} · ${c.gist.gist_access_id} | |
6 | %if c.rhodecode_name: |
|
6 | %if c.rhodecode_name: | |
7 | · ${h.branding(c.rhodecode_name)} |
|
7 | · ${h.branding(c.rhodecode_name)} | |
8 | %endif |
|
8 | %endif | |
9 | </%def> |
|
9 | </%def> | |
10 |
|
10 | |||
11 | <%def name="breadcrumbs_links()"> |
|
11 | <%def name="breadcrumbs_links()"> | |
12 | ${_('Edit Gist')} · ${c.gist.gist_access_id} |
|
12 | ${_('Edit Gist')} · ${c.gist.gist_access_id} | |
13 | </%def> |
|
13 | </%def> | |
14 |
|
14 | |||
15 | <%def name="menu_bar_nav()"> |
|
15 | <%def name="menu_bar_nav()"> | |
16 | ${self.menu_items(active='gists')} |
|
16 | ${self.menu_items(active='gists')} | |
17 | </%def> |
|
17 | </%def> | |
18 |
|
18 | |||
19 | <%def name="main()"> |
|
19 | <%def name="main()"> | |
20 | <div class="box"> |
|
20 | <div class="box"> | |
21 | <!-- box / title --> |
|
21 | <!-- box / title --> | |
22 | <div class="title"> |
|
22 | <div class="title"> | |
23 | ${self.breadcrumbs()} |
|
23 | ${self.breadcrumbs()} | |
24 | </div> |
|
24 | </div> | |
25 |
|
25 | |||
26 | <div class="table"> |
|
26 | <div class="table"> | |
27 |
|
27 | |||
28 | <div id="files_data"> |
|
28 | <div id="files_data"> | |
29 | ${h.secure_form(h.route_path('gist_update', gist_id=c.gist.gist_access_id), id='eform', method='POST')} |
|
29 | ${h.secure_form(h.route_path('gist_update', gist_id=c.gist.gist_access_id), id='eform', method='POST', request=request)} | |
30 | <div> |
|
30 | <div> | |
31 | <input type="hidden" value="${c.file_last_commit.raw_id}" name="parent_hash"> |
|
31 | <input type="hidden" value="${c.file_last_commit.raw_id}" name="parent_hash"> | |
32 | <textarea id="description" name="description" |
|
32 | <textarea id="description" name="description" | |
33 | placeholder="${_('Gist description ...')}">${c.gist.gist_description}</textarea> |
|
33 | placeholder="${_('Gist description ...')}">${c.gist.gist_description}</textarea> | |
34 | <div> |
|
34 | <div> | |
35 | <span class="gist-gravatar"> |
|
35 | <span class="gist-gravatar"> | |
36 | ${self.gravatar(h.email_or_none(c.rhodecode_user.full_contact), 30)} |
|
36 | ${self.gravatar(h.email_or_none(c.rhodecode_user.full_contact), 30)} | |
37 | </span> |
|
37 | </span> | |
38 | <label for='lifetime'>${_('Gist lifetime')}</label> |
|
38 | <label for='lifetime'>${_('Gist lifetime')}</label> | |
39 | ${h.dropdownmenu('lifetime', '0', c.lifetime_options)} |
|
39 | ${h.dropdownmenu('lifetime', '0', c.lifetime_options)} | |
40 |
|
40 | |||
41 | <label for='gist_acl_level'>${_('Gist access level')}</label> |
|
41 | <label for='gist_acl_level'>${_('Gist access level')}</label> | |
42 | ${h.dropdownmenu('gist_acl_level', c.gist.acl_level, c.acl_options)} |
|
42 | ${h.dropdownmenu('gist_acl_level', c.gist.acl_level, c.acl_options)} | |
43 | </div> |
|
43 | </div> | |
44 | </div> |
|
44 | </div> | |
45 |
|
45 | |||
46 | ## peppercorn schema |
|
46 | ## peppercorn schema | |
47 | <input type="hidden" name="__start__" value="nodes:sequence"/> |
|
47 | <input type="hidden" name="__start__" value="nodes:sequence"/> | |
48 | % for cnt, file in enumerate(c.files): |
|
48 | % for cnt, file in enumerate(c.files): | |
49 | <input type="hidden" name="__start__" value="file:mapping"/> |
|
49 | <input type="hidden" name="__start__" value="file:mapping"/> | |
50 | <div id="codeblock" class="codeblock" > |
|
50 | <div id="codeblock" class="codeblock" > | |
51 | <div class="code-header"> |
|
51 | <div class="code-header"> | |
52 | <div class="form"> |
|
52 | <div class="form"> | |
53 | <div class="fields"> |
|
53 | <div class="fields"> | |
54 | <input type="hidden" name="filename_org" value="${file.path}" > |
|
54 | <input type="hidden" name="filename_org" value="${file.path}" > | |
55 | <input id="filename_${h.FID('f',file.path)}" name="filename" size="30" type="text" value="${file.path}"> |
|
55 | <input id="filename_${h.FID('f',file.path)}" name="filename" size="30" type="text" value="${file.path}"> | |
56 | ${h.dropdownmenu('mimetype' ,'plain',[('plain',_('plain'))],enable_filter=True, id='mimetype_'+h.FID('f',file.path))} |
|
56 | ${h.dropdownmenu('mimetype' ,'plain',[('plain',_('plain'))],enable_filter=True, id='mimetype_'+h.FID('f',file.path))} | |
57 | </div> |
|
57 | </div> | |
58 | </div> |
|
58 | </div> | |
59 | </div> |
|
59 | </div> | |
60 | <div class="editor_container"> |
|
60 | <div class="editor_container"> | |
61 | <pre id="editor_pre"></pre> |
|
61 | <pre id="editor_pre"></pre> | |
62 | <textarea id="editor_${h.FID('f',file.path)}" name="content" >${file.content}</textarea> |
|
62 | <textarea id="editor_${h.FID('f',file.path)}" name="content" >${file.content}</textarea> | |
63 | </div> |
|
63 | </div> | |
64 | </div> |
|
64 | </div> | |
65 | <input type="hidden" name="__end__" /> |
|
65 | <input type="hidden" name="__end__" /> | |
66 |
|
66 | |||
67 | ## dynamic edit box. |
|
67 | ## dynamic edit box. | |
68 | <script type="text/javascript"> |
|
68 | <script type="text/javascript"> | |
69 | $(document).ready(function(){ |
|
69 | $(document).ready(function(){ | |
70 | var myCodeMirror = initCodeMirror( |
|
70 | var myCodeMirror = initCodeMirror( | |
71 | "editor_${h.FID('f',file.path)}", ''); |
|
71 | "editor_${h.FID('f',file.path)}", ''); | |
72 |
|
72 | |||
73 | var modes_select = $("#mimetype_${h.FID('f',file.path)}"); |
|
73 | var modes_select = $("#mimetype_${h.FID('f',file.path)}"); | |
74 | fillCodeMirrorOptions(modes_select); |
|
74 | fillCodeMirrorOptions(modes_select); | |
75 |
|
75 | |||
76 | // try to detect the mode based on the file we edit |
|
76 | // try to detect the mode based on the file we edit | |
77 | var mimetype = "${file.mimetype}"; |
|
77 | var mimetype = "${file.mimetype}"; | |
78 | var detected_mode = detectCodeMirrorMode( |
|
78 | var detected_mode = detectCodeMirrorMode( | |
79 | "${file.path}", mimetype); |
|
79 | "${file.path}", mimetype); | |
80 |
|
80 | |||
81 | if(detected_mode){ |
|
81 | if(detected_mode){ | |
82 | $(modes_select).select2("val", mimetype); |
|
82 | $(modes_select).select2("val", mimetype); | |
83 | $(modes_select).change(); |
|
83 | $(modes_select).change(); | |
84 | setCodeMirrorMode(myCodeMirror, detected_mode); |
|
84 | setCodeMirrorMode(myCodeMirror, detected_mode); | |
85 | } |
|
85 | } | |
86 |
|
86 | |||
87 | var filename_selector = "#filename_${h.FID('f',file.path)}"; |
|
87 | var filename_selector = "#filename_${h.FID('f',file.path)}"; | |
88 | // on change of select field set mode |
|
88 | // on change of select field set mode | |
89 | setCodeMirrorModeFromSelect( |
|
89 | setCodeMirrorModeFromSelect( | |
90 | modes_select, filename_selector, myCodeMirror, null); |
|
90 | modes_select, filename_selector, myCodeMirror, null); | |
91 |
|
91 | |||
92 | // on entering the new filename set mode, from given extension |
|
92 | // on entering the new filename set mode, from given extension | |
93 | setCodeMirrorModeFromInput( |
|
93 | setCodeMirrorModeFromInput( | |
94 | modes_select, filename_selector, myCodeMirror, null); |
|
94 | modes_select, filename_selector, myCodeMirror, null); | |
95 | }); |
|
95 | }); | |
96 | </script> |
|
96 | </script> | |
97 | %endfor |
|
97 | %endfor | |
98 | <input type="hidden" name="__end__" /> |
|
98 | <input type="hidden" name="__end__" /> | |
99 |
|
99 | |||
100 | <div class="pull-right"> |
|
100 | <div class="pull-right"> | |
101 | ${h.submit('update',_('Update Gist'),class_="btn btn-success")} |
|
101 | ${h.submit('update',_('Update Gist'),class_="btn btn-success")} | |
102 | <a class="btn" href="${h.route_path('gist_show', gist_id=c.gist.gist_access_id)}">${_('Cancel')}</a> |
|
102 | <a class="btn" href="${h.route_path('gist_show', gist_id=c.gist.gist_access_id)}">${_('Cancel')}</a> | |
103 | </div> |
|
103 | </div> | |
104 | ${h.end_form()} |
|
104 | ${h.end_form()} | |
105 | </div> |
|
105 | </div> | |
106 | </div> |
|
106 | </div> | |
107 |
|
107 | |||
108 | </div> |
|
108 | </div> | |
109 | <script> |
|
109 | <script> | |
110 | $('#update').on('click', function(e){ |
|
110 | $('#update').on('click', function(e){ | |
111 | e.preventDefault(); |
|
111 | e.preventDefault(); | |
112 |
|
112 | |||
113 | $(this).val('Updating...'); |
|
113 | $(this).val('Updating...'); | |
114 | $(this).attr('disabled', 'disabled'); |
|
114 | $(this).attr('disabled', 'disabled'); | |
115 | // check for newer version. |
|
115 | // check for newer version. | |
116 | $.ajax({ |
|
116 | $.ajax({ | |
117 | url: "${h.route_path('gist_edit_check_revision', gist_id=c.gist.gist_access_id)}", |
|
117 | url: "${h.route_path('gist_edit_check_revision', gist_id=c.gist.gist_access_id)}", | |
118 | data: { |
|
118 | data: { | |
119 | 'revision': '${c.file_last_commit.raw_id}' |
|
119 | 'revision': '${c.file_last_commit.raw_id}' | |
120 | }, |
|
120 | }, | |
121 | dataType: 'json', |
|
121 | dataType: 'json', | |
122 | type: 'GET', |
|
122 | type: 'GET', | |
123 | success: function(data) { |
|
123 | success: function(data) { | |
124 | if(data.success === false){ |
|
124 | if(data.success === false){ | |
125 | message = '${h.literal(_('Gist was updated since you started editing. Copy your changes and click %(here)s to reload the new version.') |
|
125 | message = '${h.literal(_('Gist was updated since you started editing. Copy your changes and click %(here)s to reload the new version.') | |
126 | % {'here': h.link_to('here', h.route_path('gist_edit', gist_id=c.gist.gist_access_id))})}' |
|
126 | % {'here': h.link_to('here', h.route_path('gist_edit', gist_id=c.gist.gist_access_id))})}' | |
127 | alertMessage = [{"message": { |
|
127 | alertMessage = [{"message": { | |
128 | "message": message, "force": "true", "level": "warning"}}]; |
|
128 | "message": message, "force": "true", "level": "warning"}}]; | |
129 | $.Topic('/notifications').publish(alertMessage[0]); |
|
129 | $.Topic('/notifications').publish(alertMessage[0]); | |
130 | } |
|
130 | } | |
131 | else{ |
|
131 | else{ | |
132 | $('#eform').submit(); |
|
132 | $('#eform').submit(); | |
133 | } |
|
133 | } | |
134 | } |
|
134 | } | |
135 | }); |
|
135 | }); | |
136 | }) |
|
136 | }) | |
137 |
|
137 | |||
138 | </script> |
|
138 | </script> | |
139 | </%def> |
|
139 | </%def> |
@@ -1,86 +1,86 b'' | |||||
1 | ## -*- coding: utf-8 -*- |
|
1 | ## -*- coding: utf-8 -*- | |
2 | <%inherit file="/base/base.mako"/> |
|
2 | <%inherit file="/base/base.mako"/> | |
3 |
|
3 | |||
4 | <%def name="title()"> |
|
4 | <%def name="title()"> | |
5 | ${_('New Gist')} |
|
5 | ${_('New Gist')} | |
6 | %if c.rhodecode_name: |
|
6 | %if c.rhodecode_name: | |
7 | · ${h.branding(c.rhodecode_name)} |
|
7 | · ${h.branding(c.rhodecode_name)} | |
8 | %endif |
|
8 | %endif | |
9 | </%def> |
|
9 | </%def> | |
10 |
|
10 | |||
11 | <%def name="breadcrumbs_links()"> |
|
11 | <%def name="breadcrumbs_links()"> | |
12 | ${_('New Gist')} |
|
12 | ${_('New Gist')} | |
13 | </%def> |
|
13 | </%def> | |
14 |
|
14 | |||
15 | <%def name="menu_bar_nav()"> |
|
15 | <%def name="menu_bar_nav()"> | |
16 | ${self.menu_items(active='gists')} |
|
16 | ${self.menu_items(active='gists')} | |
17 | </%def> |
|
17 | </%def> | |
18 |
|
18 | |||
19 | <%def name="main()"> |
|
19 | <%def name="main()"> | |
20 | <div class="box"> |
|
20 | <div class="box"> | |
21 | <!-- box / title --> |
|
21 | <!-- box / title --> | |
22 | <div class="title"> |
|
22 | <div class="title"> | |
23 | ${self.breadcrumbs()} |
|
23 | ${self.breadcrumbs()} | |
24 | </div> |
|
24 | </div> | |
25 |
|
25 | |||
26 | <div class="table"> |
|
26 | <div class="table"> | |
27 | <div id="files_data"> |
|
27 | <div id="files_data"> | |
28 | ${h.secure_form(h.route_path('gists_create'), id='eform', method='POST')} |
|
28 | ${h.secure_form(h.route_path('gists_create'), id='eform', method='POST', request=request)} | |
29 | <div> |
|
29 | <div> | |
30 | <textarea id="description" name="description" placeholder="${_('Gist description ...')}"></textarea> |
|
30 | <textarea id="description" name="description" placeholder="${_('Gist description ...')}"></textarea> | |
31 |
|
31 | |||
32 | <span class="gist-gravatar"> |
|
32 | <span class="gist-gravatar"> | |
33 | ${self.gravatar(c.rhodecode_user.email, 30)} |
|
33 | ${self.gravatar(c.rhodecode_user.email, 30)} | |
34 | </span> |
|
34 | </span> | |
35 | <label for='gistid'>${_('Gist id')}</label> |
|
35 | <label for='gistid'>${_('Gist id')}</label> | |
36 | ${h.text('gistid', placeholder=_('Auto generated'))} |
|
36 | ${h.text('gistid', placeholder=_('Auto generated'))} | |
37 |
|
37 | |||
38 | <label for='lifetime'>${_('Gist lifetime')}</label> |
|
38 | <label for='lifetime'>${_('Gist lifetime')}</label> | |
39 | ${h.dropdownmenu('lifetime', '', c.lifetime_options)} |
|
39 | ${h.dropdownmenu('lifetime', '', c.lifetime_options)} | |
40 |
|
40 | |||
41 | <label for='acl_level'>${_('Gist access level')}</label> |
|
41 | <label for='acl_level'>${_('Gist access level')}</label> | |
42 | ${h.dropdownmenu('gist_acl_level', '', c.acl_options)} |
|
42 | ${h.dropdownmenu('gist_acl_level', '', c.acl_options)} | |
43 |
|
43 | |||
44 | </div> |
|
44 | </div> | |
45 | <div id="codeblock" class="codeblock"> |
|
45 | <div id="codeblock" class="codeblock"> | |
46 | <div class="code-header"> |
|
46 | <div class="code-header"> | |
47 | <div class="form"> |
|
47 | <div class="form"> | |
48 | <div class="fields"> |
|
48 | <div class="fields"> | |
49 | ${h.text('filename', size=30, placeholder=_('name this file...'))} |
|
49 | ${h.text('filename', size=30, placeholder=_('name this file...'))} | |
50 | ${h.dropdownmenu('mimetype','plain',[('plain',_('plain'))],enable_filter=True)} |
|
50 | ${h.dropdownmenu('mimetype','plain',[('plain',_('plain'))],enable_filter=True)} | |
51 | </div> |
|
51 | </div> | |
52 | </div> |
|
52 | </div> | |
53 | </div> |
|
53 | </div> | |
54 | <div id="editor_container"> |
|
54 | <div id="editor_container"> | |
55 | <div id="editor_pre"></div> |
|
55 | <div id="editor_pre"></div> | |
56 | <textarea id="editor" name="content" ></textarea> |
|
56 | <textarea id="editor" name="content" ></textarea> | |
57 | </div> |
|
57 | </div> | |
58 | </div> |
|
58 | </div> | |
59 | <div class="pull-right"> |
|
59 | <div class="pull-right"> | |
60 | ${h.submit('private',_('Create Private Gist'),class_="btn")} |
|
60 | ${h.submit('private',_('Create Private Gist'),class_="btn")} | |
61 | ${h.submit('public',_('Create Public Gist'),class_="btn")} |
|
61 | ${h.submit('public',_('Create Public Gist'),class_="btn")} | |
62 | ${h.reset('reset',_('Reset'),class_="btn")} |
|
62 | ${h.reset('reset',_('Reset'),class_="btn")} | |
63 | </div> |
|
63 | </div> | |
64 | ${h.end_form()} |
|
64 | ${h.end_form()} | |
65 | </div> |
|
65 | </div> | |
66 | </div> |
|
66 | </div> | |
67 |
|
67 | |||
68 | </div> |
|
68 | </div> | |
69 |
|
69 | |||
70 | <script type="text/javascript"> |
|
70 | <script type="text/javascript"> | |
71 | var myCodeMirror = initCodeMirror('editor', ''); |
|
71 | var myCodeMirror = initCodeMirror('editor', ''); | |
72 |
|
72 | |||
73 | var modes_select = $('#mimetype'); |
|
73 | var modes_select = $('#mimetype'); | |
74 | fillCodeMirrorOptions(modes_select); |
|
74 | fillCodeMirrorOptions(modes_select); | |
75 |
|
75 | |||
76 | var filename_selector = '#filename'; |
|
76 | var filename_selector = '#filename'; | |
77 | // on change of select field set mode |
|
77 | // on change of select field set mode | |
78 | setCodeMirrorModeFromSelect( |
|
78 | setCodeMirrorModeFromSelect( | |
79 | modes_select, filename_selector, myCodeMirror, null); |
|
79 | modes_select, filename_selector, myCodeMirror, null); | |
80 |
|
80 | |||
81 | // on entering the new filename set mode, from given extension |
|
81 | // on entering the new filename set mode, from given extension | |
82 | setCodeMirrorModeFromInput( |
|
82 | setCodeMirrorModeFromInput( | |
83 | modes_select, filename_selector, myCodeMirror, null); |
|
83 | modes_select, filename_selector, myCodeMirror, null); | |
84 |
|
84 | |||
85 | </script> |
|
85 | </script> | |
86 | </%def> |
|
86 | </%def> |
@@ -1,110 +1,110 b'' | |||||
1 | ## -*- coding: utf-8 -*- |
|
1 | ## -*- coding: utf-8 -*- | |
2 | <%inherit file="/base/base.mako"/> |
|
2 | <%inherit file="/base/base.mako"/> | |
3 |
|
3 | |||
4 | <%def name="robots()"> |
|
4 | <%def name="robots()"> | |
5 | %if c.gist.gist_type != 'public': |
|
5 | %if c.gist.gist_type != 'public': | |
6 | <meta name="robots" content="noindex, nofollow"> |
|
6 | <meta name="robots" content="noindex, nofollow"> | |
7 | %else: |
|
7 | %else: | |
8 | ${parent.robots()} |
|
8 | ${parent.robots()} | |
9 | %endif |
|
9 | %endif | |
10 | </%def> |
|
10 | </%def> | |
11 |
|
11 | |||
12 | <%def name="title()"> |
|
12 | <%def name="title()"> | |
13 | ${_('Gist')} · ${c.gist.gist_access_id} |
|
13 | ${_('Gist')} · ${c.gist.gist_access_id} | |
14 | %if c.rhodecode_name: |
|
14 | %if c.rhodecode_name: | |
15 | · ${h.branding(c.rhodecode_name)} |
|
15 | · ${h.branding(c.rhodecode_name)} | |
16 | %endif |
|
16 | %endif | |
17 | </%def> |
|
17 | </%def> | |
18 |
|
18 | |||
19 | <%def name="breadcrumbs_links()"> |
|
19 | <%def name="breadcrumbs_links()"> | |
20 | ${_('Gist')} · ${c.gist.gist_access_id} |
|
20 | ${_('Gist')} · ${c.gist.gist_access_id} | |
21 | </%def> |
|
21 | </%def> | |
22 |
|
22 | |||
23 | <%def name="menu_bar_nav()"> |
|
23 | <%def name="menu_bar_nav()"> | |
24 | ${self.menu_items(active='gists')} |
|
24 | ${self.menu_items(active='gists')} | |
25 | </%def> |
|
25 | </%def> | |
26 |
|
26 | |||
27 | <%def name="main()"> |
|
27 | <%def name="main()"> | |
28 | <div class="box"> |
|
28 | <div class="box"> | |
29 | <!-- box / title --> |
|
29 | <!-- box / title --> | |
30 | <div class="title"> |
|
30 | <div class="title"> | |
31 | ${self.breadcrumbs()} |
|
31 | ${self.breadcrumbs()} | |
32 | %if c.rhodecode_user.username != h.DEFAULT_USER: |
|
32 | %if c.rhodecode_user.username != h.DEFAULT_USER: | |
33 | <ul class="links"> |
|
33 | <ul class="links"> | |
34 | <li> |
|
34 | <li> | |
35 | <a href="${h.route_path('gists_new')}" class="btn btn-primary">${_(u'Create New Gist')}</a> |
|
35 | <a href="${h.route_path('gists_new')}" class="btn btn-primary">${_(u'Create New Gist')}</a> | |
36 | </li> |
|
36 | </li> | |
37 | </ul> |
|
37 | </ul> | |
38 | %endif |
|
38 | %endif | |
39 | </div> |
|
39 | </div> | |
40 | <code>${c.gist.gist_url()}</code> |
|
40 | <code>${c.gist.gist_url()}</code> | |
41 | <div class="table"> |
|
41 | <div class="table"> | |
42 | <div id="files_data"> |
|
42 | <div id="files_data"> | |
43 | <div id="codeblock" class="codeblock"> |
|
43 | <div id="codeblock" class="codeblock"> | |
44 | <div class="code-header"> |
|
44 | <div class="code-header"> | |
45 | <div class="stats"> |
|
45 | <div class="stats"> | |
46 | %if h.HasPermissionAny('hg.admin')() or c.gist.gist_owner == c.rhodecode_user.user_id: |
|
46 | %if h.HasPermissionAny('hg.admin')() or c.gist.gist_owner == c.rhodecode_user.user_id: | |
47 | <div class="remove_gist"> |
|
47 | <div class="remove_gist"> | |
48 | ${h.secure_form(h.route_path('gist_delete', gist_id=c.gist.gist_access_id), method='POST')} |
|
48 | ${h.secure_form(h.route_path('gist_delete', gist_id=c.gist.gist_access_id), method='POST', request=request)} | |
49 | ${h.submit('remove_gist', _('Delete'),class_="btn btn-mini btn-danger",onclick="return confirm('"+_('Confirm to delete this Gist')+"');")} |
|
49 | ${h.submit('remove_gist', _('Delete'),class_="btn btn-mini btn-danger",onclick="return confirm('"+_('Confirm to delete this Gist')+"');")} | |
50 | ${h.end_form()} |
|
50 | ${h.end_form()} | |
51 | </div> |
|
51 | </div> | |
52 | %endif |
|
52 | %endif | |
53 | <div class="buttons"> |
|
53 | <div class="buttons"> | |
54 | ## only owner should see that |
|
54 | ## only owner should see that | |
55 | %if h.HasPermissionAny('hg.admin')() or c.gist.gist_owner == c.rhodecode_user.user_id: |
|
55 | %if h.HasPermissionAny('hg.admin')() or c.gist.gist_owner == c.rhodecode_user.user_id: | |
56 | ${h.link_to(_('Edit'), h.route_path('gist_edit', gist_id=c.gist.gist_access_id), class_="btn btn-mini")} |
|
56 | ${h.link_to(_('Edit'), h.route_path('gist_edit', gist_id=c.gist.gist_access_id), class_="btn btn-mini")} | |
57 | %endif |
|
57 | %endif | |
58 | ${h.link_to(_('Show as Raw'), h.route_path('gist_show_formatted', gist_id=c.gist.gist_access_id, revision='tip', format='raw'), class_="btn btn-mini")} |
|
58 | ${h.link_to(_('Show as Raw'), h.route_path('gist_show_formatted', gist_id=c.gist.gist_access_id, revision='tip', format='raw'), class_="btn btn-mini")} | |
59 | </div> |
|
59 | </div> | |
60 | <div class="left" > |
|
60 | <div class="left" > | |
61 | %if c.gist.gist_type != 'public': |
|
61 | %if c.gist.gist_type != 'public': | |
62 | <span class="tag tag-ok disabled">${_('Private Gist')}</span> |
|
62 | <span class="tag tag-ok disabled">${_('Private Gist')}</span> | |
63 | %endif |
|
63 | %endif | |
64 | <span> ${c.gist.gist_description}</span> |
|
64 | <span> ${c.gist.gist_description}</span> | |
65 | <span>${_('Expires')}: |
|
65 | <span>${_('Expires')}: | |
66 | %if c.gist.gist_expires == -1: |
|
66 | %if c.gist.gist_expires == -1: | |
67 | ${_('never')} |
|
67 | ${_('never')} | |
68 | %else: |
|
68 | %else: | |
69 | ${h.age_component(h.time_to_utcdatetime(c.gist.gist_expires))} |
|
69 | ${h.age_component(h.time_to_utcdatetime(c.gist.gist_expires))} | |
70 | %endif |
|
70 | %endif | |
71 | </span> |
|
71 | </span> | |
72 | </div> |
|
72 | </div> | |
73 | </div> |
|
73 | </div> | |
74 |
|
74 | |||
75 | <div class="author"> |
|
75 | <div class="author"> | |
76 | <div title="${h.tooltip(c.file_last_commit.author)}"> |
|
76 | <div title="${h.tooltip(c.file_last_commit.author)}"> | |
77 | ${self.gravatar_with_user(c.file_last_commit.author, 16)} - ${_('created')} ${h.age_component(c.file_last_commit.date)} |
|
77 | ${self.gravatar_with_user(c.file_last_commit.author, 16)} - ${_('created')} ${h.age_component(c.file_last_commit.date)} | |
78 | </div> |
|
78 | </div> | |
79 |
|
79 | |||
80 | </div> |
|
80 | </div> | |
81 | <div class="commit">${h.urlify_commit_message(c.file_last_commit.message, None)}</div> |
|
81 | <div class="commit">${h.urlify_commit_message(c.file_last_commit.message, None)}</div> | |
82 | </div> |
|
82 | </div> | |
83 |
|
83 | |||
84 | ## iterate over the files |
|
84 | ## iterate over the files | |
85 | % for file in c.files: |
|
85 | % for file in c.files: | |
86 | <% renderer = c.render and h.renderer_from_filename(file.path, exclude=['.txt', '.TXT'])%> |
|
86 | <% renderer = c.render and h.renderer_from_filename(file.path, exclude=['.txt', '.TXT'])%> | |
87 | <!-- |
|
87 | <!-- | |
88 | <div id="${h.FID('G', file.path)}" class="stats" > |
|
88 | <div id="${h.FID('G', file.path)}" class="stats" > | |
89 | <a href="${c.gist.gist_url()}">ΒΆ</a> |
|
89 | <a href="${c.gist.gist_url()}">ΒΆ</a> | |
90 | <b >${file.path}</b> |
|
90 | <b >${file.path}</b> | |
91 | <div> |
|
91 | <div> | |
92 | ${h.link_to(_('Show as raw'), h.route_path('gist_show_formatted_path', gist_id=c.gist.gist_access_id, revision=file.commit.raw_id, format='raw', f_path=file.path), class_="btn btn-mini")} |
|
92 | ${h.link_to(_('Show as raw'), h.route_path('gist_show_formatted_path', gist_id=c.gist.gist_access_id, revision=file.commit.raw_id, format='raw', f_path=file.path), class_="btn btn-mini")} | |
93 | </div> |
|
93 | </div> | |
94 | </div> |
|
94 | </div> | |
95 | --> |
|
95 | --> | |
96 | <div class="code-body textarea text-area editor"> |
|
96 | <div class="code-body textarea text-area editor"> | |
97 | %if renderer: |
|
97 | %if renderer: | |
98 | ${h.render(file.content, renderer=renderer)} |
|
98 | ${h.render(file.content, renderer=renderer)} | |
99 | %else: |
|
99 | %else: | |
100 | ${h.pygmentize(file,linenos=True,anchorlinenos=True,lineanchors='L',cssclass="code-highlight")} |
|
100 | ${h.pygmentize(file,linenos=True,anchorlinenos=True,lineanchors='L',cssclass="code-highlight")} | |
101 | %endif |
|
101 | %endif | |
102 | </div> |
|
102 | </div> | |
103 | %endfor |
|
103 | %endfor | |
104 | </div> |
|
104 | </div> | |
105 | </div> |
|
105 | </div> | |
106 | </div> |
|
106 | </div> | |
107 |
|
107 | |||
108 |
|
108 | |||
109 | </div> |
|
109 | </div> | |
110 | </%def> |
|
110 | </%def> |
@@ -1,160 +1,160 b'' | |||||
1 | <div class="panel panel-default"> |
|
1 | <div class="panel panel-default"> | |
2 | <div class="panel-heading"> |
|
2 | <div class="panel-heading"> | |
3 | <h3 class="panel-title">${_('Authentication Tokens')}</h3> |
|
3 | <h3 class="panel-title">${_('Authentication Tokens')}</h3> | |
4 | </div> |
|
4 | </div> | |
5 | <div class="panel-body"> |
|
5 | <div class="panel-body"> | |
6 | <div class="apikeys_wrap"> |
|
6 | <div class="apikeys_wrap"> | |
7 | <p> |
|
7 | <p> | |
8 | ${_('Each token can have a role. Token with a role can be used only in given context, ' |
|
8 | ${_('Each token can have a role. Token with a role can be used only in given context, ' | |
9 | 'e.g. VCS tokens can be used together with the authtoken auth plugin for git/hg/svn operations only.')} |
|
9 | 'e.g. VCS tokens can be used together with the authtoken auth plugin for git/hg/svn operations only.')} | |
10 | </p> |
|
10 | </p> | |
11 | <table class="rctable auth_tokens"> |
|
11 | <table class="rctable auth_tokens"> | |
12 | <tr> |
|
12 | <tr> | |
13 | <th>${_('Token')}</th> |
|
13 | <th>${_('Token')}</th> | |
14 | <th>${_('Scope')}</th> |
|
14 | <th>${_('Scope')}</th> | |
15 | <th>${_('Description')}</th> |
|
15 | <th>${_('Description')}</th> | |
16 | <th>${_('Role')}</th> |
|
16 | <th>${_('Role')}</th> | |
17 | <th>${_('Expiration')}</th> |
|
17 | <th>${_('Expiration')}</th> | |
18 | <th>${_('Action')}</th> |
|
18 | <th>${_('Action')}</th> | |
19 | </tr> |
|
19 | </tr> | |
20 | %if c.user_auth_tokens: |
|
20 | %if c.user_auth_tokens: | |
21 | %for auth_token in c.user_auth_tokens: |
|
21 | %for auth_token in c.user_auth_tokens: | |
22 | <tr class="${'expired' if auth_token.expired else ''}"> |
|
22 | <tr class="${'expired' if auth_token.expired else ''}"> | |
23 | <td class="truncate-wrap td-authtoken"> |
|
23 | <td class="truncate-wrap td-authtoken"> | |
24 | <div class="user_auth_tokens truncate autoexpand"> |
|
24 | <div class="user_auth_tokens truncate autoexpand"> | |
25 | <code>${auth_token.api_key}</code> |
|
25 | <code>${auth_token.api_key}</code> | |
26 | </div> |
|
26 | </div> | |
27 | </td> |
|
27 | </td> | |
28 | <td class="td">${auth_token.scope_humanized}</td> |
|
28 | <td class="td">${auth_token.scope_humanized}</td> | |
29 | <td class="td-wrap">${auth_token.description}</td> |
|
29 | <td class="td-wrap">${auth_token.description}</td> | |
30 | <td class="td-tags"> |
|
30 | <td class="td-tags"> | |
31 | <span class="tag disabled">${auth_token.role_humanized}</span> |
|
31 | <span class="tag disabled">${auth_token.role_humanized}</span> | |
32 | </td> |
|
32 | </td> | |
33 | <td class="td-exp"> |
|
33 | <td class="td-exp"> | |
34 | %if auth_token.expires == -1: |
|
34 | %if auth_token.expires == -1: | |
35 | ${_('never')} |
|
35 | ${_('never')} | |
36 | %else: |
|
36 | %else: | |
37 | %if auth_token.expired: |
|
37 | %if auth_token.expired: | |
38 | <span style="text-decoration: line-through">${h.age_component(h.time_to_utcdatetime(auth_token.expires))}</span> |
|
38 | <span style="text-decoration: line-through">${h.age_component(h.time_to_utcdatetime(auth_token.expires))}</span> | |
39 | %else: |
|
39 | %else: | |
40 | ${h.age_component(h.time_to_utcdatetime(auth_token.expires))} |
|
40 | ${h.age_component(h.time_to_utcdatetime(auth_token.expires))} | |
41 | %endif |
|
41 | %endif | |
42 | %endif |
|
42 | %endif | |
43 | </td> |
|
43 | </td> | |
44 | <td class="td-action"> |
|
44 | <td class="td-action"> | |
45 |
${h.secure_form(h.route_path('my_account_auth_tokens_delete'), method=' |
|
45 | ${h.secure_form(h.route_path('my_account_auth_tokens_delete'), method='POST', request=request)} | |
46 | ${h.hidden('del_auth_token', auth_token.user_api_key_id)} |
|
46 | ${h.hidden('del_auth_token', auth_token.user_api_key_id)} | |
47 | <button class="btn btn-link btn-danger" type="submit" |
|
47 | <button class="btn btn-link btn-danger" type="submit" | |
48 | onclick="return confirm('${_('Confirm to remove this auth token: %s') % auth_token.token_obfuscated}');"> |
|
48 | onclick="return confirm('${_('Confirm to remove this auth token: %s') % auth_token.token_obfuscated}');"> | |
49 | ${_('Delete')} |
|
49 | ${_('Delete')} | |
50 | </button> |
|
50 | </button> | |
51 | ${h.end_form()} |
|
51 | ${h.end_form()} | |
52 | </td> |
|
52 | </td> | |
53 | </tr> |
|
53 | </tr> | |
54 | %endfor |
|
54 | %endfor | |
55 | %else: |
|
55 | %else: | |
56 | <tr><td><div class="ip">${_('No additional auth tokens specified')}</div></td></tr> |
|
56 | <tr><td><div class="ip">${_('No additional auth tokens specified')}</div></td></tr> | |
57 | %endif |
|
57 | %endif | |
58 | </table> |
|
58 | </table> | |
59 | </div> |
|
59 | </div> | |
60 |
|
60 | |||
61 | <div class="user_auth_tokens"> |
|
61 | <div class="user_auth_tokens"> | |
62 |
${h.secure_form(h.route_path('my_account_auth_tokens_add'), method=' |
|
62 | ${h.secure_form(h.route_path('my_account_auth_tokens_add'), method='POST', request=request)} | |
63 | <div class="form form-vertical"> |
|
63 | <div class="form form-vertical"> | |
64 | <!-- fields --> |
|
64 | <!-- fields --> | |
65 | <div class="fields"> |
|
65 | <div class="fields"> | |
66 | <div class="field"> |
|
66 | <div class="field"> | |
67 | <div class="label"> |
|
67 | <div class="label"> | |
68 | <label for="new_email">${_('New authentication token')}:</label> |
|
68 | <label for="new_email">${_('New authentication token')}:</label> | |
69 | </div> |
|
69 | </div> | |
70 | <div class="input"> |
|
70 | <div class="input"> | |
71 | ${h.text('description', class_='medium', placeholder=_('Description'))} |
|
71 | ${h.text('description', class_='medium', placeholder=_('Description'))} | |
72 | ${h.select('lifetime', '', c.lifetime_options)} |
|
72 | ${h.select('lifetime', '', c.lifetime_options)} | |
73 | ${h.select('role', '', c.role_options)} |
|
73 | ${h.select('role', '', c.role_options)} | |
74 |
|
74 | |||
75 | % if c.allow_scoped_tokens: |
|
75 | % if c.allow_scoped_tokens: | |
76 | ${h.hidden('scope_repo_id')} |
|
76 | ${h.hidden('scope_repo_id')} | |
77 | % else: |
|
77 | % else: | |
78 | ${h.select('scope_repo_id_disabled', '', ['Scopes available in EE edition'], disabled='disabled')} |
|
78 | ${h.select('scope_repo_id_disabled', '', ['Scopes available in EE edition'], disabled='disabled')} | |
79 | % endif |
|
79 | % endif | |
80 | </div> |
|
80 | </div> | |
81 | <p class="help-block"> |
|
81 | <p class="help-block"> | |
82 | ${_('Repository scope works only with tokens with VCS type.')} |
|
82 | ${_('Repository scope works only with tokens with VCS type.')} | |
83 | </p> |
|
83 | </p> | |
84 | </div> |
|
84 | </div> | |
85 | <div class="buttons"> |
|
85 | <div class="buttons"> | |
86 | ${h.submit('save',_('Add'),class_="btn")} |
|
86 | ${h.submit('save',_('Add'),class_="btn")} | |
87 | ${h.reset('reset',_('Reset'),class_="btn")} |
|
87 | ${h.reset('reset',_('Reset'),class_="btn")} | |
88 | </div> |
|
88 | </div> | |
89 | </div> |
|
89 | </div> | |
90 | </div> |
|
90 | </div> | |
91 | ${h.end_form()} |
|
91 | ${h.end_form()} | |
92 | </div> |
|
92 | </div> | |
93 | </div> |
|
93 | </div> | |
94 | </div> |
|
94 | </div> | |
95 | <script> |
|
95 | <script> | |
96 | $(document).ready(function(){ |
|
96 | $(document).ready(function(){ | |
97 |
|
97 | |||
98 | var select2Options = { |
|
98 | var select2Options = { | |
99 | 'containerCssClass': "drop-menu", |
|
99 | 'containerCssClass': "drop-menu", | |
100 | 'dropdownCssClass': "drop-menu-dropdown", |
|
100 | 'dropdownCssClass': "drop-menu-dropdown", | |
101 | 'dropdownAutoWidth': true |
|
101 | 'dropdownAutoWidth': true | |
102 | }; |
|
102 | }; | |
103 | $("#lifetime").select2(select2Options); |
|
103 | $("#lifetime").select2(select2Options); | |
104 | $("#role").select2(select2Options); |
|
104 | $("#role").select2(select2Options); | |
105 |
|
105 | |||
106 | var repoFilter = function(data) { |
|
106 | var repoFilter = function(data) { | |
107 | var results = []; |
|
107 | var results = []; | |
108 |
|
108 | |||
109 | if (!data.results[0]) { |
|
109 | if (!data.results[0]) { | |
110 | return data |
|
110 | return data | |
111 | } |
|
111 | } | |
112 |
|
112 | |||
113 | $.each(data.results[0].children, function() { |
|
113 | $.each(data.results[0].children, function() { | |
114 | // replace name to ID for submision |
|
114 | // replace name to ID for submision | |
115 | this.id = this.obj.repo_id; |
|
115 | this.id = this.obj.repo_id; | |
116 | results.push(this); |
|
116 | results.push(this); | |
117 | }); |
|
117 | }); | |
118 |
|
118 | |||
119 | data.results[0].children = results; |
|
119 | data.results[0].children = results; | |
120 | return data; |
|
120 | return data; | |
121 | }; |
|
121 | }; | |
122 |
|
122 | |||
123 | $("#scope_repo_id_disabled").select2(select2Options); |
|
123 | $("#scope_repo_id_disabled").select2(select2Options); | |
124 |
|
124 | |||
125 | $("#scope_repo_id").select2({ |
|
125 | $("#scope_repo_id").select2({ | |
126 | cachedDataSource: {}, |
|
126 | cachedDataSource: {}, | |
127 | minimumInputLength: 2, |
|
127 | minimumInputLength: 2, | |
128 | placeholder: "${_('repository scope')}", |
|
128 | placeholder: "${_('repository scope')}", | |
129 | dropdownAutoWidth: true, |
|
129 | dropdownAutoWidth: true, | |
130 | containerCssClass: "drop-menu", |
|
130 | containerCssClass: "drop-menu", | |
131 | dropdownCssClass: "drop-menu-dropdown", |
|
131 | dropdownCssClass: "drop-menu-dropdown", | |
132 | formatResult: formatResult, |
|
132 | formatResult: formatResult, | |
133 | query: $.debounce(250, function(query){ |
|
133 | query: $.debounce(250, function(query){ | |
134 | self = this; |
|
134 | self = this; | |
135 | var cacheKey = query.term; |
|
135 | var cacheKey = query.term; | |
136 | var cachedData = self.cachedDataSource[cacheKey]; |
|
136 | var cachedData = self.cachedDataSource[cacheKey]; | |
137 |
|
137 | |||
138 | if (cachedData) { |
|
138 | if (cachedData) { | |
139 | query.callback({results: cachedData.results}); |
|
139 | query.callback({results: cachedData.results}); | |
140 | } else { |
|
140 | } else { | |
141 | $.ajax({ |
|
141 | $.ajax({ | |
142 | url: pyroutes.url('repo_list_data'), |
|
142 | url: pyroutes.url('repo_list_data'), | |
143 | data: {'query': query.term}, |
|
143 | data: {'query': query.term}, | |
144 | dataType: 'json', |
|
144 | dataType: 'json', | |
145 | type: 'GET', |
|
145 | type: 'GET', | |
146 | success: function(data) { |
|
146 | success: function(data) { | |
147 | data = repoFilter(data); |
|
147 | data = repoFilter(data); | |
148 | self.cachedDataSource[cacheKey] = data; |
|
148 | self.cachedDataSource[cacheKey] = data; | |
149 | query.callback({results: data.results}); |
|
149 | query.callback({results: data.results}); | |
150 | }, |
|
150 | }, | |
151 | error: function(data, textStatus, errorThrown) { |
|
151 | error: function(data, textStatus, errorThrown) { | |
152 | alert("Error while fetching entries.\nError code {0} ({1}).".format(data.status, data.statusText)); |
|
152 | alert("Error while fetching entries.\nError code {0} ({1}).".format(data.status, data.statusText)); | |
153 | } |
|
153 | } | |
154 | }) |
|
154 | }) | |
155 | } |
|
155 | } | |
156 | }) |
|
156 | }) | |
157 | }); |
|
157 | }); | |
158 |
|
158 | |||
159 | }); |
|
159 | }); | |
160 | </script> |
|
160 | </script> |
@@ -1,72 +1,72 b'' | |||||
1 | <%namespace name="base" file="/base/base.mako"/> |
|
1 | <%namespace name="base" file="/base/base.mako"/> | |
2 |
|
2 | |||
3 | <div class="panel panel-default"> |
|
3 | <div class="panel panel-default"> | |
4 | <div class="panel-heading"> |
|
4 | <div class="panel-heading"> | |
5 | <h3 class="panel-title">${_('Account Emails')}</h3> |
|
5 | <h3 class="panel-title">${_('Account Emails')}</h3> | |
6 | </div> |
|
6 | </div> | |
7 |
|
7 | |||
8 | <div class="panel-body"> |
|
8 | <div class="panel-body"> | |
9 | <div class="emails_wrap"> |
|
9 | <div class="emails_wrap"> | |
10 | <table class="rctable account_emails"> |
|
10 | <table class="rctable account_emails"> | |
11 | <tr> |
|
11 | <tr> | |
12 | <td class="td-user"> |
|
12 | <td class="td-user"> | |
13 | ${base.gravatar(c.user.email, 16)} |
|
13 | ${base.gravatar(c.user.email, 16)} | |
14 | <span class="user email">${c.user.email}</span> |
|
14 | <span class="user email">${c.user.email}</span> | |
15 | </td> |
|
15 | </td> | |
16 | <td class="td-tags"> |
|
16 | <td class="td-tags"> | |
17 | <span class="tag tag1">${_('Primary')}</span> |
|
17 | <span class="tag tag1">${_('Primary')}</span> | |
18 | </td> |
|
18 | </td> | |
19 | </tr> |
|
19 | </tr> | |
20 | %if c.user_email_map: |
|
20 | %if c.user_email_map: | |
21 | %for em in c.user_email_map: |
|
21 | %for em in c.user_email_map: | |
22 | <tr> |
|
22 | <tr> | |
23 | <td class="td-user"> |
|
23 | <td class="td-user"> | |
24 | ${base.gravatar(em.email, 16)} |
|
24 | ${base.gravatar(em.email, 16)} | |
25 | <span class="user email">${em.email}</span> |
|
25 | <span class="user email">${em.email}</span> | |
26 | </td> |
|
26 | </td> | |
27 | <td class="td-action"> |
|
27 | <td class="td-action"> | |
28 | ${h.secure_form(h.route_path('my_account_emails_delete'), method='POST')} |
|
28 | ${h.secure_form(h.route_path('my_account_emails_delete'), method='POST', request=request)} | |
29 | ${h.hidden('del_email_id',em.email_id)} |
|
29 | ${h.hidden('del_email_id',em.email_id)} | |
30 | <button class="btn btn-link btn-danger" type="submit" id="${'remove_email_%s'.format(em.email_id)}" |
|
30 | <button class="btn btn-link btn-danger" type="submit" id="${'remove_email_%s'.format(em.email_id)}" | |
31 | onclick="return confirm('${_('Confirm to delete this email: {}').format(em.email)}');"> |
|
31 | onclick="return confirm('${_('Confirm to delete this email: {}').format(em.email)}');"> | |
32 | ${_('Delete')} |
|
32 | ${_('Delete')} | |
33 | </button> |
|
33 | </button> | |
34 | ${h.end_form()} |
|
34 | ${h.end_form()} | |
35 | </td> |
|
35 | </td> | |
36 | </tr> |
|
36 | </tr> | |
37 | %endfor |
|
37 | %endfor | |
38 | %else: |
|
38 | %else: | |
39 | <tr class="noborder"> |
|
39 | <tr class="noborder"> | |
40 | <td colspan="3"> |
|
40 | <td colspan="3"> | |
41 | <div class="td-email"> |
|
41 | <div class="td-email"> | |
42 | ${_('No additional emails specified')} |
|
42 | ${_('No additional emails specified')} | |
43 | </div> |
|
43 | </div> | |
44 | </td> |
|
44 | </td> | |
45 | </tr> |
|
45 | </tr> | |
46 | %endif |
|
46 | %endif | |
47 | </table> |
|
47 | </table> | |
48 | </div> |
|
48 | </div> | |
49 |
|
49 | |||
50 | <div> |
|
50 | <div> | |
51 | ${h.secure_form(h.route_path('my_account_emails_add'), method='POST')} |
|
51 | ${h.secure_form(h.route_path('my_account_emails_add'), method='POST', request=request)} | |
52 | <div class="form"> |
|
52 | <div class="form"> | |
53 | <!-- fields --> |
|
53 | <!-- fields --> | |
54 | <div class="fields"> |
|
54 | <div class="fields"> | |
55 | <div class="field"> |
|
55 | <div class="field"> | |
56 | <div class="label"> |
|
56 | <div class="label"> | |
57 | <label for="new_email">${_('New email address')}:</label> |
|
57 | <label for="new_email">${_('New email address')}:</label> | |
58 | </div> |
|
58 | </div> | |
59 | <div class="input"> |
|
59 | <div class="input"> | |
60 | ${h.text('new_email', class_='medium')} |
|
60 | ${h.text('new_email', class_='medium')} | |
61 | </div> |
|
61 | </div> | |
62 | </div> |
|
62 | </div> | |
63 | <div class="buttons"> |
|
63 | <div class="buttons"> | |
64 | ${h.submit('save',_('Add'),class_="btn")} |
|
64 | ${h.submit('save',_('Add'),class_="btn")} | |
65 | ${h.reset('reset',_('Reset'),class_="btn")} |
|
65 | ${h.reset('reset',_('Reset'),class_="btn")} | |
66 | </div> |
|
66 | </div> | |
67 | </div> |
|
67 | </div> | |
68 | </div> |
|
68 | </div> | |
69 | ${h.end_form()} |
|
69 | ${h.end_form()} | |
70 | </div> |
|
70 | </div> | |
71 | </div> |
|
71 | </div> | |
72 | </div> |
|
72 | </div> |
@@ -1,113 +1,113 b'' | |||||
1 | <%namespace name="base" file="/base/base.mako"/> |
|
1 | <%namespace name="base" file="/base/base.mako"/> | |
2 | <div class="panel panel-default user-profile"> |
|
2 | <div class="panel panel-default user-profile"> | |
3 | <div class="panel-heading"> |
|
3 | <div class="panel-heading"> | |
4 | <h3 class="panel-title">${_('My Profile')}</h3> |
|
4 | <h3 class="panel-title">${_('My Profile')}</h3> | |
5 | <a href="${h.route_path('my_account_profile')}" class="panel-edit">Close</a> |
|
5 | <a href="${h.route_path('my_account_profile')}" class="panel-edit">Close</a> | |
6 | </div> |
|
6 | </div> | |
7 |
|
7 | |||
8 | <div class="panel-body"> |
|
8 | <div class="panel-body"> | |
9 | ${h.secure_form(h.route_path('my_account_update'), class_='form', method='POST')} |
|
9 | ${h.secure_form(h.route_path('my_account_update'), class_='form', method='POST', request=request)} | |
10 | <% readonly = None %> |
|
10 | <% readonly = None %> | |
11 | <% disabled = "" %> |
|
11 | <% disabled = "" %> | |
12 |
|
12 | |||
13 | % if c.extern_type != 'rhodecode': |
|
13 | % if c.extern_type != 'rhodecode': | |
14 | <% readonly = "readonly" %> |
|
14 | <% readonly = "readonly" %> | |
15 | <% disabled = "disabled" %> |
|
15 | <% disabled = "disabled" %> | |
16 | <div class="infoform"> |
|
16 | <div class="infoform"> | |
17 | <div class="fields"> |
|
17 | <div class="fields"> | |
18 | <p>${_('Your user account details are managed by an external source. Details cannot be managed here.')} |
|
18 | <p>${_('Your user account details are managed by an external source. Details cannot be managed here.')} | |
19 | <br/>${_('Source type')}: <strong>${c.extern_type}</strong> |
|
19 | <br/>${_('Source type')}: <strong>${c.extern_type}</strong> | |
20 | </p> |
|
20 | </p> | |
21 |
|
21 | |||
22 | <div class="field"> |
|
22 | <div class="field"> | |
23 | <div class="label"> |
|
23 | <div class="label"> | |
24 | <label for="username">${_('Username')}:</label> |
|
24 | <label for="username">${_('Username')}:</label> | |
25 | </div> |
|
25 | </div> | |
26 | <div class="input"> |
|
26 | <div class="input"> | |
27 | ${h.text('username', class_='input-valuedisplay', readonly=readonly)} |
|
27 | ${h.text('username', class_='input-valuedisplay', readonly=readonly)} | |
28 | </div> |
|
28 | </div> | |
29 | </div> |
|
29 | </div> | |
30 |
|
30 | |||
31 | <div class="field"> |
|
31 | <div class="field"> | |
32 | <div class="label"> |
|
32 | <div class="label"> | |
33 | <label for="name">${_('First Name')}:</label> |
|
33 | <label for="name">${_('First Name')}:</label> | |
34 | </div> |
|
34 | </div> | |
35 | <div class="input"> |
|
35 | <div class="input"> | |
36 | ${h.text('firstname', class_='input-valuedisplay', readonly=readonly)} |
|
36 | ${h.text('firstname', class_='input-valuedisplay', readonly=readonly)} | |
37 | </div> |
|
37 | </div> | |
38 | </div> |
|
38 | </div> | |
39 |
|
39 | |||
40 | <div class="field"> |
|
40 | <div class="field"> | |
41 | <div class="label"> |
|
41 | <div class="label"> | |
42 | <label for="lastname">${_('Last Name')}:</label> |
|
42 | <label for="lastname">${_('Last Name')}:</label> | |
43 | </div> |
|
43 | </div> | |
44 | <div class="input-valuedisplay"> |
|
44 | <div class="input-valuedisplay"> | |
45 | ${h.text('lastname', class_='input-valuedisplay', readonly=readonly)} |
|
45 | ${h.text('lastname', class_='input-valuedisplay', readonly=readonly)} | |
46 | </div> |
|
46 | </div> | |
47 | </div> |
|
47 | </div> | |
48 | </div> |
|
48 | </div> | |
49 | </div> |
|
49 | </div> | |
50 | % else: |
|
50 | % else: | |
51 | <div class="form"> |
|
51 | <div class="form"> | |
52 | <div class="fields"> |
|
52 | <div class="fields"> | |
53 | <div class="field"> |
|
53 | <div class="field"> | |
54 | <div class="label photo"> |
|
54 | <div class="label photo"> | |
55 | ${_('Photo')}: |
|
55 | ${_('Photo')}: | |
56 | </div> |
|
56 | </div> | |
57 | <div class="input profile"> |
|
57 | <div class="input profile"> | |
58 | %if c.visual.use_gravatar: |
|
58 | %if c.visual.use_gravatar: | |
59 | ${base.gravatar(c.user.email, 100)} |
|
59 | ${base.gravatar(c.user.email, 100)} | |
60 | <p class="help-block">${_('Change your avatar at')} <a href="http://gravatar.com">gravatar.com</a>.</p> |
|
60 | <p class="help-block">${_('Change your avatar at')} <a href="http://gravatar.com">gravatar.com</a>.</p> | |
61 | %else: |
|
61 | %else: | |
62 | ${base.gravatar(c.user.email, 20)} |
|
62 | ${base.gravatar(c.user.email, 20)} | |
63 | ${_('Avatars are disabled')} |
|
63 | ${_('Avatars are disabled')} | |
64 | %endif |
|
64 | %endif | |
65 | </div> |
|
65 | </div> | |
66 | </div> |
|
66 | </div> | |
67 | <div class="field"> |
|
67 | <div class="field"> | |
68 | <div class="label"> |
|
68 | <div class="label"> | |
69 | <label for="username">${_('Username')}:</label> |
|
69 | <label for="username">${_('Username')}:</label> | |
70 | </div> |
|
70 | </div> | |
71 | <div class="input"> |
|
71 | <div class="input"> | |
72 | ${h.text('username', class_='medium%s' % disabled, readonly=readonly)} |
|
72 | ${h.text('username', class_='medium%s' % disabled, readonly=readonly)} | |
73 | ${h.hidden('extern_name', c.extern_name)} |
|
73 | ${h.hidden('extern_name', c.extern_name)} | |
74 | ${h.hidden('extern_type', c.extern_type)} |
|
74 | ${h.hidden('extern_type', c.extern_type)} | |
75 | </div> |
|
75 | </div> | |
76 | </div> |
|
76 | </div> | |
77 | <div class="field"> |
|
77 | <div class="field"> | |
78 | <div class="label"> |
|
78 | <div class="label"> | |
79 | <label for="name">${_('First Name')}:</label> |
|
79 | <label for="name">${_('First Name')}:</label> | |
80 | </div> |
|
80 | </div> | |
81 | <div class="input"> |
|
81 | <div class="input"> | |
82 | ${h.text('firstname', class_="medium")} |
|
82 | ${h.text('firstname', class_="medium")} | |
83 | </div> |
|
83 | </div> | |
84 | </div> |
|
84 | </div> | |
85 |
|
85 | |||
86 | <div class="field"> |
|
86 | <div class="field"> | |
87 | <div class="label"> |
|
87 | <div class="label"> | |
88 | <label for="lastname">${_('Last Name')}:</label> |
|
88 | <label for="lastname">${_('Last Name')}:</label> | |
89 | </div> |
|
89 | </div> | |
90 | <div class="input"> |
|
90 | <div class="input"> | |
91 | ${h.text('lastname', class_="medium")} |
|
91 | ${h.text('lastname', class_="medium")} | |
92 | </div> |
|
92 | </div> | |
93 | </div> |
|
93 | </div> | |
94 |
|
94 | |||
95 | <div class="field"> |
|
95 | <div class="field"> | |
96 | <div class="label"> |
|
96 | <div class="label"> | |
97 | <label for="email">${_('Email')}:</label> |
|
97 | <label for="email">${_('Email')}:</label> | |
98 | </div> |
|
98 | </div> | |
99 | <div class="input"> |
|
99 | <div class="input"> | |
100 | ## we should be able to edit email ! |
|
100 | ## we should be able to edit email ! | |
101 | ${h.text('email', class_="medium")} |
|
101 | ${h.text('email', class_="medium")} | |
102 | </div> |
|
102 | </div> | |
103 | </div> |
|
103 | </div> | |
104 |
|
104 | |||
105 | <div class="buttons"> |
|
105 | <div class="buttons"> | |
106 | ${h.submit('save', _('Save'), class_="btn")} |
|
106 | ${h.submit('save', _('Save'), class_="btn")} | |
107 | ${h.reset('reset', _('Reset'), class_="btn")} |
|
107 | ${h.reset('reset', _('Reset'), class_="btn")} | |
108 | </div> |
|
108 | </div> | |
109 | </div> |
|
109 | </div> | |
110 | </div> |
|
110 | </div> | |
111 | % endif |
|
111 | % endif | |
112 | </div> |
|
112 | </div> | |
113 | </div> No newline at end of file |
|
113 | </div> |
@@ -1,67 +1,67 b'' | |||||
1 |
|
1 | |||
2 |
|
2 | |||
3 | <div class="panel panel-default"> |
|
3 | <div class="panel panel-default"> | |
4 | <div class="panel-heading"> |
|
4 | <div class="panel-heading"> | |
5 | <h3 class="panel-title">${_('Default IP Whitelist For All Users')}</h3> |
|
5 | <h3 class="panel-title">${_('Default IP Whitelist For All Users')}</h3> | |
6 | </div> |
|
6 | </div> | |
7 | <div class="panel-body"> |
|
7 | <div class="panel-body"> | |
8 | <div class="ips_wrap"> |
|
8 | <div class="ips_wrap"> | |
9 | <table class="rctable ip-whitelist"> |
|
9 | <table class="rctable ip-whitelist"> | |
10 | <tr> |
|
10 | <tr> | |
11 | <th>IP Address</th> |
|
11 | <th>IP Address</th> | |
12 | <th>IP Range</th> |
|
12 | <th>IP Range</th> | |
13 | <th>Description</th> |
|
13 | <th>Description</th> | |
14 | <th></th> |
|
14 | <th></th> | |
15 | </tr> |
|
15 | </tr> | |
16 | %if c.user_ip_map: |
|
16 | %if c.user_ip_map: | |
17 | %for ip in c.user_ip_map: |
|
17 | %for ip in c.user_ip_map: | |
18 | <tr> |
|
18 | <tr> | |
19 | <td class="td-ip"><div class="ip">${ip.ip_addr}</div></td> |
|
19 | <td class="td-ip"><div class="ip">${ip.ip_addr}</div></td> | |
20 | <td class="td-iprange"><div class="ip">${h.ip_range(ip.ip_addr)}</div></td> |
|
20 | <td class="td-iprange"><div class="ip">${h.ip_range(ip.ip_addr)}</div></td> | |
21 | <td class="td-description"><div class="ip">${ip.description}</div></td> |
|
21 | <td class="td-description"><div class="ip">${ip.description}</div></td> | |
22 | <td class="td-action"> |
|
22 | <td class="td-action"> | |
23 | ${h.secure_form(h.route_path('edit_user_ips_delete', user_id=c.user.user_id), method='POST')} |
|
23 | ${h.secure_form(h.route_path('edit_user_ips_delete', user_id=c.user.user_id), method='POST', request=request)} | |
24 | ${h.hidden('del_ip_id',ip.ip_id)} |
|
24 | ${h.hidden('del_ip_id',ip.ip_id)} | |
25 | ${h.hidden('default_user', 'True')} |
|
25 | ${h.hidden('default_user', 'True')} | |
26 | ${h.submit('remove_',_('Delete'),id="remove_ip_%s" % ip.ip_id, |
|
26 | ${h.submit('remove_',_('Delete'),id="remove_ip_%s" % ip.ip_id, | |
27 | class_="btn btn-link btn-danger", onclick="return confirm('"+_('Confirm to delete this ip: %s') % ip.ip_addr+"');")} |
|
27 | class_="btn btn-link btn-danger", onclick="return confirm('"+_('Confirm to delete this ip: %s') % ip.ip_addr+"');")} | |
28 | ${h.end_form()} |
|
28 | ${h.end_form()} | |
29 | </td> |
|
29 | </td> | |
30 | </tr> |
|
30 | </tr> | |
31 | %endfor |
|
31 | %endfor | |
32 | %else: |
|
32 | %else: | |
33 | <tr> |
|
33 | <tr> | |
34 | <td class="ip">${_('All IP addresses are allowed')}</td> |
|
34 | <td class="ip">${_('All IP addresses are allowed')}</td> | |
35 | <td></td> |
|
35 | <td></td> | |
36 | <td></td> |
|
36 | <td></td> | |
37 | <td></td> |
|
37 | <td></td> | |
38 | </tr> |
|
38 | </tr> | |
39 | %endif |
|
39 | %endif | |
40 | </table> |
|
40 | </table> | |
41 | </div> |
|
41 | </div> | |
42 |
|
42 | |||
43 | ${h.secure_form(h.route_path('edit_user_ips_add', user_id=c.user.user_id), method='POST')} |
|
43 | ${h.secure_form(h.route_path('edit_user_ips_add', user_id=c.user.user_id), method='POST', request=request)} | |
44 | <div class="form"> |
|
44 | <div class="form"> | |
45 | <!-- fields --> |
|
45 | <!-- fields --> | |
46 | <div class="fields"> |
|
46 | <div class="fields"> | |
47 | <div class="field"> |
|
47 | <div class="field"> | |
48 | <div class="label"> |
|
48 | <div class="label"> | |
49 | <label for="new_ip">${_('New IP Address')}:</label> |
|
49 | <label for="new_ip">${_('New IP Address')}:</label> | |
50 | </div> |
|
50 | </div> | |
51 | <div class="input"> |
|
51 | <div class="input"> | |
52 | ${h.hidden('default_user', 'True')} |
|
52 | ${h.hidden('default_user', 'True')} | |
53 | ${h.text('new_ip')} ${h.text('description', placeholder=_('Description...'))} |
|
53 | ${h.text('new_ip')} ${h.text('description', placeholder=_('Description...'))} | |
54 | <span class="help-block">${_('Enter a comma separated list of IP Addresses like 127.0.0.1,\n' |
|
54 | <span class="help-block">${_('Enter a comma separated list of IP Addresses like 127.0.0.1,\n' | |
55 | 'or use an IP Address with a mask 127.0.0.1/24, to create a network range.\n' |
|
55 | 'or use an IP Address with a mask 127.0.0.1/24, to create a network range.\n' | |
56 | 'To specify multiple addresses in a range, use the 127.0.0.1-127.0.0.10 syntax')}</span> |
|
56 | 'To specify multiple addresses in a range, use the 127.0.0.1-127.0.0.10 syntax')}</span> | |
57 | </div> |
|
57 | </div> | |
58 | </div> |
|
58 | </div> | |
59 | <div class="buttons"> |
|
59 | <div class="buttons"> | |
60 | ${h.submit('save',_('Add'),class_="btn")} |
|
60 | ${h.submit('save',_('Add'),class_="btn")} | |
61 | ${h.reset('reset',_('Reset'),class_="btn")} |
|
61 | ${h.reset('reset',_('Reset'),class_="btn")} | |
62 | </div> |
|
62 | </div> | |
63 | </div> |
|
63 | </div> | |
64 | </div> |
|
64 | </div> | |
65 | ${h.end_form()} |
|
65 | ${h.end_form()} | |
66 | </div> |
|
66 | </div> | |
67 | </div> |
|
67 | </div> |
@@ -1,210 +1,210 b'' | |||||
1 | <%namespace name="base" file="/base/base.mako"/> |
|
1 | <%namespace name="base" file="/base/base.mako"/> | |
2 |
|
2 | |||
3 | <% |
|
3 | <% | |
4 | elems = [ |
|
4 | elems = [ | |
5 | (_('Owner'), lambda:base.gravatar_with_user(c.repo_info.user.email), '', ''), |
|
5 | (_('Owner'), lambda:base.gravatar_with_user(c.repo_info.user.email), '', ''), | |
6 | (_('Created on'), h.format_date(c.repo_info.created_on), '', ''), |
|
6 | (_('Created on'), h.format_date(c.repo_info.created_on), '', ''), | |
7 | (_('Updated on'), h.format_date(c.repo_info.updated_on), '', ''), |
|
7 | (_('Updated on'), h.format_date(c.repo_info.updated_on), '', ''), | |
8 | (_('Cached Commit id'), lambda: h.link_to(c.repo_info.changeset_cache.get('short_id'), h.url('changeset_home',repo_name=c.repo_name,revision=c.repo_info.changeset_cache.get('raw_id'))), '', ''), |
|
8 | (_('Cached Commit id'), lambda: h.link_to(c.repo_info.changeset_cache.get('short_id'), h.url('changeset_home',repo_name=c.repo_name,revision=c.repo_info.changeset_cache.get('raw_id'))), '', ''), | |
9 | ] |
|
9 | ] | |
10 | %> |
|
10 | %> | |
11 |
|
11 | |||
12 | <div class="panel panel-default"> |
|
12 | <div class="panel panel-default"> | |
13 | <div class="panel-heading" id="advanced-info" > |
|
13 | <div class="panel-heading" id="advanced-info" > | |
14 | <h3 class="panel-title">${_('Repository: %s') % c.repo_info.repo_name} <a class="permalink" href="#advanced-info"> ΒΆ</a></h3> |
|
14 | <h3 class="panel-title">${_('Repository: %s') % c.repo_info.repo_name} <a class="permalink" href="#advanced-info"> ΒΆ</a></h3> | |
15 | </div> |
|
15 | </div> | |
16 | <div class="panel-body"> |
|
16 | <div class="panel-body"> | |
17 | ${base.dt_info_panel(elems)} |
|
17 | ${base.dt_info_panel(elems)} | |
18 | </div> |
|
18 | </div> | |
19 | </div> |
|
19 | </div> | |
20 |
|
20 | |||
21 |
|
21 | |||
22 | <div class="panel panel-default"> |
|
22 | <div class="panel panel-default"> | |
23 | <div class="panel-heading" id="advanced-fork"> |
|
23 | <div class="panel-heading" id="advanced-fork"> | |
24 | <h3 class="panel-title">${_('Fork Reference')} <a class="permalink" href="#advanced-fork"> ΒΆ</a></h3> |
|
24 | <h3 class="panel-title">${_('Fork Reference')} <a class="permalink" href="#advanced-fork"> ΒΆ</a></h3> | |
25 | </div> |
|
25 | </div> | |
26 | <div class="panel-body"> |
|
26 | <div class="panel-body"> | |
27 | ${h.secure_form(h.route_path('edit_repo_advanced_fork', repo_name=c.repo_info.repo_name), method='POST')} |
|
27 | ${h.secure_form(h.route_path('edit_repo_advanced_fork', repo_name=c.repo_info.repo_name), method='POST', request=request)} | |
28 |
|
28 | |||
29 | % if c.repo_info.fork: |
|
29 | % if c.repo_info.fork: | |
30 | <div class="panel-body-title-text">${h.literal(_('This repository is a fork of %(repo_link)s') % {'repo_link': h.link_to_if(c.has_origin_repo_read_perm,c.repo_info.fork.repo_name, h.route_path('repo_summary', repo_name=c.repo_info.fork.repo_name))})} |
|
30 | <div class="panel-body-title-text">${h.literal(_('This repository is a fork of %(repo_link)s') % {'repo_link': h.link_to_if(c.has_origin_repo_read_perm,c.repo_info.fork.repo_name, h.route_path('repo_summary', repo_name=c.repo_info.fork.repo_name))})} | |
31 | | <button class="btn btn-link btn-danger" type="submit">Remove fork reference</button></div> |
|
31 | | <button class="btn btn-link btn-danger" type="submit">Remove fork reference</button></div> | |
32 | % endif |
|
32 | % endif | |
33 |
|
33 | |||
34 | <div class="field"> |
|
34 | <div class="field"> | |
35 | ${h.hidden('id_fork_of')} |
|
35 | ${h.hidden('id_fork_of')} | |
36 | ${h.submit('set_as_fork_%s' % c.repo_info.repo_name,_('Set'),class_="btn btn-small",)} |
|
36 | ${h.submit('set_as_fork_%s' % c.repo_info.repo_name,_('Set'),class_="btn btn-small",)} | |
37 | </div> |
|
37 | </div> | |
38 | <div class="field"> |
|
38 | <div class="field"> | |
39 | <span class="help-block">${_('Manually set this repository as a fork of another from the list')}</span> |
|
39 | <span class="help-block">${_('Manually set this repository as a fork of another from the list')}</span> | |
40 | </div> |
|
40 | </div> | |
41 | ${h.end_form()} |
|
41 | ${h.end_form()} | |
42 | </div> |
|
42 | </div> | |
43 | </div> |
|
43 | </div> | |
44 |
|
44 | |||
45 |
|
45 | |||
46 | <div class="panel panel-default"> |
|
46 | <div class="panel panel-default"> | |
47 | <div class="panel-heading" id="advanced-journal"> |
|
47 | <div class="panel-heading" id="advanced-journal"> | |
48 | <h3 class="panel-title">${_('Public Journal Visibility')} <a class="permalink" href="#advanced-journal"> ΒΆ</a></h3> |
|
48 | <h3 class="panel-title">${_('Public Journal Visibility')} <a class="permalink" href="#advanced-journal"> ΒΆ</a></h3> | |
49 | </div> |
|
49 | </div> | |
50 | <div class="panel-body"> |
|
50 | <div class="panel-body"> | |
51 | ${h.secure_form(h.route_path('edit_repo_advanced_journal', repo_name=c.repo_info.repo_name), method='POST')} |
|
51 | ${h.secure_form(h.route_path('edit_repo_advanced_journal', repo_name=c.repo_info.repo_name), method='POST', request=request)} | |
52 | <div class="field"> |
|
52 | <div class="field"> | |
53 | %if c.in_public_journal: |
|
53 | %if c.in_public_journal: | |
54 | <button class="btn btn-small" type="submit"> |
|
54 | <button class="btn btn-small" type="submit"> | |
55 | ${_('Remove from Public Journal')} |
|
55 | ${_('Remove from Public Journal')} | |
56 | </button> |
|
56 | </button> | |
57 | %else: |
|
57 | %else: | |
58 | <button class="btn btn-small" type="submit"> |
|
58 | <button class="btn btn-small" type="submit"> | |
59 | ${_('Add to Public Journal')} |
|
59 | ${_('Add to Public Journal')} | |
60 | </button> |
|
60 | </button> | |
61 | %endif |
|
61 | %endif | |
62 | </div> |
|
62 | </div> | |
63 | <div class="field" > |
|
63 | <div class="field" > | |
64 | <span class="help-block">${_('All actions made on this repository will be visible to everyone following the public journal.')}</span> |
|
64 | <span class="help-block">${_('All actions made on this repository will be visible to everyone following the public journal.')}</span> | |
65 | </div> |
|
65 | </div> | |
66 | ${h.end_form()} |
|
66 | ${h.end_form()} | |
67 | </div> |
|
67 | </div> | |
68 | </div> |
|
68 | </div> | |
69 |
|
69 | |||
70 |
|
70 | |||
71 | <div class="panel panel-default"> |
|
71 | <div class="panel panel-default"> | |
72 | <div class="panel-heading" id="advanced-locking"> |
|
72 | <div class="panel-heading" id="advanced-locking"> | |
73 | <h3 class="panel-title">${_('Locking state')} <a class="permalink" href="#advanced-locking"> ΒΆ</a></h3> |
|
73 | <h3 class="panel-title">${_('Locking state')} <a class="permalink" href="#advanced-locking"> ΒΆ</a></h3> | |
74 | </div> |
|
74 | </div> | |
75 | <div class="panel-body"> |
|
75 | <div class="panel-body"> | |
76 | ${h.secure_form(h.route_path('edit_repo_advanced_locking', repo_name=c.repo_info.repo_name), method='POST')} |
|
76 | ${h.secure_form(h.route_path('edit_repo_advanced_locking', repo_name=c.repo_info.repo_name), method='POST', request=request)} | |
77 |
|
77 | |||
78 | %if c.repo_info.locked[0]: |
|
78 | %if c.repo_info.locked[0]: | |
79 | <div class="panel-body-title-text">${'Locked by %s on %s. Lock reason: %s' % (h.person_by_id(c.repo_info.locked[0]), |
|
79 | <div class="panel-body-title-text">${'Locked by %s on %s. Lock reason: %s' % (h.person_by_id(c.repo_info.locked[0]), | |
80 | h.format_date(h. time_to_datetime(c.repo_info.locked[1])), c.repo_info.locked[2])}</div> |
|
80 | h.format_date(h. time_to_datetime(c.repo_info.locked[1])), c.repo_info.locked[2])}</div> | |
81 | %else: |
|
81 | %else: | |
82 | <div class="panel-body-title-text">${_('This Repository is not currently locked.')}</div> |
|
82 | <div class="panel-body-title-text">${_('This Repository is not currently locked.')}</div> | |
83 | %endif |
|
83 | %endif | |
84 |
|
84 | |||
85 | <div class="field" > |
|
85 | <div class="field" > | |
86 | %if c.repo_info.locked[0]: |
|
86 | %if c.repo_info.locked[0]: | |
87 | ${h.hidden('set_unlock', '1')} |
|
87 | ${h.hidden('set_unlock', '1')} | |
88 | <button class="btn btn-small" type="submit" |
|
88 | <button class="btn btn-small" type="submit" | |
89 | onclick="return confirm('${_('Confirm to unlock repository.')}');"> |
|
89 | onclick="return confirm('${_('Confirm to unlock repository.')}');"> | |
90 | <i class="icon-unlock"></i> |
|
90 | <i class="icon-unlock"></i> | |
91 | ${_('Unlock repository')} |
|
91 | ${_('Unlock repository')} | |
92 | </button> |
|
92 | </button> | |
93 | %else: |
|
93 | %else: | |
94 | ${h.hidden('set_lock', '1')} |
|
94 | ${h.hidden('set_lock', '1')} | |
95 | <button class="btn btn-small" type="submit" |
|
95 | <button class="btn btn-small" type="submit" | |
96 | onclick="return confirm('${_('Confirm to lock repository.')}');"> |
|
96 | onclick="return confirm('${_('Confirm to lock repository.')}');"> | |
97 | <i class="icon-lock"></i> |
|
97 | <i class="icon-lock"></i> | |
98 | ${_('Lock Repository')} |
|
98 | ${_('Lock Repository')} | |
99 | </button> |
|
99 | </button> | |
100 | %endif |
|
100 | %endif | |
101 | </div> |
|
101 | </div> | |
102 | <div class="field" > |
|
102 | <div class="field" > | |
103 | <span class="help-block"> |
|
103 | <span class="help-block"> | |
104 | ${_('Force repository locking. This only works when anonymous access is disabled. Pulling from the repository locks the repository to that user until the same user pushes to that repository again.')} |
|
104 | ${_('Force repository locking. This only works when anonymous access is disabled. Pulling from the repository locks the repository to that user until the same user pushes to that repository again.')} | |
105 | </span> |
|
105 | </span> | |
106 | </div> |
|
106 | </div> | |
107 | ${h.end_form()} |
|
107 | ${h.end_form()} | |
108 | </div> |
|
108 | </div> | |
109 | </div> |
|
109 | </div> | |
110 |
|
110 | |||
111 | <div class="panel panel-danger"> |
|
111 | <div class="panel panel-danger"> | |
112 | <div class="panel-heading" id="advanced-delete"> |
|
112 | <div class="panel-heading" id="advanced-delete"> | |
113 | <h3 class="panel-title">${_('Delete repository')} <a class="permalink" href="#advanced-delete"> ΒΆ</a></h3> |
|
113 | <h3 class="panel-title">${_('Delete repository')} <a class="permalink" href="#advanced-delete"> ΒΆ</a></h3> | |
114 | </div> |
|
114 | </div> | |
115 | <div class="panel-body"> |
|
115 | <div class="panel-body"> | |
116 | ${h.secure_form(h.route_path('edit_repo_advanced_delete', repo_name=c.repo_name), method='POST')} |
|
116 | ${h.secure_form(h.route_path('edit_repo_advanced_delete', repo_name=c.repo_name), method='POST', request=request)} | |
117 | <table class="display"> |
|
117 | <table class="display"> | |
118 | <tr> |
|
118 | <tr> | |
119 | <td> |
|
119 | <td> | |
120 | ${_ungettext('This repository has %s fork.', 'This repository has %s forks.', c.repo_info.forks.count()) % c.repo_info.forks.count()} |
|
120 | ${_ungettext('This repository has %s fork.', 'This repository has %s forks.', c.repo_info.forks.count()) % c.repo_info.forks.count()} | |
121 | </td> |
|
121 | </td> | |
122 | <td> |
|
122 | <td> | |
123 | %if c.repo_info.forks.count(): |
|
123 | %if c.repo_info.forks.count(): | |
124 | <input type="radio" name="forks" value="detach_forks" checked="checked"/> <label for="forks">${_('Detach forks')}</label> |
|
124 | <input type="radio" name="forks" value="detach_forks" checked="checked"/> <label for="forks">${_('Detach forks')}</label> | |
125 | %endif |
|
125 | %endif | |
126 | </td> |
|
126 | </td> | |
127 | <td> |
|
127 | <td> | |
128 | %if c.repo_info.forks.count(): |
|
128 | %if c.repo_info.forks.count(): | |
129 | <input type="radio" name="forks" value="delete_forks"/> <label for="forks">${_('Delete forks')}</label> |
|
129 | <input type="radio" name="forks" value="delete_forks"/> <label for="forks">${_('Delete forks')}</label> | |
130 | %endif |
|
130 | %endif | |
131 | </td> |
|
131 | </td> | |
132 | </tr> |
|
132 | </tr> | |
133 | </table> |
|
133 | </table> | |
134 | <div style="margin: 0 0 20px 0" class="fake-space"></div> |
|
134 | <div style="margin: 0 0 20px 0" class="fake-space"></div> | |
135 |
|
135 | |||
136 | <div class="field"> |
|
136 | <div class="field"> | |
137 | <button class="btn btn-small btn-danger" type="submit" |
|
137 | <button class="btn btn-small btn-danger" type="submit" | |
138 | onclick="return confirm('${_('Confirm to delete this repository: %s') % c.repo_name}');"> |
|
138 | onclick="return confirm('${_('Confirm to delete this repository: %s') % c.repo_name}');"> | |
139 | <i class="icon-remove-sign"></i> |
|
139 | <i class="icon-remove-sign"></i> | |
140 | ${_('Delete This Repository')} |
|
140 | ${_('Delete This Repository')} | |
141 | </button> |
|
141 | </button> | |
142 | </div> |
|
142 | </div> | |
143 | <div class="field"> |
|
143 | <div class="field"> | |
144 | <span class="help-block"> |
|
144 | <span class="help-block"> | |
145 | ${_('This repository will be renamed in a special way in order to make it inaccessible to RhodeCode Enterprise and its VCS systems. If you need to fully delete it from the file system, please do it manually, or with rhodecode-cleanup-repos command available in rhodecode-tools.')} |
|
145 | ${_('This repository will be renamed in a special way in order to make it inaccessible to RhodeCode Enterprise and its VCS systems. If you need to fully delete it from the file system, please do it manually, or with rhodecode-cleanup-repos command available in rhodecode-tools.')} | |
146 | </span> |
|
146 | </span> | |
147 | </div> |
|
147 | </div> | |
148 |
|
148 | |||
149 | ${h.end_form()} |
|
149 | ${h.end_form()} | |
150 | </div> |
|
150 | </div> | |
151 | </div> |
|
151 | </div> | |
152 |
|
152 | |||
153 |
|
153 | |||
154 | <script> |
|
154 | <script> | |
155 |
|
155 | |||
156 | var currentRepoId = ${c.repo_info.repo_id}; |
|
156 | var currentRepoId = ${c.repo_info.repo_id}; | |
157 |
|
157 | |||
158 | var repoTypeFilter = function(data) { |
|
158 | var repoTypeFilter = function(data) { | |
159 | var results = []; |
|
159 | var results = []; | |
160 |
|
160 | |||
161 | if (!data.results[0]) { |
|
161 | if (!data.results[0]) { | |
162 | return data |
|
162 | return data | |
163 | } |
|
163 | } | |
164 |
|
164 | |||
165 | $.each(data.results[0].children, function() { |
|
165 | $.each(data.results[0].children, function() { | |
166 | // filter out the SAME repo, it cannot be used as fork of itself |
|
166 | // filter out the SAME repo, it cannot be used as fork of itself | |
167 | if (this.obj.repo_id != currentRepoId) { |
|
167 | if (this.obj.repo_id != currentRepoId) { | |
168 | this.id = this.obj.repo_id; |
|
168 | this.id = this.obj.repo_id; | |
169 | results.push(this) |
|
169 | results.push(this) | |
170 | } |
|
170 | } | |
171 | }); |
|
171 | }); | |
172 | data.results[0].children = results; |
|
172 | data.results[0].children = results; | |
173 | return data; |
|
173 | return data; | |
174 | }; |
|
174 | }; | |
175 |
|
175 | |||
176 | $("#id_fork_of").select2({ |
|
176 | $("#id_fork_of").select2({ | |
177 | cachedDataSource: {}, |
|
177 | cachedDataSource: {}, | |
178 | minimumInputLength: 2, |
|
178 | minimumInputLength: 2, | |
179 | placeholder: "${_('Change repository') if c.repo_info.fork else _('Pick repository')}", |
|
179 | placeholder: "${_('Change repository') if c.repo_info.fork else _('Pick repository')}", | |
180 | dropdownAutoWidth: true, |
|
180 | dropdownAutoWidth: true, | |
181 | containerCssClass: "drop-menu", |
|
181 | containerCssClass: "drop-menu", | |
182 | dropdownCssClass: "drop-menu-dropdown", |
|
182 | dropdownCssClass: "drop-menu-dropdown", | |
183 | formatResult: formatResult, |
|
183 | formatResult: formatResult, | |
184 | query: $.debounce(250, function(query){ |
|
184 | query: $.debounce(250, function(query){ | |
185 | self = this; |
|
185 | self = this; | |
186 | var cacheKey = query.term; |
|
186 | var cacheKey = query.term; | |
187 | var cachedData = self.cachedDataSource[cacheKey]; |
|
187 | var cachedData = self.cachedDataSource[cacheKey]; | |
188 |
|
188 | |||
189 | if (cachedData) { |
|
189 | if (cachedData) { | |
190 | query.callback({results: cachedData.results}); |
|
190 | query.callback({results: cachedData.results}); | |
191 | } else { |
|
191 | } else { | |
192 | $.ajax({ |
|
192 | $.ajax({ | |
193 | url: pyroutes.url('repo_list_data'), |
|
193 | url: pyroutes.url('repo_list_data'), | |
194 | data: {'query': query.term, repo_type: '${c.repo_info.repo_type}'}, |
|
194 | data: {'query': query.term, repo_type: '${c.repo_info.repo_type}'}, | |
195 | dataType: 'json', |
|
195 | dataType: 'json', | |
196 | type: 'GET', |
|
196 | type: 'GET', | |
197 | success: function(data) { |
|
197 | success: function(data) { | |
198 | data = repoTypeFilter(data); |
|
198 | data = repoTypeFilter(data); | |
199 | self.cachedDataSource[cacheKey] = data; |
|
199 | self.cachedDataSource[cacheKey] = data; | |
200 | query.callback({results: data.results}); |
|
200 | query.callback({results: data.results}); | |
201 | }, |
|
201 | }, | |
202 | error: function(data, textStatus, errorThrown) { |
|
202 | error: function(data, textStatus, errorThrown) { | |
203 | alert("Error while fetching entries.\nError code {0} ({1}).".format(data.status, data.statusText)); |
|
203 | alert("Error while fetching entries.\nError code {0} ({1}).".format(data.status, data.statusText)); | |
204 | } |
|
204 | } | |
205 | }) |
|
205 | }) | |
206 | } |
|
206 | } | |
207 | }) |
|
207 | }) | |
208 | }); |
|
208 | }); | |
209 | </script> |
|
209 | </script> | |
210 |
|
210 |
@@ -1,53 +1,53 b'' | |||||
1 | <div class="panel panel-default"> |
|
1 | <div class="panel panel-default"> | |
2 | <div class="panel-heading"> |
|
2 | <div class="panel-heading"> | |
3 | <h3 class="panel-title">${_('Invalidate Cache for Repository')}</h3> |
|
3 | <h3 class="panel-title">${_('Invalidate Cache for Repository')}</h3> | |
4 | </div> |
|
4 | </div> | |
5 | <div class="panel-body"> |
|
5 | <div class="panel-body"> | |
6 |
|
6 | |||
7 | <h4>${_('Manually invalidate the repository cache. On the next access a repository cache will be recreated.')}</h4> |
|
7 | <h4>${_('Manually invalidate the repository cache. On the next access a repository cache will be recreated.')}</h4> | |
8 |
|
8 | |||
9 | <p> |
|
9 | <p> | |
10 | ${_('Cache purge can be automated by such api call. Can be called periodically in crontab etc.')} |
|
10 | ${_('Cache purge can be automated by such api call. Can be called periodically in crontab etc.')} | |
11 | <br/> |
|
11 | <br/> | |
12 | <code> |
|
12 | <code> | |
13 | ${h.api_call_example(method='invalidate_cache', args={"repoid": c.repo_info.repo_name})} |
|
13 | ${h.api_call_example(method='invalidate_cache', args={"repoid": c.repo_info.repo_name})} | |
14 | </code> |
|
14 | </code> | |
15 | </p> |
|
15 | </p> | |
16 |
|
16 | |||
17 | ${h.secure_form(h.route_path('edit_repo_caches', repo_name=c.repo_name), method='POST')} |
|
17 | ${h.secure_form(h.route_path('edit_repo_caches', repo_name=c.repo_name), method='POST', request=request)} | |
18 | <div class="form"> |
|
18 | <div class="form"> | |
19 | <div class="fields"> |
|
19 | <div class="fields"> | |
20 | ${h.submit('reset_cache_%s' % c.repo_info.repo_name,_('Invalidate repository cache'),class_="btn btn-small",onclick="return confirm('"+_('Confirm to invalidate repository cache')+"');")} |
|
20 | ${h.submit('reset_cache_%s' % c.repo_info.repo_name,_('Invalidate repository cache'),class_="btn btn-small",onclick="return confirm('"+_('Confirm to invalidate repository cache')+"');")} | |
21 | </div> |
|
21 | </div> | |
22 | </div> |
|
22 | </div> | |
23 | ${h.end_form()} |
|
23 | ${h.end_form()} | |
24 |
|
24 | |||
25 | </div> |
|
25 | </div> | |
26 | </div> |
|
26 | </div> | |
27 |
|
27 | |||
28 |
|
28 | |||
29 | <div class="panel panel-default"> |
|
29 | <div class="panel panel-default"> | |
30 | <div class="panel-heading"> |
|
30 | <div class="panel-heading"> | |
31 | <h3 class="panel-title"> |
|
31 | <h3 class="panel-title"> | |
32 | ${(_ungettext('List of repository caches (%(count)s entry)', 'List of repository caches (%(count)s entries)' ,len(c.repo_info.cache_keys)) % {'count': len(c.repo_info.cache_keys)})} |
|
32 | ${(_ungettext('List of repository caches (%(count)s entry)', 'List of repository caches (%(count)s entries)' ,len(c.repo_info.cache_keys)) % {'count': len(c.repo_info.cache_keys)})} | |
33 | </h3> |
|
33 | </h3> | |
34 | </div> |
|
34 | </div> | |
35 | <div class="panel-body"> |
|
35 | <div class="panel-body"> | |
36 | <div class="field" > |
|
36 | <div class="field" > | |
37 | <table class="rctable edit_cache"> |
|
37 | <table class="rctable edit_cache"> | |
38 | <tr> |
|
38 | <tr> | |
39 | <th>${_('Prefix')}</th> |
|
39 | <th>${_('Prefix')}</th> | |
40 | <th>${_('Key')}</th> |
|
40 | <th>${_('Key')}</th> | |
41 | <th>${_('Active')}</th> |
|
41 | <th>${_('Active')}</th> | |
42 | </tr> |
|
42 | </tr> | |
43 | %for cache in c.repo_info.cache_keys: |
|
43 | %for cache in c.repo_info.cache_keys: | |
44 | <tr> |
|
44 | <tr> | |
45 | <td class="td-prefix">${cache.get_prefix() or '-'}</td> |
|
45 | <td class="td-prefix">${cache.get_prefix() or '-'}</td> | |
46 | <td class="td-cachekey">${cache.cache_key}</td> |
|
46 | <td class="td-cachekey">${cache.cache_key}</td> | |
47 | <td class="td-active">${h.bool2icon(cache.cache_active)}</td> |
|
47 | <td class="td-active">${h.bool2icon(cache.cache_active)}</td> | |
48 | </tr> |
|
48 | </tr> | |
49 | %endfor |
|
49 | %endfor | |
50 | </table> |
|
50 | </table> | |
51 | </div> |
|
51 | </div> | |
52 | </div> |
|
52 | </div> | |
53 | </div> |
|
53 | </div> |
@@ -1,123 +1,123 b'' | |||||
1 | <%namespace name="base" file="/base/base.mako"/> |
|
1 | <%namespace name="base" file="/base/base.mako"/> | |
2 |
|
2 | |||
3 | <div class="panel panel-default"> |
|
3 | <div class="panel panel-default"> | |
4 | <div class="panel-heading"> |
|
4 | <div class="panel-heading"> | |
5 | <h3 class="panel-title">${_('Repository Permissions')}</h3> |
|
5 | <h3 class="panel-title">${_('Repository Permissions')}</h3> | |
6 | </div> |
|
6 | </div> | |
7 | <div class="panel-body"> |
|
7 | <div class="panel-body"> | |
8 | ${h.secure_form(h.route_path('edit_repo_perms', repo_name=c.repo_name), method='POST')} |
|
8 | ${h.secure_form(h.route_path('edit_repo_perms', repo_name=c.repo_name), method='POST', request=request)} | |
9 | <table id="permissions_manage" class="rctable permissions"> |
|
9 | <table id="permissions_manage" class="rctable permissions"> | |
10 | <tr> |
|
10 | <tr> | |
11 | <th class="td-radio">${_('None')}</th> |
|
11 | <th class="td-radio">${_('None')}</th> | |
12 | <th class="td-radio">${_('Read')}</th> |
|
12 | <th class="td-radio">${_('Read')}</th> | |
13 | <th class="td-radio">${_('Write')}</th> |
|
13 | <th class="td-radio">${_('Write')}</th> | |
14 | <th class="td-radio">${_('Admin')}</th> |
|
14 | <th class="td-radio">${_('Admin')}</th> | |
15 | <th class="td-owner">${_('User/User Group')}</th> |
|
15 | <th class="td-owner">${_('User/User Group')}</th> | |
16 | <th></th> |
|
16 | <th></th> | |
17 | </tr> |
|
17 | </tr> | |
18 | ## USERS |
|
18 | ## USERS | |
19 | %for _user in c.repo_info.permissions(): |
|
19 | %for _user in c.repo_info.permissions(): | |
20 | %if getattr(_user, 'admin_row', None) or getattr(_user, 'owner_row', None): |
|
20 | %if getattr(_user, 'admin_row', None) or getattr(_user, 'owner_row', None): | |
21 | <tr class="perm_admin_row"> |
|
21 | <tr class="perm_admin_row"> | |
22 | <td class="td-radio">${h.radio('admin_perm_%s' % _user.user_id,'repository.none', disabled="disabled")}</td> |
|
22 | <td class="td-radio">${h.radio('admin_perm_%s' % _user.user_id,'repository.none', disabled="disabled")}</td> | |
23 | <td class="td-radio">${h.radio('admin_perm_%s' % _user.user_id,'repository.read', disabled="disabled")}</td> |
|
23 | <td class="td-radio">${h.radio('admin_perm_%s' % _user.user_id,'repository.read', disabled="disabled")}</td> | |
24 | <td class="td-radio">${h.radio('admin_perm_%s' % _user.user_id,'repository.write', disabled="disabled")}</td> |
|
24 | <td class="td-radio">${h.radio('admin_perm_%s' % _user.user_id,'repository.write', disabled="disabled")}</td> | |
25 | <td class="td-radio">${h.radio('admin_perm_%s' % _user.user_id,'repository.admin', 'repository.admin', disabled="disabled")}</td> |
|
25 | <td class="td-radio">${h.radio('admin_perm_%s' % _user.user_id,'repository.admin', 'repository.admin', disabled="disabled")}</td> | |
26 | <td class="td-user"> |
|
26 | <td class="td-user"> | |
27 | ${base.gravatar(_user.email, 16)} |
|
27 | ${base.gravatar(_user.email, 16)} | |
28 | ${h.link_to_user(_user.username)} |
|
28 | ${h.link_to_user(_user.username)} | |
29 | %if getattr(_user, 'admin_row', None): |
|
29 | %if getattr(_user, 'admin_row', None): | |
30 | (${_('super admin')}) |
|
30 | (${_('super admin')}) | |
31 | %endif |
|
31 | %endif | |
32 | %if getattr(_user, 'owner_row', None): |
|
32 | %if getattr(_user, 'owner_row', None): | |
33 | (${_('owner')}) |
|
33 | (${_('owner')}) | |
34 | %endif |
|
34 | %endif | |
35 | </td> |
|
35 | </td> | |
36 | <td></td> |
|
36 | <td></td> | |
37 | </tr> |
|
37 | </tr> | |
38 | %elif _user.username == h.DEFAULT_USER and c.repo_info.private: |
|
38 | %elif _user.username == h.DEFAULT_USER and c.repo_info.private: | |
39 | <tr> |
|
39 | <tr> | |
40 | <td colspan="4"> |
|
40 | <td colspan="4"> | |
41 | <span class="private_repo_msg"> |
|
41 | <span class="private_repo_msg"> | |
42 | <strong title="${h.tooltip(_user.permission)}">${_('private repository')}</strong> |
|
42 | <strong title="${h.tooltip(_user.permission)}">${_('private repository')}</strong> | |
43 | </span> |
|
43 | </span> | |
44 | </td> |
|
44 | </td> | |
45 | <td class="private_repo_msg"> |
|
45 | <td class="private_repo_msg"> | |
46 | ${base.gravatar(h.DEFAULT_USER_EMAIL, 16)} |
|
46 | ${base.gravatar(h.DEFAULT_USER_EMAIL, 16)} | |
47 | ${h.DEFAULT_USER} - ${_('only users/user groups explicitly added here will have access')}</td> |
|
47 | ${h.DEFAULT_USER} - ${_('only users/user groups explicitly added here will have access')}</td> | |
48 | <td></td> |
|
48 | <td></td> | |
49 | </tr> |
|
49 | </tr> | |
50 | %else: |
|
50 | %else: | |
51 | <tr> |
|
51 | <tr> | |
52 | <td class="td-radio">${h.radio('u_perm_%s' % _user.user_id,'repository.none', checked=_user.permission=='repository.none')}</td> |
|
52 | <td class="td-radio">${h.radio('u_perm_%s' % _user.user_id,'repository.none', checked=_user.permission=='repository.none')}</td> | |
53 | <td class="td-radio">${h.radio('u_perm_%s' % _user.user_id,'repository.read', checked=_user.permission=='repository.read')}</td> |
|
53 | <td class="td-radio">${h.radio('u_perm_%s' % _user.user_id,'repository.read', checked=_user.permission=='repository.read')}</td> | |
54 | <td class="td-radio">${h.radio('u_perm_%s' % _user.user_id,'repository.write', checked=_user.permission=='repository.write')}</td> |
|
54 | <td class="td-radio">${h.radio('u_perm_%s' % _user.user_id,'repository.write', checked=_user.permission=='repository.write')}</td> | |
55 | <td class="td-radio">${h.radio('u_perm_%s' % _user.user_id,'repository.admin', checked=_user.permission=='repository.admin')}</td> |
|
55 | <td class="td-radio">${h.radio('u_perm_%s' % _user.user_id,'repository.admin', checked=_user.permission=='repository.admin')}</td> | |
56 | <td class="td-user"> |
|
56 | <td class="td-user"> | |
57 | ${base.gravatar(_user.email, 16)} |
|
57 | ${base.gravatar(_user.email, 16)} | |
58 | <span class="user"> |
|
58 | <span class="user"> | |
59 | % if _user.username == h.DEFAULT_USER: |
|
59 | % if _user.username == h.DEFAULT_USER: | |
60 | ${h.DEFAULT_USER} <span class="user-perm-help-text"> - ${_('permission for all other users')}</span> |
|
60 | ${h.DEFAULT_USER} <span class="user-perm-help-text"> - ${_('permission for all other users')}</span> | |
61 | % else: |
|
61 | % else: | |
62 | ${h.link_to_user(_user.username)} |
|
62 | ${h.link_to_user(_user.username)} | |
63 | % endif |
|
63 | % endif | |
64 | </span> |
|
64 | </span> | |
65 | </td> |
|
65 | </td> | |
66 | <td class="td-action"> |
|
66 | <td class="td-action"> | |
67 | %if _user.username != h.DEFAULT_USER: |
|
67 | %if _user.username != h.DEFAULT_USER: | |
68 | <span class="btn btn-link btn-danger revoke_perm" |
|
68 | <span class="btn btn-link btn-danger revoke_perm" | |
69 | member="${_user.user_id}" member_type="user"> |
|
69 | member="${_user.user_id}" member_type="user"> | |
70 | <i class="icon-remove"></i> ${_('Revoke')} |
|
70 | <i class="icon-remove"></i> ${_('Revoke')} | |
71 | </span> |
|
71 | </span> | |
72 | %endif |
|
72 | %endif | |
73 | </td> |
|
73 | </td> | |
74 | </tr> |
|
74 | </tr> | |
75 | %endif |
|
75 | %endif | |
76 | %endfor |
|
76 | %endfor | |
77 |
|
77 | |||
78 | ## USER GROUPS |
|
78 | ## USER GROUPS | |
79 | %for _user_group in c.repo_info.permission_user_groups(): |
|
79 | %for _user_group in c.repo_info.permission_user_groups(): | |
80 | <tr> |
|
80 | <tr> | |
81 | <td class="td-radio">${h.radio('g_perm_%s' % _user_group.users_group_id,'repository.none', checked=_user_group.permission=='repository.none')}</td> |
|
81 | <td class="td-radio">${h.radio('g_perm_%s' % _user_group.users_group_id,'repository.none', checked=_user_group.permission=='repository.none')}</td> | |
82 | <td class="td-radio">${h.radio('g_perm_%s' % _user_group.users_group_id,'repository.read', checked=_user_group.permission=='repository.read')}</td> |
|
82 | <td class="td-radio">${h.radio('g_perm_%s' % _user_group.users_group_id,'repository.read', checked=_user_group.permission=='repository.read')}</td> | |
83 | <td class="td-radio">${h.radio('g_perm_%s' % _user_group.users_group_id,'repository.write', checked=_user_group.permission=='repository.write')}</td> |
|
83 | <td class="td-radio">${h.radio('g_perm_%s' % _user_group.users_group_id,'repository.write', checked=_user_group.permission=='repository.write')}</td> | |
84 | <td class="td-radio">${h.radio('g_perm_%s' % _user_group.users_group_id,'repository.admin', checked=_user_group.permission=='repository.admin')}</td> |
|
84 | <td class="td-radio">${h.radio('g_perm_%s' % _user_group.users_group_id,'repository.admin', checked=_user_group.permission=='repository.admin')}</td> | |
85 | <td class="td-componentname"> |
|
85 | <td class="td-componentname"> | |
86 | <i class="icon-group" ></i> |
|
86 | <i class="icon-group" ></i> | |
87 | %if h.HasPermissionAny('hg.admin')(): |
|
87 | %if h.HasPermissionAny('hg.admin')(): | |
88 | <a href="${h.url('edit_users_group',user_group_id=_user_group.users_group_id)}"> |
|
88 | <a href="${h.url('edit_users_group',user_group_id=_user_group.users_group_id)}"> | |
89 | ${_user_group.users_group_name} |
|
89 | ${_user_group.users_group_name} | |
90 | </a> |
|
90 | </a> | |
91 | %else: |
|
91 | %else: | |
92 | ${_user_group.users_group_name} |
|
92 | ${_user_group.users_group_name} | |
93 | %endif |
|
93 | %endif | |
94 | </td> |
|
94 | </td> | |
95 | <td class="td-action"> |
|
95 | <td class="td-action"> | |
96 | <span class="btn btn-link btn-danger revoke_perm" |
|
96 | <span class="btn btn-link btn-danger revoke_perm" | |
97 | member="${_user_group.users_group_id}" member_type="user_group"> |
|
97 | member="${_user_group.users_group_id}" member_type="user_group"> | |
98 | <i class="icon-remove"></i> ${_('Revoke')} |
|
98 | <i class="icon-remove"></i> ${_('Revoke')} | |
99 | </span> |
|
99 | </span> | |
100 | </td> |
|
100 | </td> | |
101 | </tr> |
|
101 | </tr> | |
102 | %endfor |
|
102 | %endfor | |
103 | <tr class="new_members" id="add_perm_input"></tr> |
|
103 | <tr class="new_members" id="add_perm_input"></tr> | |
104 | </table> |
|
104 | </table> | |
105 | <div id="add_perm" class="link"> |
|
105 | <div id="add_perm" class="link"> | |
106 | ${_('Add new')} |
|
106 | ${_('Add new')} | |
107 | </div> |
|
107 | </div> | |
108 | <div class="buttons"> |
|
108 | <div class="buttons"> | |
109 | ${h.submit('save',_('Save'),class_="btn btn-primary")} |
|
109 | ${h.submit('save',_('Save'),class_="btn btn-primary")} | |
110 | ${h.reset('reset',_('Reset'),class_="btn btn-danger")} |
|
110 | ${h.reset('reset',_('Reset'),class_="btn btn-danger")} | |
111 | </div> |
|
111 | </div> | |
112 | ${h.end_form()} |
|
112 | ${h.end_form()} | |
113 | </div> |
|
113 | </div> | |
114 | </div> |
|
114 | </div> | |
115 |
|
115 | |||
116 | <script type="text/javascript"> |
|
116 | <script type="text/javascript"> | |
117 | $('#add_perm').on('click', function(e){ |
|
117 | $('#add_perm').on('click', function(e){ | |
118 | addNewPermInput($(this), 'repository'); |
|
118 | addNewPermInput($(this), 'repository'); | |
119 | }); |
|
119 | }); | |
120 | $('.revoke_perm').on('click', function(e){ |
|
120 | $('.revoke_perm').on('click', function(e){ | |
121 | markRevokePermInput($(this), 'repository'); |
|
121 | markRevokePermInput($(this), 'repository'); | |
122 | }); |
|
122 | }); | |
123 | </script> |
|
123 | </script> |
@@ -1,260 +1,260 b'' | |||||
1 | ## -*- coding: utf-8 -*- |
|
1 | ## -*- coding: utf-8 -*- | |
2 | <%namespace name="base" file="/base/base.mako"/> |
|
2 | <%namespace name="base" file="/base/base.mako"/> | |
3 |
|
3 | |||
4 | <div class="panel panel-default"> |
|
4 | <div class="panel panel-default"> | |
5 | <div class="panel-heading"> |
|
5 | <div class="panel-heading"> | |
6 | <h3 class="panel-title">${_('Settings for Repository: %s') % c.rhodecode_db_repo.repo_name}</h3> |
|
6 | <h3 class="panel-title">${_('Settings for Repository: %s') % c.rhodecode_db_repo.repo_name}</h3> | |
7 | </div> |
|
7 | </div> | |
8 | <div class="panel-body"> |
|
8 | <div class="panel-body"> | |
9 | ${h.secure_form(h.route_path('edit_repo', repo_name=c.rhodecode_db_repo.repo_name), method='POST')} |
|
9 | ${h.secure_form(h.route_path('edit_repo', repo_name=c.rhodecode_db_repo.repo_name), method='POST', request=request)} | |
10 | <div class="form"> |
|
10 | <div class="form"> | |
11 | <!-- fields --> |
|
11 | <!-- fields --> | |
12 | <div class="fields"> |
|
12 | <div class="fields"> | |
13 |
|
13 | |||
14 | <div class="field"> |
|
14 | <div class="field"> | |
15 | <div class="label"> |
|
15 | <div class="label"> | |
16 | <label for="repo_name">${_('Name')}:</label> |
|
16 | <label for="repo_name">${_('Name')}:</label> | |
17 | </div> |
|
17 | </div> | |
18 | <div class="input"> |
|
18 | <div class="input"> | |
19 | ${c.form['repo_name'].render(css_class='medium', oid='repo_name')|n} |
|
19 | ${c.form['repo_name'].render(css_class='medium', oid='repo_name')|n} | |
20 | ${c.form.render_error(request, c.form['repo_name'])|n} |
|
20 | ${c.form.render_error(request, c.form['repo_name'])|n} | |
21 |
|
21 | |||
22 | <p class="help-block">${_('Non-changeable id')}: `_${c.rhodecode_db_repo.repo_id}` <span><a href="#" onclick="$('#clone_id').toggle();return false">${_('what is that ?')}</a></span></p> |
|
22 | <p class="help-block">${_('Non-changeable id')}: `_${c.rhodecode_db_repo.repo_id}` <span><a href="#" onclick="$('#clone_id').toggle();return false">${_('what is that ?')}</a></span></p> | |
23 | <p id="clone_id" style="display:none;"> |
|
23 | <p id="clone_id" style="display:none;"> | |
24 | ${_('URL by id')}: `${c.rhodecode_db_repo.clone_url(with_id=True)}` <br/> |
|
24 | ${_('URL by id')}: `${c.rhodecode_db_repo.clone_url(with_id=True)}` <br/> | |
25 | ${_('''In case this repository is renamed or moved into another group the repository url changes. |
|
25 | ${_('''In case this repository is renamed or moved into another group the repository url changes. | |
26 | Using above url guarantees that this repository will always be accessible under such url. |
|
26 | Using above url guarantees that this repository will always be accessible under such url. | |
27 | Useful for CI systems, or any other cases that you need to hardcode the url into 3rd party service.''')}</p> |
|
27 | Useful for CI systems, or any other cases that you need to hardcode the url into 3rd party service.''')}</p> | |
28 | </div> |
|
28 | </div> | |
29 | </div> |
|
29 | </div> | |
30 |
|
30 | |||
31 | <div class="field"> |
|
31 | <div class="field"> | |
32 | <div class="label"> |
|
32 | <div class="label"> | |
33 | <label for="repo_group">${_('Repository group')}:</label> |
|
33 | <label for="repo_group">${_('Repository group')}:</label> | |
34 | </div> |
|
34 | </div> | |
35 | <div class="select"> |
|
35 | <div class="select"> | |
36 | ${c.form['repo_group'].render(css_class='medium', oid='repo_group')|n} |
|
36 | ${c.form['repo_group'].render(css_class='medium', oid='repo_group')|n} | |
37 | ${c.form.render_error(request, c.form['repo_group'])|n} |
|
37 | ${c.form.render_error(request, c.form['repo_group'])|n} | |
38 |
|
38 | |||
39 | % if c.personal_repo_group: |
|
39 | % if c.personal_repo_group: | |
40 | <a class="btn" href="#" data-personal-group-name="${c.personal_repo_group.group_name}" data-personal-group-id="${c.personal_repo_group.group_id}" onclick="selectMyGroup(this); return false"> |
|
40 | <a class="btn" href="#" data-personal-group-name="${c.personal_repo_group.group_name}" data-personal-group-id="${c.personal_repo_group.group_id}" onclick="selectMyGroup(this); return false"> | |
41 | ${_('Select my personal group (`%(repo_group_name)s`)') % {'repo_group_name': c.personal_repo_group.group_name}} |
|
41 | ${_('Select my personal group (`%(repo_group_name)s`)') % {'repo_group_name': c.personal_repo_group.group_name}} | |
42 | </a> |
|
42 | </a> | |
43 | % endif |
|
43 | % endif | |
44 | <p class="help-block">${_('Optional select a group to put this repository into.')}</p> |
|
44 | <p class="help-block">${_('Optional select a group to put this repository into.')}</p> | |
45 | </div> |
|
45 | </div> | |
46 | </div> |
|
46 | </div> | |
47 |
|
47 | |||
48 | % if c.rhodecode_db_repo.repo_type != 'svn': |
|
48 | % if c.rhodecode_db_repo.repo_type != 'svn': | |
49 | <div class="field"> |
|
49 | <div class="field"> | |
50 | <div class="label"> |
|
50 | <div class="label"> | |
51 | <label for="clone_uri">${_('Remote uri')}:</label> |
|
51 | <label for="clone_uri">${_('Remote uri')}:</label> | |
52 | </div> |
|
52 | </div> | |
53 | <div class="input"> |
|
53 | <div class="input"> | |
54 | %if c.rhodecode_db_repo.clone_uri: |
|
54 | %if c.rhodecode_db_repo.clone_uri: | |
55 | ## display, if we don't have any errors |
|
55 | ## display, if we don't have any errors | |
56 | % if not c.form['repo_clone_uri'].error: |
|
56 | % if not c.form['repo_clone_uri'].error: | |
57 | <div id="clone_uri_hidden" class='text-as-placeholder'> |
|
57 | <div id="clone_uri_hidden" class='text-as-placeholder'> | |
58 | <span id="clone_uri_hidden_value">${c.rhodecode_db_repo.clone_uri_hidden}</span> |
|
58 | <span id="clone_uri_hidden_value">${c.rhodecode_db_repo.clone_uri_hidden}</span> | |
59 | <span class="link" id="edit_clone_uri"><i class="icon-edit"></i>${_('edit')}</span> |
|
59 | <span class="link" id="edit_clone_uri"><i class="icon-edit"></i>${_('edit')}</span> | |
60 | </div> |
|
60 | </div> | |
61 | % endif |
|
61 | % endif | |
62 |
|
62 | |||
63 | ## alter field |
|
63 | ## alter field | |
64 | <div id="alter_clone_uri" style="${'' if c.form['repo_clone_uri'].error else 'display: none'}"> |
|
64 | <div id="alter_clone_uri" style="${'' if c.form['repo_clone_uri'].error else 'display: none'}"> | |
65 | ${c.form['repo_clone_uri'].render(css_class='medium', oid='clone_uri', placeholder=_('enter new value, or leave empty to remove'))|n} |
|
65 | ${c.form['repo_clone_uri'].render(css_class='medium', oid='clone_uri', placeholder=_('enter new value, or leave empty to remove'))|n} | |
66 | ${c.form.render_error(request, c.form['repo_clone_uri'])|n} |
|
66 | ${c.form.render_error(request, c.form['repo_clone_uri'])|n} | |
67 | % if c.form['repo_clone_uri'].error: |
|
67 | % if c.form['repo_clone_uri'].error: | |
68 | ## we got error from form subit, means we modify the url |
|
68 | ## we got error from form subit, means we modify the url | |
69 | ${h.hidden('repo_clone_uri_change', 'MOD')} |
|
69 | ${h.hidden('repo_clone_uri_change', 'MOD')} | |
70 | % else: |
|
70 | % else: | |
71 | ${h.hidden('repo_clone_uri_change', 'OLD')} |
|
71 | ${h.hidden('repo_clone_uri_change', 'OLD')} | |
72 | % endif |
|
72 | % endif | |
73 |
|
73 | |||
74 | % if not c.form['repo_clone_uri'].error: |
|
74 | % if not c.form['repo_clone_uri'].error: | |
75 | <span class="link" id="cancel_edit_clone_uri">${_('cancel')}</span> |
|
75 | <span class="link" id="cancel_edit_clone_uri">${_('cancel')}</span> | |
76 | % endif |
|
76 | % endif | |
77 |
|
77 | |||
78 | </div> |
|
78 | </div> | |
79 | %else: |
|
79 | %else: | |
80 | ## not set yet, display form to set it |
|
80 | ## not set yet, display form to set it | |
81 | ${c.form['repo_clone_uri'].render(css_class='medium', oid='clone_uri')|n} |
|
81 | ${c.form['repo_clone_uri'].render(css_class='medium', oid='clone_uri')|n} | |
82 | ${c.form.render_error(request, c.form['repo_clone_uri'])|n} |
|
82 | ${c.form.render_error(request, c.form['repo_clone_uri'])|n} | |
83 | ${h.hidden('repo_clone_uri_change', 'NEW')} |
|
83 | ${h.hidden('repo_clone_uri_change', 'NEW')} | |
84 | %endif |
|
84 | %endif | |
85 | <p id="alter_clone_uri_help_block" class="help-block"> |
|
85 | <p id="alter_clone_uri_help_block" class="help-block"> | |
86 | <% pull_link = h.literal(h.link_to('remote sync', h.url('edit_repo_remote', repo_name=c.repo_name))) %> |
|
86 | <% pull_link = h.literal(h.link_to('remote sync', h.url('edit_repo_remote', repo_name=c.repo_name))) %> | |
87 | ${_('http[s] url where from repository was imported, this field can used for doing {pull_link}.').format(pull_link=pull_link)|n} <br/> |
|
87 | ${_('http[s] url where from repository was imported, this field can used for doing {pull_link}.').format(pull_link=pull_link)|n} <br/> | |
88 | ${_('This field is stored encrypted inside Database, a format of http://user:password@server.com/repo_name can be used and will be hidden from display.')} |
|
88 | ${_('This field is stored encrypted inside Database, a format of http://user:password@server.com/repo_name can be used and will be hidden from display.')} | |
89 | </p> |
|
89 | </p> | |
90 | </div> |
|
90 | </div> | |
91 | </div> |
|
91 | </div> | |
92 | % else: |
|
92 | % else: | |
93 | ${h.hidden('repo_clone_uri', '')} |
|
93 | ${h.hidden('repo_clone_uri', '')} | |
94 | % endif |
|
94 | % endif | |
95 |
|
95 | |||
96 | <div class="field"> |
|
96 | <div class="field"> | |
97 | <div class="label"> |
|
97 | <div class="label"> | |
98 | <label for="repo_landing_commit_ref">${_('Landing commit')}:</label> |
|
98 | <label for="repo_landing_commit_ref">${_('Landing commit')}:</label> | |
99 | </div> |
|
99 | </div> | |
100 | <div class="select"> |
|
100 | <div class="select"> | |
101 | ${c.form['repo_landing_commit_ref'].render(css_class='medium', oid='repo_landing_commit_ref')|n} |
|
101 | ${c.form['repo_landing_commit_ref'].render(css_class='medium', oid='repo_landing_commit_ref')|n} | |
102 | ${c.form.render_error(request, c.form['repo_landing_commit_ref'])|n} |
|
102 | ${c.form.render_error(request, c.form['repo_landing_commit_ref'])|n} | |
103 | <p class="help-block">${_('Default commit for files page, downloads, full text search index and readme')}</p> |
|
103 | <p class="help-block">${_('Default commit for files page, downloads, full text search index and readme')}</p> | |
104 | </div> |
|
104 | </div> | |
105 | </div> |
|
105 | </div> | |
106 |
|
106 | |||
107 | <div class="field badged-field"> |
|
107 | <div class="field badged-field"> | |
108 | <div class="label"> |
|
108 | <div class="label"> | |
109 | <label for="repo_owner">${_('Owner')}:</label> |
|
109 | <label for="repo_owner">${_('Owner')}:</label> | |
110 | </div> |
|
110 | </div> | |
111 | <div class="input"> |
|
111 | <div class="input"> | |
112 | <div class="badge-input-container"> |
|
112 | <div class="badge-input-container"> | |
113 | <div class="user-badge"> |
|
113 | <div class="user-badge"> | |
114 | ${base.gravatar_with_user(c.rhodecode_db_repo.user.email or c.rhodecode_db_repo.user.username, show_disabled=not c.rhodecode_db_repo.user.active)} |
|
114 | ${base.gravatar_with_user(c.rhodecode_db_repo.user.email or c.rhodecode_db_repo.user.username, show_disabled=not c.rhodecode_db_repo.user.active)} | |
115 | </div> |
|
115 | </div> | |
116 | <div class="badge-input-wrap"> |
|
116 | <div class="badge-input-wrap"> | |
117 | ${c.form['repo_owner'].render(css_class='medium', oid='repo_owner')|n} |
|
117 | ${c.form['repo_owner'].render(css_class='medium', oid='repo_owner')|n} | |
118 | </div> |
|
118 | </div> | |
119 | </div> |
|
119 | </div> | |
120 | ${c.form.render_error(request, c.form['repo_owner'])|n} |
|
120 | ${c.form.render_error(request, c.form['repo_owner'])|n} | |
121 | <p class="help-block">${_('Change owner of this repository.')}</p> |
|
121 | <p class="help-block">${_('Change owner of this repository.')}</p> | |
122 | </div> |
|
122 | </div> | |
123 | </div> |
|
123 | </div> | |
124 |
|
124 | |||
125 | <div class="field"> |
|
125 | <div class="field"> | |
126 | <div class="label label-textarea"> |
|
126 | <div class="label label-textarea"> | |
127 | <label for="repo_description">${_('Description')}:</label> |
|
127 | <label for="repo_description">${_('Description')}:</label> | |
128 | </div> |
|
128 | </div> | |
129 | <div class="textarea text-area editor"> |
|
129 | <div class="textarea text-area editor"> | |
130 | ${c.form['repo_description'].render(css_class='medium', oid='repo_description')|n} |
|
130 | ${c.form['repo_description'].render(css_class='medium', oid='repo_description')|n} | |
131 | ${c.form.render_error(request, c.form['repo_description'])|n} |
|
131 | ${c.form.render_error(request, c.form['repo_description'])|n} | |
132 | <p class="help-block">${_('Keep it short and to the point. Use a README file for longer descriptions.')}</p> |
|
132 | <p class="help-block">${_('Keep it short and to the point. Use a README file for longer descriptions.')}</p> | |
133 | </div> |
|
133 | </div> | |
134 | </div> |
|
134 | </div> | |
135 |
|
135 | |||
136 | <div class="field"> |
|
136 | <div class="field"> | |
137 | <div class="label label-checkbox"> |
|
137 | <div class="label label-checkbox"> | |
138 | <label for="${c.form['repo_private'].oid}">${_('Private repository')}:</label> |
|
138 | <label for="${c.form['repo_private'].oid}">${_('Private repository')}:</label> | |
139 | </div> |
|
139 | </div> | |
140 | <div class="checkboxes"> |
|
140 | <div class="checkboxes"> | |
141 | ${c.form['repo_private'].render(css_class='medium')|n} |
|
141 | ${c.form['repo_private'].render(css_class='medium')|n} | |
142 | ${c.form.render_error(request, c.form['repo_private'])|n} |
|
142 | ${c.form.render_error(request, c.form['repo_private'])|n} | |
143 | <span class="help-block">${_('Private repositories are only visible to people explicitly added as collaborators.')}</span> |
|
143 | <span class="help-block">${_('Private repositories are only visible to people explicitly added as collaborators.')}</span> | |
144 | </div> |
|
144 | </div> | |
145 | </div> |
|
145 | </div> | |
146 | <div class="field"> |
|
146 | <div class="field"> | |
147 | <div class="label label-checkbox"> |
|
147 | <div class="label label-checkbox"> | |
148 | <label for="${c.form['repo_enable_statistics'].oid}">${_('Enable statistics')}:</label> |
|
148 | <label for="${c.form['repo_enable_statistics'].oid}">${_('Enable statistics')}:</label> | |
149 | </div> |
|
149 | </div> | |
150 | <div class="checkboxes"> |
|
150 | <div class="checkboxes"> | |
151 | ${c.form['repo_enable_statistics'].render(css_class='medium')|n} |
|
151 | ${c.form['repo_enable_statistics'].render(css_class='medium')|n} | |
152 | ${c.form.render_error(request, c.form['repo_enable_statistics'])|n} |
|
152 | ${c.form.render_error(request, c.form['repo_enable_statistics'])|n} | |
153 | <span class="help-block">${_('Enable statistics window on summary page.')}</span> |
|
153 | <span class="help-block">${_('Enable statistics window on summary page.')}</span> | |
154 | </div> |
|
154 | </div> | |
155 | </div> |
|
155 | </div> | |
156 | <div class="field"> |
|
156 | <div class="field"> | |
157 | <div class="label label-checkbox"> |
|
157 | <div class="label label-checkbox"> | |
158 | <label for="${c.form['repo_enable_downloads'].oid}">${_('Enable downloads')}:</label> |
|
158 | <label for="${c.form['repo_enable_downloads'].oid}">${_('Enable downloads')}:</label> | |
159 | </div> |
|
159 | </div> | |
160 | <div class="checkboxes"> |
|
160 | <div class="checkboxes"> | |
161 | ${c.form['repo_enable_downloads'].render(css_class='medium')|n} |
|
161 | ${c.form['repo_enable_downloads'].render(css_class='medium')|n} | |
162 | ${c.form.render_error(request, c.form['repo_enable_downloads'])|n} |
|
162 | ${c.form.render_error(request, c.form['repo_enable_downloads'])|n} | |
163 | <span class="help-block">${_('Enable download menu on summary page.')}</span> |
|
163 | <span class="help-block">${_('Enable download menu on summary page.')}</span> | |
164 | </div> |
|
164 | </div> | |
165 | </div> |
|
165 | </div> | |
166 | <div class="field"> |
|
166 | <div class="field"> | |
167 | <div class="label label-checkbox"> |
|
167 | <div class="label label-checkbox"> | |
168 | <label for="${c.form['repo_enable_locking'].oid}">${_('Enable automatic locking')}:</label> |
|
168 | <label for="${c.form['repo_enable_locking'].oid}">${_('Enable automatic locking')}:</label> | |
169 | </div> |
|
169 | </div> | |
170 | <div class="checkboxes"> |
|
170 | <div class="checkboxes"> | |
171 | ${c.form['repo_enable_locking'].render(css_class='medium')|n} |
|
171 | ${c.form['repo_enable_locking'].render(css_class='medium')|n} | |
172 | ${c.form.render_error(request, c.form['repo_enable_locking'])|n} |
|
172 | ${c.form.render_error(request, c.form['repo_enable_locking'])|n} | |
173 | <span class="help-block">${_('Enable automatic locking on repository. Pulling from this repository creates a lock that can be released by pushing back by the same user')}</span> |
|
173 | <span class="help-block">${_('Enable automatic locking on repository. Pulling from this repository creates a lock that can be released by pushing back by the same user')}</span> | |
174 | </div> |
|
174 | </div> | |
175 | </div> |
|
175 | </div> | |
176 |
|
176 | |||
177 | %if c.visual.repository_fields: |
|
177 | %if c.visual.repository_fields: | |
178 | ## EXTRA FIELDS |
|
178 | ## EXTRA FIELDS | |
179 | %for field in c.repo_fields: |
|
179 | %for field in c.repo_fields: | |
180 | <div class="field"> |
|
180 | <div class="field"> | |
181 | <div class="label"> |
|
181 | <div class="label"> | |
182 | <label for="${field.field_key_prefixed}">${field.field_label} (${field.field_key}):</label> |
|
182 | <label for="${field.field_key_prefixed}">${field.field_label} (${field.field_key}):</label> | |
183 | </div> |
|
183 | </div> | |
184 | <div class="input input-medium"> |
|
184 | <div class="input input-medium"> | |
185 | ${h.text(field.field_key_prefixed, field.field_value, class_='medium')} |
|
185 | ${h.text(field.field_key_prefixed, field.field_value, class_='medium')} | |
186 | %if field.field_desc: |
|
186 | %if field.field_desc: | |
187 | <span class="help-block">${field.field_desc}</span> |
|
187 | <span class="help-block">${field.field_desc}</span> | |
188 | %endif |
|
188 | %endif | |
189 | </div> |
|
189 | </div> | |
190 | </div> |
|
190 | </div> | |
191 | %endfor |
|
191 | %endfor | |
192 | %endif |
|
192 | %endif | |
193 | <div class="buttons"> |
|
193 | <div class="buttons"> | |
194 | ${h.submit('save',_('Save'),class_="btn")} |
|
194 | ${h.submit('save',_('Save'),class_="btn")} | |
195 | ${h.reset('reset',_('Reset'),class_="btn")} |
|
195 | ${h.reset('reset',_('Reset'),class_="btn")} | |
196 | </div> |
|
196 | </div> | |
197 | </div> |
|
197 | </div> | |
198 | </div> |
|
198 | </div> | |
199 | ${h.end_form()} |
|
199 | ${h.end_form()} | |
200 | </div> |
|
200 | </div> | |
201 | </div> |
|
201 | </div> | |
202 |
|
202 | |||
203 | <script> |
|
203 | <script> | |
204 | $(document).ready(function(){ |
|
204 | $(document).ready(function(){ | |
205 | var cloneUrl = function() { |
|
205 | var cloneUrl = function() { | |
206 | var alterButton = $('#alter_clone_uri'); |
|
206 | var alterButton = $('#alter_clone_uri'); | |
207 | var editButton = $('#edit_clone_uri'); |
|
207 | var editButton = $('#edit_clone_uri'); | |
208 | var cancelEditButton = $('#cancel_edit_clone_uri'); |
|
208 | var cancelEditButton = $('#cancel_edit_clone_uri'); | |
209 | var hiddenUrl = $('#clone_uri_hidden'); |
|
209 | var hiddenUrl = $('#clone_uri_hidden'); | |
210 | var hiddenUrlValue = $('#clone_uri_hidden_value'); |
|
210 | var hiddenUrlValue = $('#clone_uri_hidden_value'); | |
211 | var input = $('#clone_uri'); |
|
211 | var input = $('#clone_uri'); | |
212 | var helpBlock = $('#alter_clone_uri_help_block'); |
|
212 | var helpBlock = $('#alter_clone_uri_help_block'); | |
213 | var changedFlag = $('#repo_clone_uri_change'); |
|
213 | var changedFlag = $('#repo_clone_uri_change'); | |
214 | var originalText = helpBlock.html(); |
|
214 | var originalText = helpBlock.html(); | |
215 | var obfuscatedUrl = hiddenUrlValue.html(); |
|
215 | var obfuscatedUrl = hiddenUrlValue.html(); | |
216 |
|
216 | |||
217 | var edit = function(e) { |
|
217 | var edit = function(e) { | |
218 | alterButton.show(); |
|
218 | alterButton.show(); | |
219 | editButton.hide(); |
|
219 | editButton.hide(); | |
220 | hiddenUrl.hide(); |
|
220 | hiddenUrl.hide(); | |
221 |
|
221 | |||
222 | //add the old value next to input for verification |
|
222 | //add the old value next to input for verification | |
223 | helpBlock.html("(" + obfuscatedUrl + ")" + "<br\>" + originalText); |
|
223 | helpBlock.html("(" + obfuscatedUrl + ")" + "<br\>" + originalText); | |
224 | changedFlag.val('MOD'); |
|
224 | changedFlag.val('MOD'); | |
225 | }; |
|
225 | }; | |
226 |
|
226 | |||
227 | var cancelEdit = function(e) { |
|
227 | var cancelEdit = function(e) { | |
228 | alterButton.hide(); |
|
228 | alterButton.hide(); | |
229 | editButton.show(); |
|
229 | editButton.show(); | |
230 | hiddenUrl.show(); |
|
230 | hiddenUrl.show(); | |
231 |
|
231 | |||
232 | helpBlock.html(originalText); |
|
232 | helpBlock.html(originalText); | |
233 | changedFlag.val('OLD'); |
|
233 | changedFlag.val('OLD'); | |
234 | input.val(''); |
|
234 | input.val(''); | |
235 | }; |
|
235 | }; | |
236 |
|
236 | |||
237 | var initEvents = function() { |
|
237 | var initEvents = function() { | |
238 | editButton.on('click', edit); |
|
238 | editButton.on('click', edit); | |
239 | cancelEditButton.on('click', cancelEdit); |
|
239 | cancelEditButton.on('click', cancelEdit); | |
240 | }; |
|
240 | }; | |
241 |
|
241 | |||
242 | var setInitialState = function() { |
|
242 | var setInitialState = function() { | |
243 | if (input.hasClass('error')) { |
|
243 | if (input.hasClass('error')) { | |
244 | alterButton.show(); |
|
244 | alterButton.show(); | |
245 | editButton.hide(); |
|
245 | editButton.hide(); | |
246 | hiddenUrl.hide(); |
|
246 | hiddenUrl.hide(); | |
247 | } |
|
247 | } | |
248 | }; |
|
248 | }; | |
249 |
|
249 | |||
250 | setInitialState(); |
|
250 | setInitialState(); | |
251 | initEvents(); |
|
251 | initEvents(); | |
252 | }(); |
|
252 | }(); | |
253 |
|
253 | |||
254 | selectMyGroup = function(element) { |
|
254 | selectMyGroup = function(element) { | |
255 | $("#repo_group").val($(element).data('personalGroupId')).trigger("change"); |
|
255 | $("#repo_group").val($(element).data('personalGroupId')).trigger("change"); | |
256 | }; |
|
256 | }; | |
257 |
|
257 | |||
258 | UsersAutoComplete('repo_owner', '${c.rhodecode_user.user_id}'); |
|
258 | UsersAutoComplete('repo_owner', '${c.rhodecode_user.user_id}'); | |
259 | }); |
|
259 | }); | |
260 | </script> |
|
260 | </script> |
@@ -1,197 +1,197 b'' | |||||
1 | <div class="panel panel-default"> |
|
1 | <div class="panel panel-default"> | |
2 | <div class="panel-heading"> |
|
2 | <div class="panel-heading"> | |
3 | <h3 class="panel-title">${_('Strip commits from repository')}</h3> |
|
3 | <h3 class="panel-title">${_('Strip commits from repository')}</h3> | |
4 | </div> |
|
4 | </div> | |
5 | <div class="panel-body"> |
|
5 | <div class="panel-body"> | |
6 | %if c.repo_info.repo_type != 'svn': |
|
6 | %if c.repo_info.repo_type != 'svn': | |
7 | <h4>${_('Please provide up to %d commits commits to strip') % c.strip_limit}</h4> |
|
7 | <h4>${_('Please provide up to %d commits commits to strip') % c.strip_limit}</h4> | |
8 | <p> |
|
8 | <p> | |
9 | ${_('In the first step commits will be verified for existance in the repository')}. </br> |
|
9 | ${_('In the first step commits will be verified for existance in the repository')}. </br> | |
10 | ${_('In the second step, correct commits will be available for stripping')}. |
|
10 | ${_('In the second step, correct commits will be available for stripping')}. | |
11 | </p> |
|
11 | </p> | |
12 |
${h.secure_form(h.route_path('strip_check', repo_name=c.repo_info.repo_name), method=' |
|
12 | ${h.secure_form(h.route_path('strip_check', repo_name=c.repo_info.repo_name), method='POST', request=request)} | |
13 | <div id="change_body" class="field"> |
|
13 | <div id="change_body" class="field"> | |
14 | <div id="box-1" class="inputx locked_input"> |
|
14 | <div id="box-1" class="inputx locked_input"> | |
15 | <input class="text" id="changeset_id-1" name="changeset_id-1" size="59" |
|
15 | <input class="text" id="changeset_id-1" name="changeset_id-1" size="59" | |
16 | placeholder="${_('Enter full 40 character commit sha')}" type="text" value=""> |
|
16 | placeholder="${_('Enter full 40 character commit sha')}" type="text" value=""> | |
17 | <div id="plus_icon-1" class="btn btn-default plus_input_button" onclick="addNew(1);return false"> |
|
17 | <div id="plus_icon-1" class="btn btn-default plus_input_button" onclick="addNew(1);return false"> | |
18 | <i class="icon-plus">${_('Add another commit')}</i> |
|
18 | <i class="icon-plus">${_('Add another commit')}</i> | |
19 | </div> |
|
19 | </div> | |
20 | </div> |
|
20 | </div> | |
21 | </div> |
|
21 | </div> | |
22 |
|
22 | |||
23 | <div id="results" style="display:none; padding: 10px 0px;"></div> |
|
23 | <div id="results" style="display:none; padding: 10px 0px;"></div> | |
24 |
|
24 | |||
25 | <div class="buttons"> |
|
25 | <div class="buttons"> | |
26 | <button id="strip_action" class="btn btn-small btn-primary" onclick="checkCommits();return false"> |
|
26 | <button id="strip_action" class="btn btn-small btn-primary" onclick="checkCommits();return false"> | |
27 | ${_('Check commits')} |
|
27 | ${_('Check commits')} | |
28 | </button> |
|
28 | </button> | |
29 | </div> |
|
29 | </div> | |
30 |
|
30 | |||
31 | ${h.end_form()} |
|
31 | ${h.end_form()} | |
32 | %else: |
|
32 | %else: | |
33 | <h4>${_('Sorry this functionality is not available for SVN repository')}</h4> |
|
33 | <h4>${_('Sorry this functionality is not available for SVN repository')}</h4> | |
34 | %endif |
|
34 | %endif | |
35 | </div> |
|
35 | </div> | |
36 | </div> |
|
36 | </div> | |
37 |
|
37 | |||
38 |
|
38 | |||
39 | <script> |
|
39 | <script> | |
40 | var plus_leaf = 1; |
|
40 | var plus_leaf = 1; | |
41 |
|
41 | |||
42 | addNew = function(number){ |
|
42 | addNew = function(number){ | |
43 | if (number >= ${c.strip_limit}){ |
|
43 | if (number >= ${c.strip_limit}){ | |
44 | return; |
|
44 | return; | |
45 | } |
|
45 | } | |
46 | var minus = '<i class="icon-minus">${_('Remove')}</i>'; |
|
46 | var minus = '<i class="icon-minus">${_('Remove')}</i>'; | |
47 | $('#plus_icon-'+number).detach(); |
|
47 | $('#plus_icon-'+number).detach(); | |
48 | number++; |
|
48 | number++; | |
49 |
|
49 | |||
50 | var input = '<div id="box-'+number+'" class="inputx locked_input">'+ |
|
50 | var input = '<div id="box-'+number+'" class="inputx locked_input">'+ | |
51 | '<input class="text" id="changeset_id-'+number+'" name="changeset_id-'+number+'" size="59" type="text" value=""' + |
|
51 | '<input class="text" id="changeset_id-'+number+'" name="changeset_id-'+number+'" size="59" type="text" value=""' + | |
52 | 'placeholder="${_('Enter full 40 character commit sha')}">'+ |
|
52 | 'placeholder="${_('Enter full 40 character commit sha')}">'+ | |
53 | '<div id="plus_icon-'+number+'" class="btn btn-default plus_input_button" onclick="addNew('+number+');return false">'+ |
|
53 | '<div id="plus_icon-'+number+'" class="btn btn-default plus_input_button" onclick="addNew('+number+');return false">'+ | |
54 | '<i class="icon-plus">${_('Add another commit')}</i>'+ |
|
54 | '<i class="icon-plus">${_('Add another commit')}</i>'+ | |
55 | '</div>'+ |
|
55 | '</div>'+ | |
56 | '<div id="minus_icon-'+number+'" class="btn btn-default minus_input_button" onclick="delOld('+(number)+');return false">'+ |
|
56 | '<div id="minus_icon-'+number+'" class="btn btn-default minus_input_button" onclick="delOld('+(number)+');return false">'+ | |
57 | minus + |
|
57 | minus + | |
58 | '</div>' + |
|
58 | '</div>' + | |
59 | '</div>'; |
|
59 | '</div>'; | |
60 | $('#change_body').append(input); |
|
60 | $('#change_body').append(input); | |
61 | plus_leaf++; |
|
61 | plus_leaf++; | |
62 | }; |
|
62 | }; | |
63 |
|
63 | |||
64 | reIndex = function(number){ |
|
64 | reIndex = function(number){ | |
65 | for(var i=number;i<=plus_leaf;i++){ |
|
65 | for(var i=number;i<=plus_leaf;i++){ | |
66 | var check = $('#box-'+i); |
|
66 | var check = $('#box-'+i); | |
67 | if (check.length == 0){ |
|
67 | if (check.length == 0){ | |
68 | var change = $('#box-'+(i+1)); |
|
68 | var change = $('#box-'+(i+1)); | |
69 | change.attr('id','box-'+i); |
|
69 | change.attr('id','box-'+i); | |
70 | var plus = $('#plus_icon-'+(i+1)); |
|
70 | var plus = $('#plus_icon-'+(i+1)); | |
71 |
|
71 | |||
72 | if (plus.length != 0){ |
|
72 | if (plus.length != 0){ | |
73 | plus.attr('id','plus_icon-'+i); |
|
73 | plus.attr('id','plus_icon-'+i); | |
74 | plus.attr('onclick','addNew('+i+');return false'); |
|
74 | plus.attr('onclick','addNew('+i+');return false'); | |
75 | plus_leaf--; |
|
75 | plus_leaf--; | |
76 | } |
|
76 | } | |
77 | var minus = $('#minus_icon-'+(i+1)); |
|
77 | var minus = $('#minus_icon-'+(i+1)); | |
78 |
|
78 | |||
79 | minus.attr('id','minus_icon-'+i); |
|
79 | minus.attr('id','minus_icon-'+i); | |
80 |
|
80 | |||
81 | minus.attr('onclick','delOld('+i+');re' + |
|
81 | minus.attr('onclick','delOld('+i+');re' + | |
82 | 'turn false'); |
|
82 | 'turn false'); | |
83 | var input = $('input#changeset_id-'+(i+1)); |
|
83 | var input = $('input#changeset_id-'+(i+1)); | |
84 | input.attr('name','changeset_id-'+i); |
|
84 | input.attr('name','changeset_id-'+i); | |
85 | input.attr('id','changeset_id-'+i); |
|
85 | input.attr('id','changeset_id-'+i); | |
86 | } |
|
86 | } | |
87 | } |
|
87 | } | |
88 | }; |
|
88 | }; | |
89 |
|
89 | |||
90 | delOld = function(number){ |
|
90 | delOld = function(number){ | |
91 | $('#box-'+number).remove(); |
|
91 | $('#box-'+number).remove(); | |
92 | number = number - 1; |
|
92 | number = number - 1; | |
93 | var box = $('#box-'+number); |
|
93 | var box = $('#box-'+number); | |
94 | var plus = '<div id="plus_icon-'+number+'" class="btn btn-default plus_input_button" onclick="addNew('+number +');return false">'+ |
|
94 | var plus = '<div id="plus_icon-'+number+'" class="btn btn-default plus_input_button" onclick="addNew('+number +');return false">'+ | |
95 | '<i id="i_plus_icon-'+number+'" class="icon-plus">${_('Add another commit')}</i></div>'; |
|
95 | '<i id="i_plus_icon-'+number+'" class="icon-plus">${_('Add another commit')}</i></div>'; | |
96 | var minus = $('#minus_icon-'+number); |
|
96 | var minus = $('#minus_icon-'+number); | |
97 | if(number +1 == plus_leaf){ |
|
97 | if(number +1 == plus_leaf){ | |
98 | minus.detach(); |
|
98 | minus.detach(); | |
99 | box.append(plus); |
|
99 | box.append(plus); | |
100 | box.append(minus); |
|
100 | box.append(minus); | |
101 | plus_leaf --; |
|
101 | plus_leaf --; | |
102 | } |
|
102 | } | |
103 | reIndex(number+1); |
|
103 | reIndex(number+1); | |
104 |
|
104 | |||
105 | }; |
|
105 | }; | |
106 |
|
106 | |||
107 | var resultData = { |
|
107 | var resultData = { | |
108 | 'csrf_token': CSRF_TOKEN |
|
108 | 'csrf_token': CSRF_TOKEN | |
109 | }; |
|
109 | }; | |
110 |
|
110 | |||
111 | checkCommits = function() { |
|
111 | checkCommits = function() { | |
112 | var postData = $('form').serialize(); |
|
112 | var postData = $('form').serialize(); | |
113 | $('#results').show(); |
|
113 | $('#results').show(); | |
114 | $('#results').html('<h4>${_('Checking commits')}...</h4>'); |
|
114 | $('#results').html('<h4>${_('Checking commits')}...</h4>'); | |
115 | var url = "${h.route_path('strip_check', repo_name=c.repo_info.repo_name)}"; |
|
115 | var url = "${h.route_path('strip_check', repo_name=c.repo_info.repo_name)}"; | |
116 | var btn = $('#strip_action'); |
|
116 | var btn = $('#strip_action'); | |
117 | btn.attr('disabled', 'disabled'); |
|
117 | btn.attr('disabled', 'disabled'); | |
118 | btn.addClass('disabled'); |
|
118 | btn.addClass('disabled'); | |
119 |
|
119 | |||
120 | var success = function (data) { |
|
120 | var success = function (data) { | |
121 | resultData = { |
|
121 | resultData = { | |
122 | 'csrf_token': CSRF_TOKEN |
|
122 | 'csrf_token': CSRF_TOKEN | |
123 | }; |
|
123 | }; | |
124 | var i = 0; |
|
124 | var i = 0; | |
125 | var result = '<ol>'; |
|
125 | var result = '<ol>'; | |
126 | $.each(data, function(index, value){ |
|
126 | $.each(data, function(index, value){ | |
127 | i= index; |
|
127 | i= index; | |
128 | var box = $('#box-'+index); |
|
128 | var box = $('#box-'+index); | |
129 | if (value.rev){ |
|
129 | if (value.rev){ | |
130 | resultData[index] = JSON.stringify(value); |
|
130 | resultData[index] = JSON.stringify(value); | |
131 |
|
131 | |||
132 | var verifiedHtml = ( |
|
132 | var verifiedHtml = ( | |
133 | '<li style="line-height:1.2em">' + |
|
133 | '<li style="line-height:1.2em">' + | |
134 | '<code>{0}</code>' + |
|
134 | '<code>{0}</code>' + | |
135 | '{1}' + |
|
135 | '{1}' + | |
136 | '<div style="white-space:pre">' + |
|
136 | '<div style="white-space:pre">' + | |
137 | 'author: {2}\n' + |
|
137 | 'author: {2}\n' + | |
138 | 'description: {3}' + |
|
138 | 'description: {3}' + | |
139 | '</div>' + |
|
139 | '</div>' + | |
140 | '</li>').format( |
|
140 | '</li>').format( | |
141 | value.rev, |
|
141 | value.rev, | |
142 | "${_(' commit verified positive')}", |
|
142 | "${_(' commit verified positive')}", | |
143 | value.author, value.comment |
|
143 | value.author, value.comment | |
144 | ); |
|
144 | ); | |
145 | result += verifiedHtml; |
|
145 | result += verifiedHtml; | |
146 | } |
|
146 | } | |
147 | else { |
|
147 | else { | |
148 | var verifiedHtml = ( |
|
148 | var verifiedHtml = ( | |
149 | '<li style="line-height:1.2em">' + |
|
149 | '<li style="line-height:1.2em">' + | |
150 | '<code><strike>{0}</strike></code>' + |
|
150 | '<code><strike>{0}</strike></code>' + | |
151 | '{1}' + |
|
151 | '{1}' + | |
152 | '</li>').format( |
|
152 | '</li>').format( | |
153 | value.commit, |
|
153 | value.commit, | |
154 | "${_(' commit verified negative')}" |
|
154 | "${_(' commit verified negative')}" | |
155 | ); |
|
155 | ); | |
156 | result += verifiedHtml; |
|
156 | result += verifiedHtml; | |
157 | } |
|
157 | } | |
158 | box.remove(); |
|
158 | box.remove(); | |
159 | }); |
|
159 | }); | |
160 | result += '</ol>'; |
|
160 | result += '</ol>'; | |
161 | var box = $('#box-'+(parseInt(i)+1)); |
|
161 | var box = $('#box-'+(parseInt(i)+1)); | |
162 | box.remove(); |
|
162 | box.remove(); | |
163 | $('#results').html(result); |
|
163 | $('#results').html(result); | |
164 | }; |
|
164 | }; | |
165 |
|
165 | |||
166 | btn.html('Strip'); |
|
166 | btn.html('Strip'); | |
167 | btn.removeAttr('disabled'); |
|
167 | btn.removeAttr('disabled'); | |
168 | btn.removeClass('disabled'); |
|
168 | btn.removeClass('disabled'); | |
169 | btn.attr('onclick','strip();return false;'); |
|
169 | btn.attr('onclick','strip();return false;'); | |
170 | ajaxPOST(url, postData, success, null); |
|
170 | ajaxPOST(url, postData, success, null); | |
171 | }; |
|
171 | }; | |
172 |
|
172 | |||
173 | strip = function() { |
|
173 | strip = function() { | |
174 | var url = "${h.route_path('strip_execute', repo_name=c.repo_info.repo_name)}"; |
|
174 | var url = "${h.route_path('strip_execute', repo_name=c.repo_info.repo_name)}"; | |
175 | var success = function(data) { |
|
175 | var success = function(data) { | |
176 | var result = '<h4>Strip executed</h4><ol>'; |
|
176 | var result = '<h4>Strip executed</h4><ol>'; | |
177 | $.each(data, function(index, value){ |
|
177 | $.each(data, function(index, value){ | |
178 | if(data[index]) { |
|
178 | if(data[index]) { | |
179 | result += '<li><code>' +index+ '</code> ${_(' commit striped successfully')}' + '</li>'; |
|
179 | result += '<li><code>' +index+ '</code> ${_(' commit striped successfully')}' + '</li>'; | |
180 | } |
|
180 | } | |
181 | else { |
|
181 | else { | |
182 | result += '<li><code>' +index+ '</code> ${_(' commit strip failed')}' + '</li>'; |
|
182 | result += '<li><code>' +index+ '</code> ${_(' commit strip failed')}' + '</li>'; | |
183 | } |
|
183 | } | |
184 | }); |
|
184 | }); | |
185 | if ($.isEmptyObject(data)) { |
|
185 | if ($.isEmptyObject(data)) { | |
186 | result += '<li>Nothing done...</li>' |
|
186 | result += '<li>Nothing done...</li>' | |
187 | } |
|
187 | } | |
188 | result += '</ol>'; |
|
188 | result += '</ol>'; | |
189 | $('#results').html(result); |
|
189 | $('#results').html(result); | |
190 |
|
190 | |||
191 | }; |
|
191 | }; | |
192 | ajaxPOST(url, resultData, success, null); |
|
192 | ajaxPOST(url, resultData, success, null); | |
193 | var btn = $('#strip_action'); |
|
193 | var btn = $('#strip_action'); | |
194 | btn.remove(); |
|
194 | btn.remove(); | |
195 |
|
195 | |||
196 | }; |
|
196 | }; | |
197 | </script> |
|
197 | </script> |
@@ -1,62 +1,62 b'' | |||||
1 | <div class="panel panel-default"> |
|
1 | <div class="panel panel-default"> | |
2 | <div class="panel-heading"> |
|
2 | <div class="panel-heading"> | |
3 | <h3 class="panel-title">${_('User Sessions Configuration')}</h3> |
|
3 | <h3 class="panel-title">${_('User Sessions Configuration')}</h3> | |
4 | </div> |
|
4 | </div> | |
5 | <div class="panel-body"> |
|
5 | <div class="panel-body"> | |
6 | <% |
|
6 | <% | |
7 | elems = [ |
|
7 | elems = [ | |
8 | (_('Session type'), c.session_model.SESSION_TYPE, ''), |
|
8 | (_('Session type'), c.session_model.SESSION_TYPE, ''), | |
9 | (_('Session expiration period'), '{} seconds'.format(c.session_conf.get('beaker.session.timeout', 0)), ''), |
|
9 | (_('Session expiration period'), '{} seconds'.format(c.session_conf.get('beaker.session.timeout', 0)), ''), | |
10 |
|
10 | |||
11 | (_('Total sessions'), c.session_count, ''), |
|
11 | (_('Total sessions'), c.session_count, ''), | |
12 | (_('Expired sessions ({} days)').format(c.cleanup_older_days ), c.session_expired_count, ''), |
|
12 | (_('Expired sessions ({} days)').format(c.cleanup_older_days ), c.session_expired_count, ''), | |
13 |
|
13 | |||
14 | ] |
|
14 | ] | |
15 | %> |
|
15 | %> | |
16 | <dl class="dl-horizontal settings"> |
|
16 | <dl class="dl-horizontal settings"> | |
17 | %for dt, dd, tt in elems: |
|
17 | %for dt, dd, tt in elems: | |
18 | <dt>${dt}:</dt> |
|
18 | <dt>${dt}:</dt> | |
19 | <dd title="${h.tooltip(tt)}">${dd}</dd> |
|
19 | <dd title="${h.tooltip(tt)}">${dd}</dd> | |
20 | %endfor |
|
20 | %endfor | |
21 | </dl> |
|
21 | </dl> | |
22 | </div> |
|
22 | </div> | |
23 | </div> |
|
23 | </div> | |
24 |
|
24 | |||
25 |
|
25 | |||
26 | <div class="panel panel-warning"> |
|
26 | <div class="panel panel-warning"> | |
27 | <div class="panel-heading"> |
|
27 | <div class="panel-heading"> | |
28 | <h3 class="panel-title">${_('Cleanup Old Sessions')}</h3> |
|
28 | <h3 class="panel-title">${_('Cleanup Old Sessions')}</h3> | |
29 | </div> |
|
29 | </div> | |
30 | <div class="panel-body"> |
|
30 | <div class="panel-body"> | |
31 |
${h.secure_form(h.route_path('admin_settings_sessions_cleanup'), method=' |
|
31 | ${h.secure_form(h.route_path('admin_settings_sessions_cleanup'), method='POST', request=request)} | |
32 |
|
32 | |||
33 | <p> |
|
33 | <p> | |
34 | ${_('Cleanup user sessions that were not active during chosen time frame.')} <br/> |
|
34 | ${_('Cleanup user sessions that were not active during chosen time frame.')} <br/> | |
35 | ${_('After performing this action users whose session will be removed will be required to log in again.')} <br/> |
|
35 | ${_('After performing this action users whose session will be removed will be required to log in again.')} <br/> | |
36 | <strong>${_('Picking `All` will log-out you, and all users in the system.')}</strong> |
|
36 | <strong>${_('Picking `All` will log-out you, and all users in the system.')}</strong> | |
37 | </p> |
|
37 | </p> | |
38 |
|
38 | |||
39 | <script type="text/javascript"> |
|
39 | <script type="text/javascript"> | |
40 | $(document).ready(function() { |
|
40 | $(document).ready(function() { | |
41 | $('#expire_days').select2({ |
|
41 | $('#expire_days').select2({ | |
42 | containerCssClass: 'drop-menu', |
|
42 | containerCssClass: 'drop-menu', | |
43 | dropdownCssClass: 'drop-menu-dropdown', |
|
43 | dropdownCssClass: 'drop-menu-dropdown', | |
44 | dropdownAutoWidth: true, |
|
44 | dropdownAutoWidth: true, | |
45 | minimumResultsForSearch: -1 |
|
45 | minimumResultsForSearch: -1 | |
46 | }); |
|
46 | }); | |
47 | }); |
|
47 | }); | |
48 | </script> |
|
48 | </script> | |
49 | <select id="expire_days" name="expire_days"> |
|
49 | <select id="expire_days" name="expire_days"> | |
50 | % for n in [60, 90, 30, 7, 0]: |
|
50 | % for n in [60, 90, 30, 7, 0]: | |
51 | <option value="${n}">${'{} days'.format(n) if n != 0 else 'All'}</option> |
|
51 | <option value="${n}">${'{} days'.format(n) if n != 0 else 'All'}</option> | |
52 | % endfor |
|
52 | % endfor | |
53 | </select> |
|
53 | </select> | |
54 | <button class="btn btn-small" type="submit" |
|
54 | <button class="btn btn-small" type="submit" | |
55 | onclick="return confirm('${_('Confirm to cleanup user sessions')}');"> |
|
55 | onclick="return confirm('${_('Confirm to cleanup user sessions')}');"> | |
56 | ${_('Cleanup sessions')} |
|
56 | ${_('Cleanup sessions')} | |
57 | </button> |
|
57 | </button> | |
58 | ${h.end_form()} |
|
58 | ${h.end_form()} | |
59 | </div> |
|
59 | </div> | |
60 | </div> |
|
60 | </div> | |
61 |
|
61 | |||
62 |
|
62 |
@@ -1,157 +1,157 b'' | |||||
1 | <div class="panel panel-default"> |
|
1 | <div class="panel panel-default"> | |
2 | <div class="panel-heading"> |
|
2 | <div class="panel-heading"> | |
3 | <h3 class="panel-title">${_('Authentication Tokens')}</h3> |
|
3 | <h3 class="panel-title">${_('Authentication Tokens')}</h3> | |
4 | </div> |
|
4 | </div> | |
5 | <div class="panel-body"> |
|
5 | <div class="panel-body"> | |
6 | <div class="apikeys_wrap"> |
|
6 | <div class="apikeys_wrap"> | |
7 | <p> |
|
7 | <p> | |
8 | ${_('Each token can have a role. Token with a role can be used only in given context, ' |
|
8 | ${_('Each token can have a role. Token with a role can be used only in given context, ' | |
9 | 'e.g. VCS tokens can be used together with the authtoken auth plugin for git/hg/svn operations only.')} |
|
9 | 'e.g. VCS tokens can be used together with the authtoken auth plugin for git/hg/svn operations only.')} | |
10 | </p> |
|
10 | </p> | |
11 | <table class="rctable auth_tokens"> |
|
11 | <table class="rctable auth_tokens"> | |
12 | <tr> |
|
12 | <tr> | |
13 | <th>${_('Token')}</th> |
|
13 | <th>${_('Token')}</th> | |
14 | <th>${_('Scope')}</th> |
|
14 | <th>${_('Scope')}</th> | |
15 | <th>${_('Description')}</th> |
|
15 | <th>${_('Description')}</th> | |
16 | <th>${_('Role')}</th> |
|
16 | <th>${_('Role')}</th> | |
17 | <th>${_('Expiration')}</th> |
|
17 | <th>${_('Expiration')}</th> | |
18 | <th>${_('Action')}</th> |
|
18 | <th>${_('Action')}</th> | |
19 | </tr> |
|
19 | </tr> | |
20 | %if c.user_auth_tokens: |
|
20 | %if c.user_auth_tokens: | |
21 | %for auth_token in c.user_auth_tokens: |
|
21 | %for auth_token in c.user_auth_tokens: | |
22 | <tr class="${'expired' if auth_token.expired else ''}"> |
|
22 | <tr class="${'expired' if auth_token.expired else ''}"> | |
23 | <td class="truncate-wrap td-authtoken"><div class="user_auth_tokens truncate autoexpand"><code>${auth_token.api_key}</code></div></td> |
|
23 | <td class="truncate-wrap td-authtoken"><div class="user_auth_tokens truncate autoexpand"><code>${auth_token.api_key}</code></div></td> | |
24 | <td class="td">${auth_token.scope_humanized}</td> |
|
24 | <td class="td">${auth_token.scope_humanized}</td> | |
25 | <td class="td-wrap">${auth_token.description}</td> |
|
25 | <td class="td-wrap">${auth_token.description}</td> | |
26 | <td class="td-tags"> |
|
26 | <td class="td-tags"> | |
27 | <span class="tag disabled">${auth_token.role_humanized}</span> |
|
27 | <span class="tag disabled">${auth_token.role_humanized}</span> | |
28 | </td> |
|
28 | </td> | |
29 | <td class="td-exp"> |
|
29 | <td class="td-exp"> | |
30 | %if auth_token.expires == -1: |
|
30 | %if auth_token.expires == -1: | |
31 | ${_('never')} |
|
31 | ${_('never')} | |
32 | %else: |
|
32 | %else: | |
33 | %if auth_token.expired: |
|
33 | %if auth_token.expired: | |
34 | <span style="text-decoration: line-through">${h.age_component(h.time_to_utcdatetime(auth_token.expires))}</span> |
|
34 | <span style="text-decoration: line-through">${h.age_component(h.time_to_utcdatetime(auth_token.expires))}</span> | |
35 | %else: |
|
35 | %else: | |
36 | ${h.age_component(h.time_to_utcdatetime(auth_token.expires))} |
|
36 | ${h.age_component(h.time_to_utcdatetime(auth_token.expires))} | |
37 | %endif |
|
37 | %endif | |
38 | %endif |
|
38 | %endif | |
39 | </td> |
|
39 | </td> | |
40 | <td class="td-action"> |
|
40 | <td class="td-action"> | |
41 | ${h.secure_form(h.route_path('edit_user_auth_tokens_delete', user_id=c.user.user_id), method='POST')} |
|
41 | ${h.secure_form(h.route_path('edit_user_auth_tokens_delete', user_id=c.user.user_id), method='POST', request=request)} | |
42 | ${h.hidden('del_auth_token', auth_token.user_api_key_id)} |
|
42 | ${h.hidden('del_auth_token', auth_token.user_api_key_id)} | |
43 | <button class="btn btn-link btn-danger" type="submit" |
|
43 | <button class="btn btn-link btn-danger" type="submit" | |
44 | onclick="return confirm('${_('Confirm to remove this auth token: %s') % auth_token.token_obfuscated}');"> |
|
44 | onclick="return confirm('${_('Confirm to remove this auth token: %s') % auth_token.token_obfuscated}');"> | |
45 | ${_('Delete')} |
|
45 | ${_('Delete')} | |
46 | </button> |
|
46 | </button> | |
47 | ${h.end_form()} |
|
47 | ${h.end_form()} | |
48 | </td> |
|
48 | </td> | |
49 | </tr> |
|
49 | </tr> | |
50 | %endfor |
|
50 | %endfor | |
51 | %else: |
|
51 | %else: | |
52 | <tr><td><div class="ip">${_('No additional auth tokens specified')}</div></td></tr> |
|
52 | <tr><td><div class="ip">${_('No additional auth tokens specified')}</div></td></tr> | |
53 | %endif |
|
53 | %endif | |
54 | </table> |
|
54 | </table> | |
55 | </div> |
|
55 | </div> | |
56 |
|
56 | |||
57 | <div class="user_auth_tokens"> |
|
57 | <div class="user_auth_tokens"> | |
58 | ${h.secure_form(h.route_path('edit_user_auth_tokens_add', user_id=c.user.user_id), method='POST')} |
|
58 | ${h.secure_form(h.route_path('edit_user_auth_tokens_add', user_id=c.user.user_id), method='POST', request=request)} | |
59 | <div class="form form-vertical"> |
|
59 | <div class="form form-vertical"> | |
60 | <!-- fields --> |
|
60 | <!-- fields --> | |
61 | <div class="fields"> |
|
61 | <div class="fields"> | |
62 | <div class="field"> |
|
62 | <div class="field"> | |
63 | <div class="label"> |
|
63 | <div class="label"> | |
64 | <label for="new_email">${_('New authentication token')}:</label> |
|
64 | <label for="new_email">${_('New authentication token')}:</label> | |
65 | </div> |
|
65 | </div> | |
66 | <div class="input"> |
|
66 | <div class="input"> | |
67 | ${h.text('description', class_='medium', placeholder=_('Description'))} |
|
67 | ${h.text('description', class_='medium', placeholder=_('Description'))} | |
68 | ${h.select('lifetime', '', c.lifetime_options)} |
|
68 | ${h.select('lifetime', '', c.lifetime_options)} | |
69 | ${h.select('role', '', c.role_options)} |
|
69 | ${h.select('role', '', c.role_options)} | |
70 |
|
70 | |||
71 | % if c.allow_scoped_tokens: |
|
71 | % if c.allow_scoped_tokens: | |
72 | ${h.hidden('scope_repo_id')} |
|
72 | ${h.hidden('scope_repo_id')} | |
73 | % else: |
|
73 | % else: | |
74 | ${h.select('scope_repo_id_disabled', '', ['Scopes available in EE edition'], disabled='disabled')} |
|
74 | ${h.select('scope_repo_id_disabled', '', ['Scopes available in EE edition'], disabled='disabled')} | |
75 | % endif |
|
75 | % endif | |
76 | </div> |
|
76 | </div> | |
77 | <p class="help-block"> |
|
77 | <p class="help-block"> | |
78 | ${_('Repository scope works only with tokens with VCS type.')} |
|
78 | ${_('Repository scope works only with tokens with VCS type.')} | |
79 | </p> |
|
79 | </p> | |
80 | </div> |
|
80 | </div> | |
81 | <div class="buttons"> |
|
81 | <div class="buttons"> | |
82 | ${h.submit('save',_('Add'),class_="btn")} |
|
82 | ${h.submit('save',_('Add'),class_="btn")} | |
83 | ${h.reset('reset',_('Reset'),class_="btn")} |
|
83 | ${h.reset('reset',_('Reset'),class_="btn")} | |
84 | </div> |
|
84 | </div> | |
85 | </div> |
|
85 | </div> | |
86 | </div> |
|
86 | </div> | |
87 | ${h.end_form()} |
|
87 | ${h.end_form()} | |
88 | </div> |
|
88 | </div> | |
89 | </div> |
|
89 | </div> | |
90 | </div> |
|
90 | </div> | |
91 |
|
91 | |||
92 | <script> |
|
92 | <script> | |
93 |
|
93 | |||
94 | $(document).ready(function(){ |
|
94 | $(document).ready(function(){ | |
95 | var select2Options = { |
|
95 | var select2Options = { | |
96 | 'containerCssClass': "drop-menu", |
|
96 | 'containerCssClass': "drop-menu", | |
97 | 'dropdownCssClass': "drop-menu-dropdown", |
|
97 | 'dropdownCssClass': "drop-menu-dropdown", | |
98 | 'dropdownAutoWidth': true |
|
98 | 'dropdownAutoWidth': true | |
99 | }; |
|
99 | }; | |
100 | $("#lifetime").select2(select2Options); |
|
100 | $("#lifetime").select2(select2Options); | |
101 | $("#role").select2(select2Options); |
|
101 | $("#role").select2(select2Options); | |
102 |
|
102 | |||
103 | var repoFilter = function(data) { |
|
103 | var repoFilter = function(data) { | |
104 | var results = []; |
|
104 | var results = []; | |
105 |
|
105 | |||
106 | if (!data.results[0]) { |
|
106 | if (!data.results[0]) { | |
107 | return data |
|
107 | return data | |
108 | } |
|
108 | } | |
109 |
|
109 | |||
110 | $.each(data.results[0].children, function() { |
|
110 | $.each(data.results[0].children, function() { | |
111 | // replace name to ID for submision |
|
111 | // replace name to ID for submision | |
112 | this.id = this.obj.repo_id; |
|
112 | this.id = this.obj.repo_id; | |
113 | results.push(this); |
|
113 | results.push(this); | |
114 | }); |
|
114 | }); | |
115 |
|
115 | |||
116 | data.results[0].children = results; |
|
116 | data.results[0].children = results; | |
117 | return data; |
|
117 | return data; | |
118 | }; |
|
118 | }; | |
119 |
|
119 | |||
120 | $("#scope_repo_id_disabled").select2(select2Options); |
|
120 | $("#scope_repo_id_disabled").select2(select2Options); | |
121 |
|
121 | |||
122 | $("#scope_repo_id").select2({ |
|
122 | $("#scope_repo_id").select2({ | |
123 | cachedDataSource: {}, |
|
123 | cachedDataSource: {}, | |
124 | minimumInputLength: 2, |
|
124 | minimumInputLength: 2, | |
125 | placeholder: "${_('repository scope')}", |
|
125 | placeholder: "${_('repository scope')}", | |
126 | dropdownAutoWidth: true, |
|
126 | dropdownAutoWidth: true, | |
127 | containerCssClass: "drop-menu", |
|
127 | containerCssClass: "drop-menu", | |
128 | dropdownCssClass: "drop-menu-dropdown", |
|
128 | dropdownCssClass: "drop-menu-dropdown", | |
129 | formatResult: formatResult, |
|
129 | formatResult: formatResult, | |
130 | query: $.debounce(250, function(query){ |
|
130 | query: $.debounce(250, function(query){ | |
131 | self = this; |
|
131 | self = this; | |
132 | var cacheKey = query.term; |
|
132 | var cacheKey = query.term; | |
133 | var cachedData = self.cachedDataSource[cacheKey]; |
|
133 | var cachedData = self.cachedDataSource[cacheKey]; | |
134 |
|
134 | |||
135 | if (cachedData) { |
|
135 | if (cachedData) { | |
136 | query.callback({results: cachedData.results}); |
|
136 | query.callback({results: cachedData.results}); | |
137 | } else { |
|
137 | } else { | |
138 | $.ajax({ |
|
138 | $.ajax({ | |
139 | url: pyroutes.url('repo_list_data'), |
|
139 | url: pyroutes.url('repo_list_data'), | |
140 | data: {'query': query.term}, |
|
140 | data: {'query': query.term}, | |
141 | dataType: 'json', |
|
141 | dataType: 'json', | |
142 | type: 'GET', |
|
142 | type: 'GET', | |
143 | success: function(data) { |
|
143 | success: function(data) { | |
144 | data = repoFilter(data); |
|
144 | data = repoFilter(data); | |
145 | self.cachedDataSource[cacheKey] = data; |
|
145 | self.cachedDataSource[cacheKey] = data; | |
146 | query.callback({results: data.results}); |
|
146 | query.callback({results: data.results}); | |
147 | }, |
|
147 | }, | |
148 | error: function(data, textStatus, errorThrown) { |
|
148 | error: function(data, textStatus, errorThrown) { | |
149 | alert("Error while fetching entries.\nError code {0} ({1}).".format(data.status, data.statusText)); |
|
149 | alert("Error while fetching entries.\nError code {0} ({1}).".format(data.status, data.statusText)); | |
150 | } |
|
150 | } | |
151 | }) |
|
151 | }) | |
152 | } |
|
152 | } | |
153 | }) |
|
153 | }) | |
154 | }); |
|
154 | }); | |
155 |
|
155 | |||
156 | }); |
|
156 | }); | |
157 | </script> |
|
157 | </script> |
@@ -1,71 +1,71 b'' | |||||
1 | <%namespace name="base" file="/base/base.mako"/> |
|
1 | <%namespace name="base" file="/base/base.mako"/> | |
2 |
|
2 | |||
3 | <div class="panel panel-default"> |
|
3 | <div class="panel panel-default"> | |
4 | <div class="panel-heading"> |
|
4 | <div class="panel-heading"> | |
5 | <h3 class="panel-title">${_('Additional Email Addresses')}</h3> |
|
5 | <h3 class="panel-title">${_('Additional Email Addresses')}</h3> | |
6 | </div> |
|
6 | </div> | |
7 | <div class="panel-body"> |
|
7 | <div class="panel-body"> | |
8 | <div class="emails_wrap"> |
|
8 | <div class="emails_wrap"> | |
9 | <table class="rctable account_emails useremails"> |
|
9 | <table class="rctable account_emails useremails"> | |
10 | <tr> |
|
10 | <tr> | |
11 | <td class="td-user"> |
|
11 | <td class="td-user"> | |
12 | ${base.gravatar(c.user.email, 16)} |
|
12 | ${base.gravatar(c.user.email, 16)} | |
13 | <span class="user email">${c.user.email}</span> |
|
13 | <span class="user email">${c.user.email}</span> | |
14 | </td> |
|
14 | </td> | |
15 | <td class="td-tags"> |
|
15 | <td class="td-tags"> | |
16 | <span class="tag">${_('Primary')}</span> |
|
16 | <span class="tag">${_('Primary')}</span> | |
17 | </td> |
|
17 | </td> | |
18 | </tr> |
|
18 | </tr> | |
19 | %if c.user_email_map: |
|
19 | %if c.user_email_map: | |
20 | %for em in c.user_email_map: |
|
20 | %for em in c.user_email_map: | |
21 | <tr> |
|
21 | <tr> | |
22 | <td class="td-user"> |
|
22 | <td class="td-user"> | |
23 | ${base.gravatar(em.email, 16)} |
|
23 | ${base.gravatar(em.email, 16)} | |
24 | <span class="user email">${em.email}</span> |
|
24 | <span class="user email">${em.email}</span> | |
25 | </td> |
|
25 | </td> | |
26 | <td class="td-action"> |
|
26 | <td class="td-action"> | |
27 | ${h.secure_form(h.route_path('edit_user_emails_delete', user_id=c.user.user_id), method='POST')} |
|
27 | ${h.secure_form(h.route_path('edit_user_emails_delete', user_id=c.user.user_id), method='POST', request=request)} | |
28 | ${h.hidden('del_email_id', em.email_id)} |
|
28 | ${h.hidden('del_email_id', em.email_id)} | |
29 | <button class="btn btn-link btn-danger" type="submit" |
|
29 | <button class="btn btn-link btn-danger" type="submit" | |
30 | onclick="return confirm('${_('Confirm to delete this email: %s') % em.email}');"> |
|
30 | onclick="return confirm('${_('Confirm to delete this email: %s') % em.email}');"> | |
31 | ${_('Delete')} |
|
31 | ${_('Delete')} | |
32 | </button> |
|
32 | </button> | |
33 | ${h.end_form()} |
|
33 | ${h.end_form()} | |
34 | </td> |
|
34 | </td> | |
35 | </tr> |
|
35 | </tr> | |
36 | %endfor |
|
36 | %endfor | |
37 | %else: |
|
37 | %else: | |
38 | <tr class="noborder"> |
|
38 | <tr class="noborder"> | |
39 | <td colspan="3"> |
|
39 | <td colspan="3"> | |
40 | <div class="td-email"> |
|
40 | <div class="td-email"> | |
41 | ${_('No additional emails specified')} |
|
41 | ${_('No additional emails specified')} | |
42 | </div> |
|
42 | </div> | |
43 | </td> |
|
43 | </td> | |
44 | </tr> |
|
44 | </tr> | |
45 | %endif |
|
45 | %endif | |
46 | </table> |
|
46 | </table> | |
47 | </div> |
|
47 | </div> | |
48 |
|
48 | |||
49 | ${h.secure_form(h.route_path('edit_user_emails_add', user_id=c.user.user_id), method='POST')} |
|
49 | ${h.secure_form(h.route_path('edit_user_emails_add', user_id=c.user.user_id), method='POST', request=request)} | |
50 | <div class="form"> |
|
50 | <div class="form"> | |
51 | <!-- fields --> |
|
51 | <!-- fields --> | |
52 | <div class="fields"> |
|
52 | <div class="fields"> | |
53 | <div class="field"> |
|
53 | <div class="field"> | |
54 | <div class="label"> |
|
54 | <div class="label"> | |
55 | <label for="new_email">${_('New email address')}:</label> |
|
55 | <label for="new_email">${_('New email address')}:</label> | |
56 | </div> |
|
56 | </div> | |
57 | <div class="input"> |
|
57 | <div class="input"> | |
58 | ${h.text('new_email', class_='medium')} |
|
58 | ${h.text('new_email', class_='medium')} | |
59 | </div> |
|
59 | </div> | |
60 | </div> |
|
60 | </div> | |
61 | <div class="buttons"> |
|
61 | <div class="buttons"> | |
62 | ${h.submit('save',_('Add'),class_="btn btn-small")} |
|
62 | ${h.submit('save',_('Add'),class_="btn btn-small")} | |
63 | ${h.reset('reset',_('Reset'),class_="btn btn-small")} |
|
63 | ${h.reset('reset',_('Reset'),class_="btn btn-small")} | |
64 | </div> |
|
64 | </div> | |
65 | </div> |
|
65 | </div> | |
66 | </div> |
|
66 | </div> | |
67 | ${h.end_form()} |
|
67 | ${h.end_form()} | |
68 | </div> |
|
68 | </div> | |
69 | </div> |
|
69 | </div> | |
70 |
|
70 | |||
71 |
|
71 |
@@ -1,147 +1,147 b'' | |||||
1 | ## -*- coding: utf-8 -*- |
|
1 | ## -*- coding: utf-8 -*- | |
2 |
|
2 | |||
3 |
|
3 | |||
4 | <div class="panel panel-default"> |
|
4 | <div class="panel panel-default"> | |
5 | <div class="panel-heading"> |
|
5 | <div class="panel-heading"> | |
6 | <h3 class="panel-title">${_('User groups administration')}</h3> |
|
6 | <h3 class="panel-title">${_('User groups administration')}</h3> | |
7 | </div> |
|
7 | </div> | |
8 | <div class="panel-body"> |
|
8 | <div class="panel-body"> | |
9 | <div class="fields"> |
|
9 | <div class="fields"> | |
10 | <div class="field"> |
|
10 | <div class="field"> | |
11 | <div class="label label-checkbox"> |
|
11 | <div class="label label-checkbox"> | |
12 | <label for="users_group_active">${_('Add `%s` to user group') % c.user.username}:</label> |
|
12 | <label for="users_group_active">${_('Add `%s` to user group') % c.user.username}:</label> | |
13 | </div> |
|
13 | </div> | |
14 | <div class="input"> |
|
14 | <div class="input"> | |
15 | ${h.text('add_user_to_group', placeholder="user group name", class_="medium")} |
|
15 | ${h.text('add_user_to_group', placeholder="user group name", class_="medium")} | |
16 | </div> |
|
16 | </div> | |
17 |
|
17 | |||
18 | </div> |
|
18 | </div> | |
19 | </div> |
|
19 | </div> | |
20 |
|
20 | |||
21 | <div class="groups_management"> |
|
21 | <div class="groups_management"> | |
22 |
${h.secure_form(h.route_path('edit_user_groups_management_updates', user_id=c.user.user_id), method=' |
|
22 | ${h.secure_form(h.route_path('edit_user_groups_management_updates', user_id=c.user.user_id), method='POST', request=request)} | |
23 | <div id="repos_list_wrap"> |
|
23 | <div id="repos_list_wrap"> | |
24 | <table id="user_group_list_table" class="display"></table> |
|
24 | <table id="user_group_list_table" class="display"></table> | |
25 | </div> |
|
25 | </div> | |
26 | <div class="buttons"> |
|
26 | <div class="buttons"> | |
27 | ${h.submit('save',_('Save'),class_="btn")} |
|
27 | ${h.submit('save',_('Save'),class_="btn")} | |
28 | </div> |
|
28 | </div> | |
29 | ${h.end_form()} |
|
29 | ${h.end_form()} | |
30 | </div> |
|
30 | </div> | |
31 | </div> |
|
31 | </div> | |
32 | </div> |
|
32 | </div> | |
33 | <script> |
|
33 | <script> | |
34 | var api; |
|
34 | var api; | |
35 | $(document).ready(function() { |
|
35 | $(document).ready(function() { | |
36 |
|
36 | |||
37 | var get_datatable_count = function(){ |
|
37 | var get_datatable_count = function(){ | |
38 | $('#user_group_count').text(api.page.info().recordsDisplay); |
|
38 | $('#user_group_count').text(api.page.info().recordsDisplay); | |
39 | }; |
|
39 | }; | |
40 |
|
40 | |||
41 | $('#user_group_list_table').on('click', 'a.editor_remove', function (e) { |
|
41 | $('#user_group_list_table').on('click', 'a.editor_remove', function (e) { | |
42 | e.preventDefault(); |
|
42 | e.preventDefault(); | |
43 | var row = api.row($(this).closest('tr')); |
|
43 | var row = api.row($(this).closest('tr')); | |
44 | row.remove().draw(); |
|
44 | row.remove().draw(); | |
45 | } ); |
|
45 | } ); | |
46 |
|
46 | |||
47 | $('#user_group_list_table').DataTable({ |
|
47 | $('#user_group_list_table').DataTable({ | |
48 | data: ${c.groups|n}, |
|
48 | data: ${c.groups|n}, | |
49 | dom: 'rtp', |
|
49 | dom: 'rtp', | |
50 | pageLength: ${c.visual.admin_grid_items}, |
|
50 | pageLength: ${c.visual.admin_grid_items}, | |
51 | order: [[ 0, "asc" ]], |
|
51 | order: [[ 0, "asc" ]], | |
52 | columns: [ |
|
52 | columns: [ | |
53 | { data: {"_": "group_name", |
|
53 | { data: {"_": "group_name", | |
54 | "sort": "group_name"}, title: "${_('Name')}", className: "td-componentname," , |
|
54 | "sort": "group_name"}, title: "${_('Name')}", className: "td-componentname," , | |
55 | render: function (data,type,full,meta) |
|
55 | render: function (data,type,full,meta) | |
56 | {return '<div><i class="icon-group" title="User group">'+data+'</i></div>'}}, |
|
56 | {return '<div><i class="icon-group" title="User group">'+data+'</i></div>'}}, | |
57 |
|
57 | |||
58 | { data: {"_": "group_description", |
|
58 | { data: {"_": "group_description", | |
59 | "sort": "group_description"}, title: "${_('Description')}", className: "td-description" }, |
|
59 | "sort": "group_description"}, title: "${_('Description')}", className: "td-description" }, | |
60 | { data: {"_": "users_group_id"}, className: "td-user", |
|
60 | { data: {"_": "users_group_id"}, className: "td-user", | |
61 | render: function (data,type,full,meta) |
|
61 | render: function (data,type,full,meta) | |
62 | {return '<input type="hidden" name="users_group_id" value="'+data+'">'}}, |
|
62 | {return '<input type="hidden" name="users_group_id" value="'+data+'">'}}, | |
63 | { data: {"_": "active", |
|
63 | { data: {"_": "active", | |
64 | "sort": "active"}, title: "${_('Active')}", className: "td-active", className: "td-number"}, |
|
64 | "sort": "active"}, title: "${_('Active')}", className: "td-active", className: "td-number"}, | |
65 | { data: {"_": "owner_data"}, title: "${_('Owner')}", className: "td-user", |
|
65 | { data: {"_": "owner_data"}, title: "${_('Owner')}", className: "td-user", | |
66 | render: function (data,type,full,meta) |
|
66 | render: function (data,type,full,meta) | |
67 | {return '<div class="rc-user tooltip">'+ |
|
67 | {return '<div class="rc-user tooltip">'+ | |
68 | '<img class="gravatar" src="'+ data.owner_icon +'" height="16" width="16">'+ |
|
68 | '<img class="gravatar" src="'+ data.owner_icon +'" height="16" width="16">'+ | |
69 | data.owner +'</div>' |
|
69 | data.owner +'</div>' | |
70 | } |
|
70 | } | |
71 | }, |
|
71 | }, | |
72 | { data: null, |
|
72 | { data: null, | |
73 | title: "${_('Action')}", |
|
73 | title: "${_('Action')}", | |
74 | className: "td-action", |
|
74 | className: "td-action", | |
75 | defaultContent: '<a href="" class="btn btn-link btn-danger">Delete</a>' |
|
75 | defaultContent: '<a href="" class="btn btn-link btn-danger">Delete</a>' | |
76 | }, |
|
76 | }, | |
77 | ], |
|
77 | ], | |
78 | language: { |
|
78 | language: { | |
79 | paginate: DEFAULT_GRID_PAGINATION, |
|
79 | paginate: DEFAULT_GRID_PAGINATION, | |
80 | emptyTable: _gettext("No user groups available yet.") |
|
80 | emptyTable: _gettext("No user groups available yet.") | |
81 | }, |
|
81 | }, | |
82 | "initComplete": function( settings, json ) { |
|
82 | "initComplete": function( settings, json ) { | |
83 | var data_grid = $('#user_group_list_table').dataTable(); |
|
83 | var data_grid = $('#user_group_list_table').dataTable(); | |
84 | api = data_grid.api(); |
|
84 | api = data_grid.api(); | |
85 | get_datatable_count(); |
|
85 | get_datatable_count(); | |
86 | } |
|
86 | } | |
87 | }); |
|
87 | }); | |
88 |
|
88 | |||
89 | // update the counter when doing search |
|
89 | // update the counter when doing search | |
90 | $('#user_group_list_table').on( 'search.dt', function (e,settings) { |
|
90 | $('#user_group_list_table').on( 'search.dt', function (e,settings) { | |
91 | get_datatable_count(); |
|
91 | get_datatable_count(); | |
92 | }); |
|
92 | }); | |
93 |
|
93 | |||
94 | // filter, filter both grids |
|
94 | // filter, filter both grids | |
95 | $('#q_filter').on( 'keyup', function () { |
|
95 | $('#q_filter').on( 'keyup', function () { | |
96 | var user_api = $('#user_group_list_table').dataTable().api(); |
|
96 | var user_api = $('#user_group_list_table').dataTable().api(); | |
97 | user_api |
|
97 | user_api | |
98 | .columns(0) |
|
98 | .columns(0) | |
99 | .search(this.value) |
|
99 | .search(this.value) | |
100 | .draw(); |
|
100 | .draw(); | |
101 | }); |
|
101 | }); | |
102 |
|
102 | |||
103 | // refilter table if page load via back button |
|
103 | // refilter table if page load via back button | |
104 | $("#q_filter").trigger('keyup'); |
|
104 | $("#q_filter").trigger('keyup'); | |
105 |
|
105 | |||
106 | }); |
|
106 | }); | |
107 |
|
107 | |||
108 | $('#language').select2({ |
|
108 | $('#language').select2({ | |
109 | 'containerCssClass': "drop-menu", |
|
109 | 'containerCssClass': "drop-menu", | |
110 | 'dropdownCssClass': "drop-menu-dropdown", |
|
110 | 'dropdownCssClass': "drop-menu-dropdown", | |
111 | 'dropdownAutoWidth': true |
|
111 | 'dropdownAutoWidth': true | |
112 | }); |
|
112 | }); | |
113 |
|
113 | |||
114 |
|
114 | |||
115 |
|
115 | |||
116 | $(document).ready(function(){ |
|
116 | $(document).ready(function(){ | |
117 | $("#group_parent_id").select2({ |
|
117 | $("#group_parent_id").select2({ | |
118 | 'containerCssClass': "drop-menu", |
|
118 | 'containerCssClass': "drop-menu", | |
119 | 'dropdownCssClass': "drop-menu-dropdown", |
|
119 | 'dropdownCssClass': "drop-menu-dropdown", | |
120 | 'dropdownAutoWidth': true |
|
120 | 'dropdownAutoWidth': true | |
121 | }); |
|
121 | }); | |
122 |
|
122 | |||
123 | $('#add_user_to_group').autocomplete({ |
|
123 | $('#add_user_to_group').autocomplete({ | |
124 | serviceUrl: pyroutes.url('user_group_autocomplete_data'), |
|
124 | serviceUrl: pyroutes.url('user_group_autocomplete_data'), | |
125 | minChars:2, |
|
125 | minChars:2, | |
126 | maxHeight:400, |
|
126 | maxHeight:400, | |
127 | width:300, |
|
127 | width:300, | |
128 | deferRequestBy: 300, //miliseconds |
|
128 | deferRequestBy: 300, //miliseconds | |
129 | showNoSuggestionNotice: true, |
|
129 | showNoSuggestionNotice: true, | |
130 | params: { user_groups:true }, |
|
130 | params: { user_groups:true }, | |
131 | formatResult: autocompleteFormatResult, |
|
131 | formatResult: autocompleteFormatResult, | |
132 | lookupFilter: autocompleteFilterResult, |
|
132 | lookupFilter: autocompleteFilterResult, | |
133 | onSelect: function(element, suggestion){ |
|
133 | onSelect: function(element, suggestion){ | |
134 | var owner = {owner_icon: suggestion.owner_icon, owner:suggestion.owner}; |
|
134 | var owner = {owner_icon: suggestion.owner_icon, owner:suggestion.owner}; | |
135 | api.row.add( |
|
135 | api.row.add( | |
136 | {"active": suggestion.active, |
|
136 | {"active": suggestion.active, | |
137 | "owner_data": owner, |
|
137 | "owner_data": owner, | |
138 | "users_group_id": suggestion.id, |
|
138 | "users_group_id": suggestion.id, | |
139 | "group_description": suggestion.description, |
|
139 | "group_description": suggestion.description, | |
140 | "group_name": suggestion.value}).draw(); |
|
140 | "group_name": suggestion.value}).draw(); | |
141 | } |
|
141 | } | |
142 | }); |
|
142 | }); | |
143 | }) |
|
143 | }) | |
144 |
|
144 | |||
145 | </script> |
|
145 | </script> | |
146 |
|
146 | |||
147 |
|
147 |
@@ -1,78 +1,78 b'' | |||||
1 | <div class="panel panel-default"> |
|
1 | <div class="panel panel-default"> | |
2 | <div class="panel-heading"> |
|
2 | <div class="panel-heading"> | |
3 | <h3 class="panel-title">${_('Custom IP Whitelist')}</h3> |
|
3 | <h3 class="panel-title">${_('Custom IP Whitelist')}</h3> | |
4 | </div> |
|
4 | </div> | |
5 | <div class="panel-body"> |
|
5 | <div class="panel-body"> | |
6 | <div class="ips_wrap"> |
|
6 | <div class="ips_wrap"> | |
7 | <h5>${_('Current IP address')}: <code>${c.rhodecode_user.ip_addr}</code></h5> |
|
7 | <h5>${_('Current IP address')}: <code>${c.rhodecode_user.ip_addr}</code></h5> | |
8 | <table class="rctable ip-whitelist"> |
|
8 | <table class="rctable ip-whitelist"> | |
9 | <tr> |
|
9 | <tr> | |
10 | <th>${_('IP Address')}</th> |
|
10 | <th>${_('IP Address')}</th> | |
11 | <th>${_('IP Range')}</th> |
|
11 | <th>${_('IP Range')}</th> | |
12 | <th>${_('Description')}</th> |
|
12 | <th>${_('Description')}</th> | |
13 | <th></th> |
|
13 | <th></th> | |
14 | </tr> |
|
14 | </tr> | |
15 | %if c.default_user_ip_map and c.inherit_default_ips: |
|
15 | %if c.default_user_ip_map and c.inherit_default_ips: | |
16 | %for ip in c.default_user_ip_map: |
|
16 | %for ip in c.default_user_ip_map: | |
17 | <tr> |
|
17 | <tr> | |
18 | <td class="td-ip"><div class="ip">${ip.ip_addr}</div></td> |
|
18 | <td class="td-ip"><div class="ip">${ip.ip_addr}</div></td> | |
19 | <td class="td-iprange"><div class="ip">${h.ip_range(ip.ip_addr)}</div></td> |
|
19 | <td class="td-iprange"><div class="ip">${h.ip_range(ip.ip_addr)}</div></td> | |
20 | <td class="td-description">${h.literal(_('Inherited from %s') % h.link_to('*default*',h.url('admin_permissions_ips')))}</td> |
|
20 | <td class="td-description">${h.literal(_('Inherited from %s') % h.link_to('*default*',h.url('admin_permissions_ips')))}</td> | |
21 | <td></td> |
|
21 | <td></td> | |
22 | </tr> |
|
22 | </tr> | |
23 | %endfor |
|
23 | %endfor | |
24 | %endif |
|
24 | %endif | |
25 |
|
25 | |||
26 | %if c.user_ip_map: |
|
26 | %if c.user_ip_map: | |
27 | %for ip in c.user_ip_map: |
|
27 | %for ip in c.user_ip_map: | |
28 | <tr> |
|
28 | <tr> | |
29 | <td class="td-ip"><div class="ip">${ip.ip_addr}</div></td> |
|
29 | <td class="td-ip"><div class="ip">${ip.ip_addr}</div></td> | |
30 | <td class="td-iprange"><div class="ip">${h.ip_range(ip.ip_addr)}</div></td> |
|
30 | <td class="td-iprange"><div class="ip">${h.ip_range(ip.ip_addr)}</div></td> | |
31 | <td class="td-description"><div class="ip">${ip.description}</div></td> |
|
31 | <td class="td-description"><div class="ip">${ip.description}</div></td> | |
32 | <td class="td-action"> |
|
32 | <td class="td-action"> | |
33 | ${h.secure_form(h.route_path('edit_user_ips_delete', user_id=c.user.user_id), method='POST')} |
|
33 | ${h.secure_form(h.route_path('edit_user_ips_delete', user_id=c.user.user_id), method='POST', request=request)} | |
34 | ${h.hidden('del_ip_id', ip.ip_id)} |
|
34 | ${h.hidden('del_ip_id', ip.ip_id)} | |
35 | ${h.submit('remove_', _('Delete'),id="remove_ip_%s" % ip.ip_id, |
|
35 | ${h.submit('remove_', _('Delete'),id="remove_ip_%s" % ip.ip_id, | |
36 | class_="btn btn-link btn-danger", onclick="return confirm('"+_('Confirm to delete this ip: %s') % ip.ip_addr+"');")} |
|
36 | class_="btn btn-link btn-danger", onclick="return confirm('"+_('Confirm to delete this ip: %s') % ip.ip_addr+"');")} | |
37 | ${h.end_form()} |
|
37 | ${h.end_form()} | |
38 | </td> |
|
38 | </td> | |
39 | </tr> |
|
39 | </tr> | |
40 | %endfor |
|
40 | %endfor | |
41 | %endif |
|
41 | %endif | |
42 | %if not c.default_user_ip_map and not c.user_ip_map: |
|
42 | %if not c.default_user_ip_map and not c.user_ip_map: | |
43 | <tr> |
|
43 | <tr> | |
44 | <td><h2 class="ip">${_('All IP addresses are allowed')}</h2></td> |
|
44 | <td><h2 class="ip">${_('All IP addresses are allowed')}</h2></td> | |
45 | <td></td> |
|
45 | <td></td> | |
46 | <td></td> |
|
46 | <td></td> | |
47 | <td></td> |
|
47 | <td></td> | |
48 | </tr> |
|
48 | </tr> | |
49 | %endif |
|
49 | %endif | |
50 | </table> |
|
50 | </table> | |
51 | </div> |
|
51 | </div> | |
52 |
|
52 | |||
53 | <div> |
|
53 | <div> | |
54 | ${h.secure_form(h.route_path('edit_user_ips_add', user_id=c.user.user_id), method='POST')} |
|
54 | ${h.secure_form(h.route_path('edit_user_ips_add', user_id=c.user.user_id), method='POST', request=request)} | |
55 | <div class="form"> |
|
55 | <div class="form"> | |
56 | <!-- fields --> |
|
56 | <!-- fields --> | |
57 | <div class="fields"> |
|
57 | <div class="fields"> | |
58 | <div class="field"> |
|
58 | <div class="field"> | |
59 | <div class="label"> |
|
59 | <div class="label"> | |
60 | <label for="new_ip">${_('New IP Address')}:</label> |
|
60 | <label for="new_ip">${_('New IP Address')}:</label> | |
61 | </div> |
|
61 | </div> | |
62 | <div class="input"> |
|
62 | <div class="input"> | |
63 | ${h.text('new_ip')} ${h.text('description', placeholder=_('Description...'))} |
|
63 | ${h.text('new_ip')} ${h.text('description', placeholder=_('Description...'))} | |
64 | <span class="help-block">${_('Enter comma separated list of ip addresses like 127.0.0.1,\n' |
|
64 | <span class="help-block">${_('Enter comma separated list of ip addresses like 127.0.0.1,\n' | |
65 | 'or use a ip address with a mask 127.0.0.1/24, to create a network range.\n' |
|
65 | 'or use a ip address with a mask 127.0.0.1/24, to create a network range.\n' | |
66 | 'To specify multiple address range use 127.0.0.1-127.0.0.10 syntax')}</span> |
|
66 | 'To specify multiple address range use 127.0.0.1-127.0.0.10 syntax')}</span> | |
67 | </div> |
|
67 | </div> | |
68 | </div> |
|
68 | </div> | |
69 | <div class="buttons"> |
|
69 | <div class="buttons"> | |
70 | ${h.submit('save',_('Add'),class_="btn btn-small")} |
|
70 | ${h.submit('save',_('Add'),class_="btn btn-small")} | |
71 | ${h.reset('reset',_('Reset'),class_="btn btn-small")} |
|
71 | ${h.reset('reset',_('Reset'),class_="btn btn-small")} | |
72 | </div> |
|
72 | </div> | |
73 | </div> |
|
73 | </div> | |
74 | </div> |
|
74 | </div> | |
75 | ${h.end_form()} |
|
75 | ${h.end_form()} | |
76 | </div> |
|
76 | </div> | |
77 | </div> |
|
77 | </div> | |
78 | </div> |
|
78 | </div> |
@@ -1,604 +1,604 b'' | |||||
1 | ## -*- coding: utf-8 -*- |
|
1 | ## -*- coding: utf-8 -*- | |
2 | <%inherit file="root.mako"/> |
|
2 | <%inherit file="root.mako"/> | |
3 |
|
3 | |||
4 | <div class="outerwrapper"> |
|
4 | <div class="outerwrapper"> | |
5 | <!-- HEADER --> |
|
5 | <!-- HEADER --> | |
6 | <div class="header"> |
|
6 | <div class="header"> | |
7 | <div id="header-inner" class="wrapper"> |
|
7 | <div id="header-inner" class="wrapper"> | |
8 | <div id="logo"> |
|
8 | <div id="logo"> | |
9 | <div class="logo-wrapper"> |
|
9 | <div class="logo-wrapper"> | |
10 | <a href="${h.route_path('home')}"><img src="${h.asset('images/rhodecode-logo-white-216x60.png')}" alt="RhodeCode"/></a> |
|
10 | <a href="${h.route_path('home')}"><img src="${h.asset('images/rhodecode-logo-white-216x60.png')}" alt="RhodeCode"/></a> | |
11 | </div> |
|
11 | </div> | |
12 | %if c.rhodecode_name: |
|
12 | %if c.rhodecode_name: | |
13 | <div class="branding">- ${h.branding(c.rhodecode_name)}</div> |
|
13 | <div class="branding">- ${h.branding(c.rhodecode_name)}</div> | |
14 | %endif |
|
14 | %endif | |
15 | </div> |
|
15 | </div> | |
16 | <!-- MENU BAR NAV --> |
|
16 | <!-- MENU BAR NAV --> | |
17 | ${self.menu_bar_nav()} |
|
17 | ${self.menu_bar_nav()} | |
18 | <!-- END MENU BAR NAV --> |
|
18 | <!-- END MENU BAR NAV --> | |
19 | </div> |
|
19 | </div> | |
20 | </div> |
|
20 | </div> | |
21 | ${self.menu_bar_subnav()} |
|
21 | ${self.menu_bar_subnav()} | |
22 | <!-- END HEADER --> |
|
22 | <!-- END HEADER --> | |
23 |
|
23 | |||
24 | <!-- CONTENT --> |
|
24 | <!-- CONTENT --> | |
25 | <div id="content" class="wrapper"> |
|
25 | <div id="content" class="wrapper"> | |
26 |
|
26 | |||
27 | <rhodecode-toast id="notifications"></rhodecode-toast> |
|
27 | <rhodecode-toast id="notifications"></rhodecode-toast> | |
28 |
|
28 | |||
29 | <div class="main"> |
|
29 | <div class="main"> | |
30 | ${next.main()} |
|
30 | ${next.main()} | |
31 | </div> |
|
31 | </div> | |
32 | </div> |
|
32 | </div> | |
33 | <!-- END CONTENT --> |
|
33 | <!-- END CONTENT --> | |
34 |
|
34 | |||
35 | </div> |
|
35 | </div> | |
36 | <!-- FOOTER --> |
|
36 | <!-- FOOTER --> | |
37 | <div id="footer"> |
|
37 | <div id="footer"> | |
38 | <div id="footer-inner" class="title wrapper"> |
|
38 | <div id="footer-inner" class="title wrapper"> | |
39 | <div> |
|
39 | <div> | |
40 | <p class="footer-link-right"> |
|
40 | <p class="footer-link-right"> | |
41 | % if c.visual.show_version: |
|
41 | % if c.visual.show_version: | |
42 | RhodeCode Enterprise ${c.rhodecode_version} ${c.rhodecode_edition} |
|
42 | RhodeCode Enterprise ${c.rhodecode_version} ${c.rhodecode_edition} | |
43 | % endif |
|
43 | % endif | |
44 | © 2010-${h.datetime.today().year}, <a href="${h.route_url('rhodecode_official')}" target="_blank">RhodeCode GmbH</a>. All rights reserved. |
|
44 | © 2010-${h.datetime.today().year}, <a href="${h.route_url('rhodecode_official')}" target="_blank">RhodeCode GmbH</a>. All rights reserved. | |
45 | % if c.visual.rhodecode_support_url: |
|
45 | % if c.visual.rhodecode_support_url: | |
46 | <a href="${c.visual.rhodecode_support_url}" target="_blank">${_('Support')}</a> |
|
46 | <a href="${c.visual.rhodecode_support_url}" target="_blank">${_('Support')}</a> | |
47 | % endif |
|
47 | % endif | |
48 | </p> |
|
48 | </p> | |
49 | <% sid = 'block' if request.GET.get('showrcid') else 'none' %> |
|
49 | <% sid = 'block' if request.GET.get('showrcid') else 'none' %> | |
50 | <p class="server-instance" style="display:${sid}"> |
|
50 | <p class="server-instance" style="display:${sid}"> | |
51 | ## display hidden instance ID if specially defined |
|
51 | ## display hidden instance ID if specially defined | |
52 | % if c.rhodecode_instanceid: |
|
52 | % if c.rhodecode_instanceid: | |
53 | ${_('RhodeCode instance id: %s') % c.rhodecode_instanceid} |
|
53 | ${_('RhodeCode instance id: %s') % c.rhodecode_instanceid} | |
54 | % endif |
|
54 | % endif | |
55 | </p> |
|
55 | </p> | |
56 | </div> |
|
56 | </div> | |
57 | </div> |
|
57 | </div> | |
58 | </div> |
|
58 | </div> | |
59 |
|
59 | |||
60 | <!-- END FOOTER --> |
|
60 | <!-- END FOOTER --> | |
61 |
|
61 | |||
62 | ### MAKO DEFS ### |
|
62 | ### MAKO DEFS ### | |
63 |
|
63 | |||
64 | <%def name="menu_bar_subnav()"> |
|
64 | <%def name="menu_bar_subnav()"> | |
65 | </%def> |
|
65 | </%def> | |
66 |
|
66 | |||
67 | <%def name="breadcrumbs(class_='breadcrumbs')"> |
|
67 | <%def name="breadcrumbs(class_='breadcrumbs')"> | |
68 | <div class="${class_}"> |
|
68 | <div class="${class_}"> | |
69 | ${self.breadcrumbs_links()} |
|
69 | ${self.breadcrumbs_links()} | |
70 | </div> |
|
70 | </div> | |
71 | </%def> |
|
71 | </%def> | |
72 |
|
72 | |||
73 | <%def name="admin_menu()"> |
|
73 | <%def name="admin_menu()"> | |
74 | <ul class="admin_menu submenu"> |
|
74 | <ul class="admin_menu submenu"> | |
75 | <li><a href="${h.route_path('admin_audit_logs')}">${_('Admin audit logs')}</a></li> |
|
75 | <li><a href="${h.route_path('admin_audit_logs')}">${_('Admin audit logs')}</a></li> | |
76 | <li><a href="${h.url('repos')}">${_('Repositories')}</a></li> |
|
76 | <li><a href="${h.url('repos')}">${_('Repositories')}</a></li> | |
77 | <li><a href="${h.url('repo_groups')}">${_('Repository groups')}</a></li> |
|
77 | <li><a href="${h.url('repo_groups')}">${_('Repository groups')}</a></li> | |
78 | <li><a href="${h.route_path('users')}">${_('Users')}</a></li> |
|
78 | <li><a href="${h.route_path('users')}">${_('Users')}</a></li> | |
79 | <li><a href="${h.url('users_groups')}">${_('User groups')}</a></li> |
|
79 | <li><a href="${h.url('users_groups')}">${_('User groups')}</a></li> | |
80 | <li><a href="${h.url('admin_permissions_application')}">${_('Permissions')}</a></li> |
|
80 | <li><a href="${h.url('admin_permissions_application')}">${_('Permissions')}</a></li> | |
81 | <li><a href="${h.route_path('auth_home', traverse='')}">${_('Authentication')}</a></li> |
|
81 | <li><a href="${h.route_path('auth_home', traverse='')}">${_('Authentication')}</a></li> | |
82 | <li><a href="${h.route_path('global_integrations_home')}">${_('Integrations')}</a></li> |
|
82 | <li><a href="${h.route_path('global_integrations_home')}">${_('Integrations')}</a></li> | |
83 | <li><a href="${h.url('admin_defaults_repositories')}">${_('Defaults')}</a></li> |
|
83 | <li><a href="${h.url('admin_defaults_repositories')}">${_('Defaults')}</a></li> | |
84 | <li class="last"><a href="${h.url('admin_settings')}">${_('Settings')}</a></li> |
|
84 | <li class="last"><a href="${h.url('admin_settings')}">${_('Settings')}</a></li> | |
85 | </ul> |
|
85 | </ul> | |
86 | </%def> |
|
86 | </%def> | |
87 |
|
87 | |||
88 |
|
88 | |||
89 | <%def name="dt_info_panel(elements)"> |
|
89 | <%def name="dt_info_panel(elements)"> | |
90 | <dl class="dl-horizontal"> |
|
90 | <dl class="dl-horizontal"> | |
91 | %for dt, dd, title, show_items in elements: |
|
91 | %for dt, dd, title, show_items in elements: | |
92 | <dt>${dt}:</dt> |
|
92 | <dt>${dt}:</dt> | |
93 | <dd title="${h.tooltip(title)}"> |
|
93 | <dd title="${h.tooltip(title)}"> | |
94 | %if callable(dd): |
|
94 | %if callable(dd): | |
95 | ## allow lazy evaluation of elements |
|
95 | ## allow lazy evaluation of elements | |
96 | ${dd()} |
|
96 | ${dd()} | |
97 | %else: |
|
97 | %else: | |
98 | ${dd} |
|
98 | ${dd} | |
99 | %endif |
|
99 | %endif | |
100 | %if show_items: |
|
100 | %if show_items: | |
101 | <span class="btn-collapse" data-toggle="item-${h.md5_safe(dt)[:6]}-details">${_('Show More')} </span> |
|
101 | <span class="btn-collapse" data-toggle="item-${h.md5_safe(dt)[:6]}-details">${_('Show More')} </span> | |
102 | %endif |
|
102 | %endif | |
103 | </dd> |
|
103 | </dd> | |
104 |
|
104 | |||
105 | %if show_items: |
|
105 | %if show_items: | |
106 | <div class="collapsable-content" data-toggle="item-${h.md5_safe(dt)[:6]}-details" style="display: none"> |
|
106 | <div class="collapsable-content" data-toggle="item-${h.md5_safe(dt)[:6]}-details" style="display: none"> | |
107 | %for item in show_items: |
|
107 | %for item in show_items: | |
108 | <dt></dt> |
|
108 | <dt></dt> | |
109 | <dd>${item}</dd> |
|
109 | <dd>${item}</dd> | |
110 | %endfor |
|
110 | %endfor | |
111 | </div> |
|
111 | </div> | |
112 | %endif |
|
112 | %endif | |
113 |
|
113 | |||
114 | %endfor |
|
114 | %endfor | |
115 | </dl> |
|
115 | </dl> | |
116 | </%def> |
|
116 | </%def> | |
117 |
|
117 | |||
118 |
|
118 | |||
119 | <%def name="gravatar(email, size=16)"> |
|
119 | <%def name="gravatar(email, size=16)"> | |
120 | <% |
|
120 | <% | |
121 | if (size > 16): |
|
121 | if (size > 16): | |
122 | gravatar_class = 'gravatar gravatar-large' |
|
122 | gravatar_class = 'gravatar gravatar-large' | |
123 | else: |
|
123 | else: | |
124 | gravatar_class = 'gravatar' |
|
124 | gravatar_class = 'gravatar' | |
125 | %> |
|
125 | %> | |
126 | <%doc> |
|
126 | <%doc> | |
127 | TODO: johbo: For now we serve double size images to make it smooth |
|
127 | TODO: johbo: For now we serve double size images to make it smooth | |
128 | for retina. This is how it worked until now. Should be replaced |
|
128 | for retina. This is how it worked until now. Should be replaced | |
129 | with a better solution at some point. |
|
129 | with a better solution at some point. | |
130 | </%doc> |
|
130 | </%doc> | |
131 | <img class="${gravatar_class}" src="${h.gravatar_url(email, size * 2)}" height="${size}" width="${size}"> |
|
131 | <img class="${gravatar_class}" src="${h.gravatar_url(email, size * 2)}" height="${size}" width="${size}"> | |
132 | </%def> |
|
132 | </%def> | |
133 |
|
133 | |||
134 |
|
134 | |||
135 | <%def name="gravatar_with_user(contact, size=16, show_disabled=False)"> |
|
135 | <%def name="gravatar_with_user(contact, size=16, show_disabled=False)"> | |
136 | <% email = h.email_or_none(contact) %> |
|
136 | <% email = h.email_or_none(contact) %> | |
137 | <div class="rc-user tooltip" title="${h.tooltip(h.author_string(email))}"> |
|
137 | <div class="rc-user tooltip" title="${h.tooltip(h.author_string(email))}"> | |
138 | ${self.gravatar(email, size)} |
|
138 | ${self.gravatar(email, size)} | |
139 | <span class="${'user user-disabled' if show_disabled else 'user'}"> ${h.link_to_user(contact)}</span> |
|
139 | <span class="${'user user-disabled' if show_disabled else 'user'}"> ${h.link_to_user(contact)}</span> | |
140 | </div> |
|
140 | </div> | |
141 | </%def> |
|
141 | </%def> | |
142 |
|
142 | |||
143 |
|
143 | |||
144 | ## admin menu used for people that have some admin resources |
|
144 | ## admin menu used for people that have some admin resources | |
145 | <%def name="admin_menu_simple(repositories=None, repository_groups=None, user_groups=None)"> |
|
145 | <%def name="admin_menu_simple(repositories=None, repository_groups=None, user_groups=None)"> | |
146 | <ul class="submenu"> |
|
146 | <ul class="submenu"> | |
147 | %if repositories: |
|
147 | %if repositories: | |
148 | <li class="local-admin-repos"><a href="${h.url('repos')}">${_('Repositories')}</a></li> |
|
148 | <li class="local-admin-repos"><a href="${h.url('repos')}">${_('Repositories')}</a></li> | |
149 | %endif |
|
149 | %endif | |
150 | %if repository_groups: |
|
150 | %if repository_groups: | |
151 | <li class="local-admin-repo-groups"><a href="${h.url('repo_groups')}">${_('Repository groups')}</a></li> |
|
151 | <li class="local-admin-repo-groups"><a href="${h.url('repo_groups')}">${_('Repository groups')}</a></li> | |
152 | %endif |
|
152 | %endif | |
153 | %if user_groups: |
|
153 | %if user_groups: | |
154 | <li class="local-admin-user-groups"><a href="${h.url('users_groups')}">${_('User groups')}</a></li> |
|
154 | <li class="local-admin-user-groups"><a href="${h.url('users_groups')}">${_('User groups')}</a></li> | |
155 | %endif |
|
155 | %endif | |
156 | </ul> |
|
156 | </ul> | |
157 | </%def> |
|
157 | </%def> | |
158 |
|
158 | |||
159 | <%def name="repo_page_title(repo_instance)"> |
|
159 | <%def name="repo_page_title(repo_instance)"> | |
160 | <div class="title-content"> |
|
160 | <div class="title-content"> | |
161 | <div class="title-main"> |
|
161 | <div class="title-main"> | |
162 | ## SVN/HG/GIT icons |
|
162 | ## SVN/HG/GIT icons | |
163 | %if h.is_hg(repo_instance): |
|
163 | %if h.is_hg(repo_instance): | |
164 | <i class="icon-hg"></i> |
|
164 | <i class="icon-hg"></i> | |
165 | %endif |
|
165 | %endif | |
166 | %if h.is_git(repo_instance): |
|
166 | %if h.is_git(repo_instance): | |
167 | <i class="icon-git"></i> |
|
167 | <i class="icon-git"></i> | |
168 | %endif |
|
168 | %endif | |
169 | %if h.is_svn(repo_instance): |
|
169 | %if h.is_svn(repo_instance): | |
170 | <i class="icon-svn"></i> |
|
170 | <i class="icon-svn"></i> | |
171 | %endif |
|
171 | %endif | |
172 |
|
172 | |||
173 | ## public/private |
|
173 | ## public/private | |
174 | %if repo_instance.private: |
|
174 | %if repo_instance.private: | |
175 | <i class="icon-repo-private"></i> |
|
175 | <i class="icon-repo-private"></i> | |
176 | %else: |
|
176 | %else: | |
177 | <i class="icon-repo-public"></i> |
|
177 | <i class="icon-repo-public"></i> | |
178 | %endif |
|
178 | %endif | |
179 |
|
179 | |||
180 | ## repo name with group name |
|
180 | ## repo name with group name | |
181 | ${h.breadcrumb_repo_link(c.rhodecode_db_repo)} |
|
181 | ${h.breadcrumb_repo_link(c.rhodecode_db_repo)} | |
182 |
|
182 | |||
183 | </div> |
|
183 | </div> | |
184 |
|
184 | |||
185 | ## FORKED |
|
185 | ## FORKED | |
186 | %if repo_instance.fork: |
|
186 | %if repo_instance.fork: | |
187 | <p> |
|
187 | <p> | |
188 | <i class="icon-code-fork"></i> ${_('Fork of')} |
|
188 | <i class="icon-code-fork"></i> ${_('Fork of')} | |
189 | <a href="${h.route_path('repo_summary',repo_name=repo_instance.fork.repo_name)}">${repo_instance.fork.repo_name}</a> |
|
189 | <a href="${h.route_path('repo_summary',repo_name=repo_instance.fork.repo_name)}">${repo_instance.fork.repo_name}</a> | |
190 | </p> |
|
190 | </p> | |
191 | %endif |
|
191 | %endif | |
192 |
|
192 | |||
193 | ## IMPORTED FROM REMOTE |
|
193 | ## IMPORTED FROM REMOTE | |
194 | %if repo_instance.clone_uri: |
|
194 | %if repo_instance.clone_uri: | |
195 | <p> |
|
195 | <p> | |
196 | <i class="icon-code-fork"></i> ${_('Clone from')} |
|
196 | <i class="icon-code-fork"></i> ${_('Clone from')} | |
197 | <a href="${h.url(h.safe_str(h.hide_credentials(repo_instance.clone_uri)))}">${h.hide_credentials(repo_instance.clone_uri)}</a> |
|
197 | <a href="${h.url(h.safe_str(h.hide_credentials(repo_instance.clone_uri)))}">${h.hide_credentials(repo_instance.clone_uri)}</a> | |
198 | </p> |
|
198 | </p> | |
199 | %endif |
|
199 | %endif | |
200 |
|
200 | |||
201 | ## LOCKING STATUS |
|
201 | ## LOCKING STATUS | |
202 | %if repo_instance.locked[0]: |
|
202 | %if repo_instance.locked[0]: | |
203 | <p class="locking_locked"> |
|
203 | <p class="locking_locked"> | |
204 | <i class="icon-repo-lock"></i> |
|
204 | <i class="icon-repo-lock"></i> | |
205 | ${_('Repository locked by %(user)s') % {'user': h.person_by_id(repo_instance.locked[0])}} |
|
205 | ${_('Repository locked by %(user)s') % {'user': h.person_by_id(repo_instance.locked[0])}} | |
206 | </p> |
|
206 | </p> | |
207 | %elif repo_instance.enable_locking: |
|
207 | %elif repo_instance.enable_locking: | |
208 | <p class="locking_unlocked"> |
|
208 | <p class="locking_unlocked"> | |
209 | <i class="icon-repo-unlock"></i> |
|
209 | <i class="icon-repo-unlock"></i> | |
210 | ${_('Repository not locked. Pull repository to lock it.')} |
|
210 | ${_('Repository not locked. Pull repository to lock it.')} | |
211 | </p> |
|
211 | </p> | |
212 | %endif |
|
212 | %endif | |
213 |
|
213 | |||
214 | </div> |
|
214 | </div> | |
215 | </%def> |
|
215 | </%def> | |
216 |
|
216 | |||
217 | <%def name="repo_menu(active=None)"> |
|
217 | <%def name="repo_menu(active=None)"> | |
218 | <% |
|
218 | <% | |
219 | def is_active(selected): |
|
219 | def is_active(selected): | |
220 | if selected == active: |
|
220 | if selected == active: | |
221 | return "active" |
|
221 | return "active" | |
222 | %> |
|
222 | %> | |
223 |
|
223 | |||
224 | <!--- CONTEXT BAR --> |
|
224 | <!--- CONTEXT BAR --> | |
225 | <div id="context-bar"> |
|
225 | <div id="context-bar"> | |
226 | <div class="wrapper"> |
|
226 | <div class="wrapper"> | |
227 | <ul id="context-pages" class="horizontal-list navigation"> |
|
227 | <ul id="context-pages" class="horizontal-list navigation"> | |
228 | <li class="${is_active('summary')}"><a class="menulink" href="${h.route_path('repo_summary', repo_name=c.repo_name)}"><div class="menulabel">${_('Summary')}</div></a></li> |
|
228 | <li class="${is_active('summary')}"><a class="menulink" href="${h.route_path('repo_summary', repo_name=c.repo_name)}"><div class="menulabel">${_('Summary')}</div></a></li> | |
229 | <li class="${is_active('changelog')}"><a class="menulink" href="${h.url('changelog_home', repo_name=c.repo_name)}"><div class="menulabel">${_('Changelog')}</div></a></li> |
|
229 | <li class="${is_active('changelog')}"><a class="menulink" href="${h.url('changelog_home', repo_name=c.repo_name)}"><div class="menulabel">${_('Changelog')}</div></a></li> | |
230 | <li class="${is_active('files')}"><a class="menulink" href="${h.url('files_home', repo_name=c.repo_name, revision=c.rhodecode_db_repo.landing_rev[1])}"><div class="menulabel">${_('Files')}</div></a></li> |
|
230 | <li class="${is_active('files')}"><a class="menulink" href="${h.url('files_home', repo_name=c.repo_name, revision=c.rhodecode_db_repo.landing_rev[1])}"><div class="menulabel">${_('Files')}</div></a></li> | |
231 | <li class="${is_active('compare')}"> |
|
231 | <li class="${is_active('compare')}"> | |
232 | <a class="menulink" href="${h.url('compare_home',repo_name=c.repo_name)}"><div class="menulabel">${_('Compare')}</div></a> |
|
232 | <a class="menulink" href="${h.url('compare_home',repo_name=c.repo_name)}"><div class="menulabel">${_('Compare')}</div></a> | |
233 | </li> |
|
233 | </li> | |
234 | ## TODO: anderson: ideally it would have a function on the scm_instance "enable_pullrequest() and enable_fork()" |
|
234 | ## TODO: anderson: ideally it would have a function on the scm_instance "enable_pullrequest() and enable_fork()" | |
235 | %if c.rhodecode_db_repo.repo_type in ['git','hg']: |
|
235 | %if c.rhodecode_db_repo.repo_type in ['git','hg']: | |
236 | <li class="${is_active('showpullrequest')}"> |
|
236 | <li class="${is_active('showpullrequest')}"> | |
237 | <a class="menulink" href="${h.route_path('pullrequest_show_all', repo_name=c.repo_name)}" title="${h.tooltip(_('Show Pull Requests for %s') % c.repo_name)}"> |
|
237 | <a class="menulink" href="${h.route_path('pullrequest_show_all', repo_name=c.repo_name)}" title="${h.tooltip(_('Show Pull Requests for %s') % c.repo_name)}"> | |
238 | %if c.repository_pull_requests: |
|
238 | %if c.repository_pull_requests: | |
239 | <span class="pr_notifications">${c.repository_pull_requests}</span> |
|
239 | <span class="pr_notifications">${c.repository_pull_requests}</span> | |
240 | %endif |
|
240 | %endif | |
241 | <div class="menulabel">${_('Pull Requests')}</div> |
|
241 | <div class="menulabel">${_('Pull Requests')}</div> | |
242 | </a> |
|
242 | </a> | |
243 | </li> |
|
243 | </li> | |
244 | %endif |
|
244 | %endif | |
245 | <li class="${is_active('options')}"> |
|
245 | <li class="${is_active('options')}"> | |
246 | <a class="menulink dropdown"> |
|
246 | <a class="menulink dropdown"> | |
247 | <div class="menulabel">${_('Options')} <div class="show_more"></div></div> |
|
247 | <div class="menulabel">${_('Options')} <div class="show_more"></div></div> | |
248 | </a> |
|
248 | </a> | |
249 | <ul class="submenu"> |
|
249 | <ul class="submenu"> | |
250 | %if h.HasRepoPermissionAll('repository.admin')(c.repo_name): |
|
250 | %if h.HasRepoPermissionAll('repository.admin')(c.repo_name): | |
251 | <li><a href="${h.route_path('edit_repo',repo_name=c.repo_name)}">${_('Settings')}</a></li> |
|
251 | <li><a href="${h.route_path('edit_repo',repo_name=c.repo_name)}">${_('Settings')}</a></li> | |
252 | %endif |
|
252 | %endif | |
253 | %if c.rhodecode_db_repo.fork: |
|
253 | %if c.rhodecode_db_repo.fork: | |
254 | <li><a href="${h.url('compare_url',repo_name=c.rhodecode_db_repo.fork.repo_name,source_ref_type=c.rhodecode_db_repo.landing_rev[0],source_ref=c.rhodecode_db_repo.landing_rev[1], target_repo=c.repo_name,target_ref_type='branch' if request.GET.get('branch') else c.rhodecode_db_repo.landing_rev[0],target_ref=request.GET.get('branch') or c.rhodecode_db_repo.landing_rev[1], merge=1)}"> |
|
254 | <li><a href="${h.url('compare_url',repo_name=c.rhodecode_db_repo.fork.repo_name,source_ref_type=c.rhodecode_db_repo.landing_rev[0],source_ref=c.rhodecode_db_repo.landing_rev[1], target_repo=c.repo_name,target_ref_type='branch' if request.GET.get('branch') else c.rhodecode_db_repo.landing_rev[0],target_ref=request.GET.get('branch') or c.rhodecode_db_repo.landing_rev[1], merge=1)}"> | |
255 | ${_('Compare fork')}</a></li> |
|
255 | ${_('Compare fork')}</a></li> | |
256 | %endif |
|
256 | %endif | |
257 |
|
257 | |||
258 | <li><a href="${h.route_path('search_repo',repo_name=c.repo_name)}">${_('Search')}</a></li> |
|
258 | <li><a href="${h.route_path('search_repo',repo_name=c.repo_name)}">${_('Search')}</a></li> | |
259 |
|
259 | |||
260 | %if h.HasRepoPermissionAny('repository.write','repository.admin')(c.repo_name) and c.rhodecode_db_repo.enable_locking: |
|
260 | %if h.HasRepoPermissionAny('repository.write','repository.admin')(c.repo_name) and c.rhodecode_db_repo.enable_locking: | |
261 | %if c.rhodecode_db_repo.locked[0]: |
|
261 | %if c.rhodecode_db_repo.locked[0]: | |
262 | <li><a class="locking_del" href="${h.url('toggle_locking',repo_name=c.repo_name)}">${_('Unlock')}</a></li> |
|
262 | <li><a class="locking_del" href="${h.url('toggle_locking',repo_name=c.repo_name)}">${_('Unlock')}</a></li> | |
263 | %else: |
|
263 | %else: | |
264 | <li><a class="locking_add" href="${h.url('toggle_locking',repo_name=c.repo_name)}">${_('Lock')}</a></li> |
|
264 | <li><a class="locking_add" href="${h.url('toggle_locking',repo_name=c.repo_name)}">${_('Lock')}</a></li> | |
265 | %endif |
|
265 | %endif | |
266 | %endif |
|
266 | %endif | |
267 | %if c.rhodecode_user.username != h.DEFAULT_USER: |
|
267 | %if c.rhodecode_user.username != h.DEFAULT_USER: | |
268 | %if c.rhodecode_db_repo.repo_type in ['git','hg']: |
|
268 | %if c.rhodecode_db_repo.repo_type in ['git','hg']: | |
269 | <li><a href="${h.url('repo_fork_home',repo_name=c.repo_name)}">${_('Fork')}</a></li> |
|
269 | <li><a href="${h.url('repo_fork_home',repo_name=c.repo_name)}">${_('Fork')}</a></li> | |
270 | <li><a href="${h.url('pullrequest_home',repo_name=c.repo_name)}">${_('Create Pull Request')}</a></li> |
|
270 | <li><a href="${h.url('pullrequest_home',repo_name=c.repo_name)}">${_('Create Pull Request')}</a></li> | |
271 | %endif |
|
271 | %endif | |
272 | %endif |
|
272 | %endif | |
273 | </ul> |
|
273 | </ul> | |
274 | </li> |
|
274 | </li> | |
275 | </ul> |
|
275 | </ul> | |
276 | </div> |
|
276 | </div> | |
277 | <div class="clear"></div> |
|
277 | <div class="clear"></div> | |
278 | </div> |
|
278 | </div> | |
279 | <!--- END CONTEXT BAR --> |
|
279 | <!--- END CONTEXT BAR --> | |
280 |
|
280 | |||
281 | </%def> |
|
281 | </%def> | |
282 |
|
282 | |||
283 | <%def name="usermenu(active=False)"> |
|
283 | <%def name="usermenu(active=False)"> | |
284 | ## USER MENU |
|
284 | ## USER MENU | |
285 | <li id="quick_login_li" class="${'active' if active else ''}"> |
|
285 | <li id="quick_login_li" class="${'active' if active else ''}"> | |
286 | <a id="quick_login_link" class="menulink childs"> |
|
286 | <a id="quick_login_link" class="menulink childs"> | |
287 | ${gravatar(c.rhodecode_user.email, 20)} |
|
287 | ${gravatar(c.rhodecode_user.email, 20)} | |
288 | <span class="user"> |
|
288 | <span class="user"> | |
289 | %if c.rhodecode_user.username != h.DEFAULT_USER: |
|
289 | %if c.rhodecode_user.username != h.DEFAULT_USER: | |
290 | <span class="menu_link_user">${c.rhodecode_user.username}</span><div class="show_more"></div> |
|
290 | <span class="menu_link_user">${c.rhodecode_user.username}</span><div class="show_more"></div> | |
291 | %else: |
|
291 | %else: | |
292 | <span>${_('Sign in')}</span> |
|
292 | <span>${_('Sign in')}</span> | |
293 | %endif |
|
293 | %endif | |
294 | </span> |
|
294 | </span> | |
295 | </a> |
|
295 | </a> | |
296 |
|
296 | |||
297 | <div class="user-menu submenu"> |
|
297 | <div class="user-menu submenu"> | |
298 | <div id="quick_login"> |
|
298 | <div id="quick_login"> | |
299 | %if c.rhodecode_user.username == h.DEFAULT_USER: |
|
299 | %if c.rhodecode_user.username == h.DEFAULT_USER: | |
300 | <h4>${_('Sign in to your account')}</h4> |
|
300 | <h4>${_('Sign in to your account')}</h4> | |
301 | ${h.form(h.route_path('login', _query={'came_from': h.url.current()}), needs_csrf_token=False)} |
|
301 | ${h.form(h.route_path('login', _query={'came_from': h.url.current()}), needs_csrf_token=False)} | |
302 | <div class="form form-vertical"> |
|
302 | <div class="form form-vertical"> | |
303 | <div class="fields"> |
|
303 | <div class="fields"> | |
304 | <div class="field"> |
|
304 | <div class="field"> | |
305 | <div class="label"> |
|
305 | <div class="label"> | |
306 | <label for="username">${_('Username')}:</label> |
|
306 | <label for="username">${_('Username')}:</label> | |
307 | </div> |
|
307 | </div> | |
308 | <div class="input"> |
|
308 | <div class="input"> | |
309 | ${h.text('username',class_='focus',tabindex=1)} |
|
309 | ${h.text('username',class_='focus',tabindex=1)} | |
310 | </div> |
|
310 | </div> | |
311 |
|
311 | |||
312 | </div> |
|
312 | </div> | |
313 | <div class="field"> |
|
313 | <div class="field"> | |
314 | <div class="label"> |
|
314 | <div class="label"> | |
315 | <label for="password">${_('Password')}:</label> |
|
315 | <label for="password">${_('Password')}:</label> | |
316 | %if h.HasPermissionAny('hg.password_reset.enabled')(): |
|
316 | %if h.HasPermissionAny('hg.password_reset.enabled')(): | |
317 | <span class="forgot_password">${h.link_to(_('(Forgot password?)'),h.route_path('reset_password'), class_='pwd_reset')}</span> |
|
317 | <span class="forgot_password">${h.link_to(_('(Forgot password?)'),h.route_path('reset_password'), class_='pwd_reset')}</span> | |
318 | %endif |
|
318 | %endif | |
319 | </div> |
|
319 | </div> | |
320 | <div class="input"> |
|
320 | <div class="input"> | |
321 | ${h.password('password',class_='focus',tabindex=2)} |
|
321 | ${h.password('password',class_='focus',tabindex=2)} | |
322 | </div> |
|
322 | </div> | |
323 | </div> |
|
323 | </div> | |
324 | <div class="buttons"> |
|
324 | <div class="buttons"> | |
325 | <div class="register"> |
|
325 | <div class="register"> | |
326 | %if h.HasPermissionAny('hg.admin', 'hg.register.auto_activate', 'hg.register.manual_activate')(): |
|
326 | %if h.HasPermissionAny('hg.admin', 'hg.register.auto_activate', 'hg.register.manual_activate')(): | |
327 | ${h.link_to(_("Don't have an account?"),h.route_path('register'))} <br/> |
|
327 | ${h.link_to(_("Don't have an account?"),h.route_path('register'))} <br/> | |
328 | %endif |
|
328 | %endif | |
329 | ${h.link_to(_("Using external auth? Sign In here."),h.route_path('login'))} |
|
329 | ${h.link_to(_("Using external auth? Sign In here."),h.route_path('login'))} | |
330 | </div> |
|
330 | </div> | |
331 | <div class="submit"> |
|
331 | <div class="submit"> | |
332 | ${h.submit('sign_in',_('Sign In'),class_="btn btn-small",tabindex=3)} |
|
332 | ${h.submit('sign_in',_('Sign In'),class_="btn btn-small",tabindex=3)} | |
333 | </div> |
|
333 | </div> | |
334 | </div> |
|
334 | </div> | |
335 | </div> |
|
335 | </div> | |
336 | </div> |
|
336 | </div> | |
337 | ${h.end_form()} |
|
337 | ${h.end_form()} | |
338 | %else: |
|
338 | %else: | |
339 | <div class=""> |
|
339 | <div class=""> | |
340 | <div class="big_gravatar">${gravatar(c.rhodecode_user.email, 48)}</div> |
|
340 | <div class="big_gravatar">${gravatar(c.rhodecode_user.email, 48)}</div> | |
341 | <div class="full_name">${c.rhodecode_user.full_name_or_username}</div> |
|
341 | <div class="full_name">${c.rhodecode_user.full_name_or_username}</div> | |
342 | <div class="email">${c.rhodecode_user.email}</div> |
|
342 | <div class="email">${c.rhodecode_user.email}</div> | |
343 | </div> |
|
343 | </div> | |
344 | <div class=""> |
|
344 | <div class=""> | |
345 | <ol class="links"> |
|
345 | <ol class="links"> | |
346 | <li>${h.link_to(_(u'My account'),h.route_path('my_account_profile'))}</li> |
|
346 | <li>${h.link_to(_(u'My account'),h.route_path('my_account_profile'))}</li> | |
347 | % if c.rhodecode_user.personal_repo_group: |
|
347 | % if c.rhodecode_user.personal_repo_group: | |
348 | <li>${h.link_to(_(u'My personal group'), h.route_path('repo_group_home', repo_group_name=c.rhodecode_user.personal_repo_group.group_name))}</li> |
|
348 | <li>${h.link_to(_(u'My personal group'), h.route_path('repo_group_home', repo_group_name=c.rhodecode_user.personal_repo_group.group_name))}</li> | |
349 | % endif |
|
349 | % endif | |
350 | <li class="logout"> |
|
350 | <li class="logout"> | |
351 | ${h.secure_form(h.route_path('logout'))} |
|
351 | ${h.secure_form(h.route_path('logout'), request=request)} | |
352 | ${h.submit('log_out', _(u'Sign Out'),class_="btn btn-primary")} |
|
352 | ${h.submit('log_out', _(u'Sign Out'),class_="btn btn-primary")} | |
353 | ${h.end_form()} |
|
353 | ${h.end_form()} | |
354 | </li> |
|
354 | </li> | |
355 | </ol> |
|
355 | </ol> | |
356 | </div> |
|
356 | </div> | |
357 | %endif |
|
357 | %endif | |
358 | </div> |
|
358 | </div> | |
359 | </div> |
|
359 | </div> | |
360 | %if c.rhodecode_user.username != h.DEFAULT_USER: |
|
360 | %if c.rhodecode_user.username != h.DEFAULT_USER: | |
361 | <div class="pill_container"> |
|
361 | <div class="pill_container"> | |
362 | % if c.unread_notifications == 0: |
|
362 | % if c.unread_notifications == 0: | |
363 | <a class="menu_link_notifications empty" href="${h.url('notifications')}">${c.unread_notifications}</a> |
|
363 | <a class="menu_link_notifications empty" href="${h.url('notifications')}">${c.unread_notifications}</a> | |
364 | % else: |
|
364 | % else: | |
365 | <a class="menu_link_notifications" href="${h.url('notifications')}">${c.unread_notifications}</a> |
|
365 | <a class="menu_link_notifications" href="${h.url('notifications')}">${c.unread_notifications}</a> | |
366 | % endif |
|
366 | % endif | |
367 | </div> |
|
367 | </div> | |
368 | % endif |
|
368 | % endif | |
369 | </li> |
|
369 | </li> | |
370 | </%def> |
|
370 | </%def> | |
371 |
|
371 | |||
372 | <%def name="menu_items(active=None)"> |
|
372 | <%def name="menu_items(active=None)"> | |
373 | <% |
|
373 | <% | |
374 | def is_active(selected): |
|
374 | def is_active(selected): | |
375 | if selected == active: |
|
375 | if selected == active: | |
376 | return "active" |
|
376 | return "active" | |
377 | return "" |
|
377 | return "" | |
378 | %> |
|
378 | %> | |
379 | <ul id="quick" class="main_nav navigation horizontal-list"> |
|
379 | <ul id="quick" class="main_nav navigation horizontal-list"> | |
380 | <!-- repo switcher --> |
|
380 | <!-- repo switcher --> | |
381 | <li class="${is_active('repositories')} repo_switcher_li has_select2"> |
|
381 | <li class="${is_active('repositories')} repo_switcher_li has_select2"> | |
382 | <input id="repo_switcher" name="repo_switcher" type="hidden"> |
|
382 | <input id="repo_switcher" name="repo_switcher" type="hidden"> | |
383 | </li> |
|
383 | </li> | |
384 |
|
384 | |||
385 | ## ROOT MENU |
|
385 | ## ROOT MENU | |
386 | %if c.rhodecode_user.username != h.DEFAULT_USER: |
|
386 | %if c.rhodecode_user.username != h.DEFAULT_USER: | |
387 | <li class="${is_active('journal')}"> |
|
387 | <li class="${is_active('journal')}"> | |
388 | <a class="menulink" title="${_('Show activity journal')}" href="${h.url('journal')}"> |
|
388 | <a class="menulink" title="${_('Show activity journal')}" href="${h.url('journal')}"> | |
389 | <div class="menulabel">${_('Journal')}</div> |
|
389 | <div class="menulabel">${_('Journal')}</div> | |
390 | </a> |
|
390 | </a> | |
391 | </li> |
|
391 | </li> | |
392 | %else: |
|
392 | %else: | |
393 | <li class="${is_active('journal')}"> |
|
393 | <li class="${is_active('journal')}"> | |
394 | <a class="menulink" title="${_('Show Public activity journal')}" href="${h.url('public_journal')}"> |
|
394 | <a class="menulink" title="${_('Show Public activity journal')}" href="${h.url('public_journal')}"> | |
395 | <div class="menulabel">${_('Public journal')}</div> |
|
395 | <div class="menulabel">${_('Public journal')}</div> | |
396 | </a> |
|
396 | </a> | |
397 | </li> |
|
397 | </li> | |
398 | %endif |
|
398 | %endif | |
399 | <li class="${is_active('gists')}"> |
|
399 | <li class="${is_active('gists')}"> | |
400 | <a class="menulink childs" title="${_('Show Gists')}" href="${h.route_path('gists_show')}"> |
|
400 | <a class="menulink childs" title="${_('Show Gists')}" href="${h.route_path('gists_show')}"> | |
401 | <div class="menulabel">${_('Gists')}</div> |
|
401 | <div class="menulabel">${_('Gists')}</div> | |
402 | </a> |
|
402 | </a> | |
403 | </li> |
|
403 | </li> | |
404 | <li class="${is_active('search')}"> |
|
404 | <li class="${is_active('search')}"> | |
405 | <a class="menulink" title="${_('Search in repositories you have access to')}" href="${h.route_path('search')}"> |
|
405 | <a class="menulink" title="${_('Search in repositories you have access to')}" href="${h.route_path('search')}"> | |
406 | <div class="menulabel">${_('Search')}</div> |
|
406 | <div class="menulabel">${_('Search')}</div> | |
407 | </a> |
|
407 | </a> | |
408 | </li> |
|
408 | </li> | |
409 | % if h.HasPermissionAll('hg.admin')('access admin main page'): |
|
409 | % if h.HasPermissionAll('hg.admin')('access admin main page'): | |
410 | <li class="${is_active('admin')}"> |
|
410 | <li class="${is_active('admin')}"> | |
411 | <a class="menulink childs" title="${_('Admin settings')}" href="#" onclick="return false;"> |
|
411 | <a class="menulink childs" title="${_('Admin settings')}" href="#" onclick="return false;"> | |
412 | <div class="menulabel">${_('Admin')} <div class="show_more"></div></div> |
|
412 | <div class="menulabel">${_('Admin')} <div class="show_more"></div></div> | |
413 | </a> |
|
413 | </a> | |
414 | ${admin_menu()} |
|
414 | ${admin_menu()} | |
415 | </li> |
|
415 | </li> | |
416 | % elif c.rhodecode_user.repositories_admin or c.rhodecode_user.repository_groups_admin or c.rhodecode_user.user_groups_admin: |
|
416 | % elif c.rhodecode_user.repositories_admin or c.rhodecode_user.repository_groups_admin or c.rhodecode_user.user_groups_admin: | |
417 | <li class="${is_active('admin')}"> |
|
417 | <li class="${is_active('admin')}"> | |
418 | <a class="menulink childs" title="${_('Delegated Admin settings')}"> |
|
418 | <a class="menulink childs" title="${_('Delegated Admin settings')}"> | |
419 | <div class="menulabel">${_('Admin')} <div class="show_more"></div></div> |
|
419 | <div class="menulabel">${_('Admin')} <div class="show_more"></div></div> | |
420 | </a> |
|
420 | </a> | |
421 | ${admin_menu_simple(c.rhodecode_user.repositories_admin, |
|
421 | ${admin_menu_simple(c.rhodecode_user.repositories_admin, | |
422 | c.rhodecode_user.repository_groups_admin, |
|
422 | c.rhodecode_user.repository_groups_admin, | |
423 | c.rhodecode_user.user_groups_admin or h.HasPermissionAny('hg.usergroup.create.true')())} |
|
423 | c.rhodecode_user.user_groups_admin or h.HasPermissionAny('hg.usergroup.create.true')())} | |
424 | </li> |
|
424 | </li> | |
425 | % endif |
|
425 | % endif | |
426 | % if c.debug_style: |
|
426 | % if c.debug_style: | |
427 | <li class="${is_active('debug_style')}"> |
|
427 | <li class="${is_active('debug_style')}"> | |
428 | <a class="menulink" title="${_('Style')}" href="${h.route_path('debug_style_home')}"> |
|
428 | <a class="menulink" title="${_('Style')}" href="${h.route_path('debug_style_home')}"> | |
429 | <div class="menulabel">${_('Style')}</div> |
|
429 | <div class="menulabel">${_('Style')}</div> | |
430 | </a> |
|
430 | </a> | |
431 | </li> |
|
431 | </li> | |
432 | % endif |
|
432 | % endif | |
433 | ## render extra user menu |
|
433 | ## render extra user menu | |
434 | ${usermenu(active=(active=='my_account'))} |
|
434 | ${usermenu(active=(active=='my_account'))} | |
435 | </ul> |
|
435 | </ul> | |
436 |
|
436 | |||
437 | <script type="text/javascript"> |
|
437 | <script type="text/javascript"> | |
438 | var visual_show_public_icon = "${c.visual.show_public_icon}" == "True"; |
|
438 | var visual_show_public_icon = "${c.visual.show_public_icon}" == "True"; | |
439 |
|
439 | |||
440 | /*format the look of items in the list*/ |
|
440 | /*format the look of items in the list*/ | |
441 | var format = function(state, escapeMarkup){ |
|
441 | var format = function(state, escapeMarkup){ | |
442 | if (!state.id){ |
|
442 | if (!state.id){ | |
443 | return state.text; // optgroup |
|
443 | return state.text; // optgroup | |
444 | } |
|
444 | } | |
445 | var obj_dict = state.obj; |
|
445 | var obj_dict = state.obj; | |
446 | var tmpl = ''; |
|
446 | var tmpl = ''; | |
447 |
|
447 | |||
448 | if(obj_dict && state.type == 'repo'){ |
|
448 | if(obj_dict && state.type == 'repo'){ | |
449 | if(obj_dict['repo_type'] === 'hg'){ |
|
449 | if(obj_dict['repo_type'] === 'hg'){ | |
450 | tmpl += '<i class="icon-hg"></i> '; |
|
450 | tmpl += '<i class="icon-hg"></i> '; | |
451 | } |
|
451 | } | |
452 | else if(obj_dict['repo_type'] === 'git'){ |
|
452 | else if(obj_dict['repo_type'] === 'git'){ | |
453 | tmpl += '<i class="icon-git"></i> '; |
|
453 | tmpl += '<i class="icon-git"></i> '; | |
454 | } |
|
454 | } | |
455 | else if(obj_dict['repo_type'] === 'svn'){ |
|
455 | else if(obj_dict['repo_type'] === 'svn'){ | |
456 | tmpl += '<i class="icon-svn"></i> '; |
|
456 | tmpl += '<i class="icon-svn"></i> '; | |
457 | } |
|
457 | } | |
458 | if(obj_dict['private']){ |
|
458 | if(obj_dict['private']){ | |
459 | tmpl += '<i class="icon-lock" ></i> '; |
|
459 | tmpl += '<i class="icon-lock" ></i> '; | |
460 | } |
|
460 | } | |
461 | else if(visual_show_public_icon){ |
|
461 | else if(visual_show_public_icon){ | |
462 | tmpl += '<i class="icon-unlock-alt"></i> '; |
|
462 | tmpl += '<i class="icon-unlock-alt"></i> '; | |
463 | } |
|
463 | } | |
464 | } |
|
464 | } | |
465 | if(obj_dict && state.type == 'commit') { |
|
465 | if(obj_dict && state.type == 'commit') { | |
466 | tmpl += '<i class="icon-tag"></i>'; |
|
466 | tmpl += '<i class="icon-tag"></i>'; | |
467 | } |
|
467 | } | |
468 | if(obj_dict && state.type == 'group'){ |
|
468 | if(obj_dict && state.type == 'group'){ | |
469 | tmpl += '<i class="icon-folder-close"></i> '; |
|
469 | tmpl += '<i class="icon-folder-close"></i> '; | |
470 | } |
|
470 | } | |
471 | tmpl += escapeMarkup(state.text); |
|
471 | tmpl += escapeMarkup(state.text); | |
472 | return tmpl; |
|
472 | return tmpl; | |
473 | }; |
|
473 | }; | |
474 |
|
474 | |||
475 | var formatResult = function(result, container, query, escapeMarkup) { |
|
475 | var formatResult = function(result, container, query, escapeMarkup) { | |
476 | return format(result, escapeMarkup); |
|
476 | return format(result, escapeMarkup); | |
477 | }; |
|
477 | }; | |
478 |
|
478 | |||
479 | var formatSelection = function(data, container, escapeMarkup) { |
|
479 | var formatSelection = function(data, container, escapeMarkup) { | |
480 | return format(data, escapeMarkup); |
|
480 | return format(data, escapeMarkup); | |
481 | }; |
|
481 | }; | |
482 |
|
482 | |||
483 | $("#repo_switcher").select2({ |
|
483 | $("#repo_switcher").select2({ | |
484 | cachedDataSource: {}, |
|
484 | cachedDataSource: {}, | |
485 | minimumInputLength: 2, |
|
485 | minimumInputLength: 2, | |
486 | placeholder: '<div class="menulabel">${_('Go to')} <div class="show_more"></div></div>', |
|
486 | placeholder: '<div class="menulabel">${_('Go to')} <div class="show_more"></div></div>', | |
487 | dropdownAutoWidth: true, |
|
487 | dropdownAutoWidth: true, | |
488 | formatResult: formatResult, |
|
488 | formatResult: formatResult, | |
489 | formatSelection: formatSelection, |
|
489 | formatSelection: formatSelection, | |
490 | containerCssClass: "repo-switcher", |
|
490 | containerCssClass: "repo-switcher", | |
491 | dropdownCssClass: "repo-switcher-dropdown", |
|
491 | dropdownCssClass: "repo-switcher-dropdown", | |
492 | escapeMarkup: function(m){ |
|
492 | escapeMarkup: function(m){ | |
493 | // don't escape our custom placeholder |
|
493 | // don't escape our custom placeholder | |
494 | if(m.substr(0,23) == '<div class="menulabel">'){ |
|
494 | if(m.substr(0,23) == '<div class="menulabel">'){ | |
495 | return m; |
|
495 | return m; | |
496 | } |
|
496 | } | |
497 |
|
497 | |||
498 | return Select2.util.escapeMarkup(m); |
|
498 | return Select2.util.escapeMarkup(m); | |
499 | }, |
|
499 | }, | |
500 | query: $.debounce(250, function(query){ |
|
500 | query: $.debounce(250, function(query){ | |
501 | self = this; |
|
501 | self = this; | |
502 | var cacheKey = query.term; |
|
502 | var cacheKey = query.term; | |
503 | var cachedData = self.cachedDataSource[cacheKey]; |
|
503 | var cachedData = self.cachedDataSource[cacheKey]; | |
504 |
|
504 | |||
505 | if (cachedData) { |
|
505 | if (cachedData) { | |
506 | query.callback({results: cachedData.results}); |
|
506 | query.callback({results: cachedData.results}); | |
507 | } else { |
|
507 | } else { | |
508 | $.ajax({ |
|
508 | $.ajax({ | |
509 | url: pyroutes.url('goto_switcher_data'), |
|
509 | url: pyroutes.url('goto_switcher_data'), | |
510 | data: {'query': query.term}, |
|
510 | data: {'query': query.term}, | |
511 | dataType: 'json', |
|
511 | dataType: 'json', | |
512 | type: 'GET', |
|
512 | type: 'GET', | |
513 | success: function(data) { |
|
513 | success: function(data) { | |
514 | self.cachedDataSource[cacheKey] = data; |
|
514 | self.cachedDataSource[cacheKey] = data; | |
515 | query.callback({results: data.results}); |
|
515 | query.callback({results: data.results}); | |
516 | }, |
|
516 | }, | |
517 | error: function(data, textStatus, errorThrown) { |
|
517 | error: function(data, textStatus, errorThrown) { | |
518 | alert("Error while fetching entries.\nError code {0} ({1}).".format(data.status, data.statusText)); |
|
518 | alert("Error while fetching entries.\nError code {0} ({1}).".format(data.status, data.statusText)); | |
519 | } |
|
519 | } | |
520 | }) |
|
520 | }) | |
521 | } |
|
521 | } | |
522 | }) |
|
522 | }) | |
523 | }); |
|
523 | }); | |
524 |
|
524 | |||
525 | $("#repo_switcher").on('select2-selecting', function(e){ |
|
525 | $("#repo_switcher").on('select2-selecting', function(e){ | |
526 | e.preventDefault(); |
|
526 | e.preventDefault(); | |
527 | window.location = e.choice.url; |
|
527 | window.location = e.choice.url; | |
528 | }); |
|
528 | }); | |
529 |
|
529 | |||
530 | </script> |
|
530 | </script> | |
531 | <script src="${h.asset('js/rhodecode/base/keyboard-bindings.js', ver=c.rhodecode_version_hash)}"></script> |
|
531 | <script src="${h.asset('js/rhodecode/base/keyboard-bindings.js', ver=c.rhodecode_version_hash)}"></script> | |
532 | </%def> |
|
532 | </%def> | |
533 |
|
533 | |||
534 | <div class="modal" id="help_kb" tabindex="-1" role="dialog" aria-labelledby="myModalLabel" aria-hidden="true"> |
|
534 | <div class="modal" id="help_kb" tabindex="-1" role="dialog" aria-labelledby="myModalLabel" aria-hidden="true"> | |
535 | <div class="modal-dialog"> |
|
535 | <div class="modal-dialog"> | |
536 | <div class="modal-content"> |
|
536 | <div class="modal-content"> | |
537 | <div class="modal-header"> |
|
537 | <div class="modal-header"> | |
538 | <button type="button" class="close" data-dismiss="modal" aria-hidden="true">×</button> |
|
538 | <button type="button" class="close" data-dismiss="modal" aria-hidden="true">×</button> | |
539 | <h4 class="modal-title" id="myModalLabel">${_('Keyboard shortcuts')}</h4> |
|
539 | <h4 class="modal-title" id="myModalLabel">${_('Keyboard shortcuts')}</h4> | |
540 | </div> |
|
540 | </div> | |
541 | <div class="modal-body"> |
|
541 | <div class="modal-body"> | |
542 | <div class="block-left"> |
|
542 | <div class="block-left"> | |
543 | <table class="keyboard-mappings"> |
|
543 | <table class="keyboard-mappings"> | |
544 | <tbody> |
|
544 | <tbody> | |
545 | <tr> |
|
545 | <tr> | |
546 | <th></th> |
|
546 | <th></th> | |
547 | <th>${_('Site-wide shortcuts')}</th> |
|
547 | <th>${_('Site-wide shortcuts')}</th> | |
548 | </tr> |
|
548 | </tr> | |
549 | <% |
|
549 | <% | |
550 | elems = [ |
|
550 | elems = [ | |
551 | ('/', 'Open quick search box'), |
|
551 | ('/', 'Open quick search box'), | |
552 | ('g h', 'Goto home page'), |
|
552 | ('g h', 'Goto home page'), | |
553 | ('g g', 'Goto my private gists page'), |
|
553 | ('g g', 'Goto my private gists page'), | |
554 | ('g G', 'Goto my public gists page'), |
|
554 | ('g G', 'Goto my public gists page'), | |
555 | ('n r', 'New repository page'), |
|
555 | ('n r', 'New repository page'), | |
556 | ('n g', 'New gist page'), |
|
556 | ('n g', 'New gist page'), | |
557 | ] |
|
557 | ] | |
558 | %> |
|
558 | %> | |
559 | %for key, desc in elems: |
|
559 | %for key, desc in elems: | |
560 | <tr> |
|
560 | <tr> | |
561 | <td class="keys"> |
|
561 | <td class="keys"> | |
562 | <span class="key tag">${key}</span> |
|
562 | <span class="key tag">${key}</span> | |
563 | </td> |
|
563 | </td> | |
564 | <td>${desc}</td> |
|
564 | <td>${desc}</td> | |
565 | </tr> |
|
565 | </tr> | |
566 | %endfor |
|
566 | %endfor | |
567 | </tbody> |
|
567 | </tbody> | |
568 | </table> |
|
568 | </table> | |
569 | </div> |
|
569 | </div> | |
570 | <div class="block-left"> |
|
570 | <div class="block-left"> | |
571 | <table class="keyboard-mappings"> |
|
571 | <table class="keyboard-mappings"> | |
572 | <tbody> |
|
572 | <tbody> | |
573 | <tr> |
|
573 | <tr> | |
574 | <th></th> |
|
574 | <th></th> | |
575 | <th>${_('Repositories')}</th> |
|
575 | <th>${_('Repositories')}</th> | |
576 | </tr> |
|
576 | </tr> | |
577 | <% |
|
577 | <% | |
578 | elems = [ |
|
578 | elems = [ | |
579 | ('g s', 'Goto summary page'), |
|
579 | ('g s', 'Goto summary page'), | |
580 | ('g c', 'Goto changelog page'), |
|
580 | ('g c', 'Goto changelog page'), | |
581 | ('g f', 'Goto files page'), |
|
581 | ('g f', 'Goto files page'), | |
582 | ('g F', 'Goto files page with file search activated'), |
|
582 | ('g F', 'Goto files page with file search activated'), | |
583 | ('g p', 'Goto pull requests page'), |
|
583 | ('g p', 'Goto pull requests page'), | |
584 | ('g o', 'Goto repository settings'), |
|
584 | ('g o', 'Goto repository settings'), | |
585 | ('g O', 'Goto repository permissions settings'), |
|
585 | ('g O', 'Goto repository permissions settings'), | |
586 | ] |
|
586 | ] | |
587 | %> |
|
587 | %> | |
588 | %for key, desc in elems: |
|
588 | %for key, desc in elems: | |
589 | <tr> |
|
589 | <tr> | |
590 | <td class="keys"> |
|
590 | <td class="keys"> | |
591 | <span class="key tag">${key}</span> |
|
591 | <span class="key tag">${key}</span> | |
592 | </td> |
|
592 | </td> | |
593 | <td>${desc}</td> |
|
593 | <td>${desc}</td> | |
594 | </tr> |
|
594 | </tr> | |
595 | %endfor |
|
595 | %endfor | |
596 | </tbody> |
|
596 | </tbody> | |
597 | </table> |
|
597 | </table> | |
598 | </div> |
|
598 | </div> | |
599 | </div> |
|
599 | </div> | |
600 | <div class="modal-footer"> |
|
600 | <div class="modal-footer"> | |
601 | </div> |
|
601 | </div> | |
602 | </div><!-- /.modal-content --> |
|
602 | </div><!-- /.modal-content --> | |
603 | </div><!-- /.modal-dialog --> |
|
603 | </div><!-- /.modal-dialog --> | |
604 | </div><!-- /.modal --> |
|
604 | </div><!-- /.modal --> |
@@ -1,317 +1,317 b'' | |||||
1 | ## DATA TABLE RE USABLE ELEMENTS |
|
1 | ## DATA TABLE RE USABLE ELEMENTS | |
2 | ## usage: |
|
2 | ## usage: | |
3 | ## <%namespace name="dt" file="/data_table/_dt_elements.mako"/> |
|
3 | ## <%namespace name="dt" file="/data_table/_dt_elements.mako"/> | |
4 | <%namespace name="base" file="/base/base.mako"/> |
|
4 | <%namespace name="base" file="/base/base.mako"/> | |
5 |
|
5 | |||
6 | ## REPOSITORY RENDERERS |
|
6 | ## REPOSITORY RENDERERS | |
7 | <%def name="quick_menu(repo_name)"> |
|
7 | <%def name="quick_menu(repo_name)"> | |
8 | <i class="pointer icon-more"></i> |
|
8 | <i class="pointer icon-more"></i> | |
9 | <div class="menu_items_container hidden"> |
|
9 | <div class="menu_items_container hidden"> | |
10 | <ul class="menu_items"> |
|
10 | <ul class="menu_items"> | |
11 | <li> |
|
11 | <li> | |
12 | <a title="${_('Summary')}" href="${h.route_path('repo_summary',repo_name=repo_name)}"> |
|
12 | <a title="${_('Summary')}" href="${h.route_path('repo_summary',repo_name=repo_name)}"> | |
13 | <span>${_('Summary')}</span> |
|
13 | <span>${_('Summary')}</span> | |
14 | </a> |
|
14 | </a> | |
15 | </li> |
|
15 | </li> | |
16 | <li> |
|
16 | <li> | |
17 | <a title="${_('Changelog')}" href="${h.url('changelog_home',repo_name=repo_name)}"> |
|
17 | <a title="${_('Changelog')}" href="${h.url('changelog_home',repo_name=repo_name)}"> | |
18 | <span>${_('Changelog')}</span> |
|
18 | <span>${_('Changelog')}</span> | |
19 | </a> |
|
19 | </a> | |
20 | </li> |
|
20 | </li> | |
21 | <li> |
|
21 | <li> | |
22 | <a title="${_('Files')}" href="${h.url('files_home',repo_name=repo_name)}"> |
|
22 | <a title="${_('Files')}" href="${h.url('files_home',repo_name=repo_name)}"> | |
23 | <span>${_('Files')}</span> |
|
23 | <span>${_('Files')}</span> | |
24 | </a> |
|
24 | </a> | |
25 | </li> |
|
25 | </li> | |
26 | <li> |
|
26 | <li> | |
27 | <a title="${_('Fork')}" href="${h.url('repo_fork_home',repo_name=repo_name)}"> |
|
27 | <a title="${_('Fork')}" href="${h.url('repo_fork_home',repo_name=repo_name)}"> | |
28 | <span>${_('Fork')}</span> |
|
28 | <span>${_('Fork')}</span> | |
29 | </a> |
|
29 | </a> | |
30 | </li> |
|
30 | </li> | |
31 | </ul> |
|
31 | </ul> | |
32 | </div> |
|
32 | </div> | |
33 | </%def> |
|
33 | </%def> | |
34 |
|
34 | |||
35 | <%def name="repo_name(name,rtype,rstate,private,fork_of,short_name=False,admin=False)"> |
|
35 | <%def name="repo_name(name,rtype,rstate,private,fork_of,short_name=False,admin=False)"> | |
36 | <% |
|
36 | <% | |
37 | def get_name(name,short_name=short_name): |
|
37 | def get_name(name,short_name=short_name): | |
38 | if short_name: |
|
38 | if short_name: | |
39 | return name.split('/')[-1] |
|
39 | return name.split('/')[-1] | |
40 | else: |
|
40 | else: | |
41 | return name |
|
41 | return name | |
42 | %> |
|
42 | %> | |
43 | <div class="${'repo_state_pending' if rstate == 'repo_state_pending' else ''} truncate"> |
|
43 | <div class="${'repo_state_pending' if rstate == 'repo_state_pending' else ''} truncate"> | |
44 | ##NAME |
|
44 | ##NAME | |
45 | <a href="${h.route_path('edit_repo',repo_name=name) if admin else h.route_path('repo_summary',repo_name=name)}"> |
|
45 | <a href="${h.route_path('edit_repo',repo_name=name) if admin else h.route_path('repo_summary',repo_name=name)}"> | |
46 |
|
46 | |||
47 | ##TYPE OF REPO |
|
47 | ##TYPE OF REPO | |
48 | %if h.is_hg(rtype): |
|
48 | %if h.is_hg(rtype): | |
49 | <span title="${_('Mercurial repository')}"><i class="icon-hg"></i></span> |
|
49 | <span title="${_('Mercurial repository')}"><i class="icon-hg"></i></span> | |
50 | %elif h.is_git(rtype): |
|
50 | %elif h.is_git(rtype): | |
51 | <span title="${_('Git repository')}"><i class="icon-git"></i></span> |
|
51 | <span title="${_('Git repository')}"><i class="icon-git"></i></span> | |
52 | %elif h.is_svn(rtype): |
|
52 | %elif h.is_svn(rtype): | |
53 | <span title="${_('Subversion repository')}"><i class="icon-svn"></i></span> |
|
53 | <span title="${_('Subversion repository')}"><i class="icon-svn"></i></span> | |
54 | %endif |
|
54 | %endif | |
55 |
|
55 | |||
56 | ##PRIVATE/PUBLIC |
|
56 | ##PRIVATE/PUBLIC | |
57 | %if private and c.visual.show_private_icon: |
|
57 | %if private and c.visual.show_private_icon: | |
58 | <i class="icon-lock" title="${_('Private repository')}"></i> |
|
58 | <i class="icon-lock" title="${_('Private repository')}"></i> | |
59 | %elif not private and c.visual.show_public_icon: |
|
59 | %elif not private and c.visual.show_public_icon: | |
60 | <i class="icon-unlock-alt" title="${_('Public repository')}"></i> |
|
60 | <i class="icon-unlock-alt" title="${_('Public repository')}"></i> | |
61 | %else: |
|
61 | %else: | |
62 | <span></span> |
|
62 | <span></span> | |
63 | %endif |
|
63 | %endif | |
64 | ${get_name(name)} |
|
64 | ${get_name(name)} | |
65 | </a> |
|
65 | </a> | |
66 | %if fork_of: |
|
66 | %if fork_of: | |
67 | <a href="${h.route_path('repo_summary',repo_name=fork_of.repo_name)}"><i class="icon-code-fork"></i></a> |
|
67 | <a href="${h.route_path('repo_summary',repo_name=fork_of.repo_name)}"><i class="icon-code-fork"></i></a> | |
68 | %endif |
|
68 | %endif | |
69 | %if rstate == 'repo_state_pending': |
|
69 | %if rstate == 'repo_state_pending': | |
70 | <i class="icon-cogs" title="${_('Repository creating in progress...')}"></i> |
|
70 | <i class="icon-cogs" title="${_('Repository creating in progress...')}"></i> | |
71 | %endif |
|
71 | %endif | |
72 | </div> |
|
72 | </div> | |
73 | </%def> |
|
73 | </%def> | |
74 |
|
74 | |||
75 | <%def name="repo_desc(description)"> |
|
75 | <%def name="repo_desc(description)"> | |
76 | <div class="truncate-wrap">${description}</div> |
|
76 | <div class="truncate-wrap">${description}</div> | |
77 | </%def> |
|
77 | </%def> | |
78 |
|
78 | |||
79 | <%def name="last_change(last_change)"> |
|
79 | <%def name="last_change(last_change)"> | |
80 | ${h.age_component(last_change)} |
|
80 | ${h.age_component(last_change)} | |
81 | </%def> |
|
81 | </%def> | |
82 |
|
82 | |||
83 | <%def name="revision(name,rev,tip,author,last_msg)"> |
|
83 | <%def name="revision(name,rev,tip,author,last_msg)"> | |
84 | <div> |
|
84 | <div> | |
85 | %if rev >= 0: |
|
85 | %if rev >= 0: | |
86 | <code><a title="${h.tooltip('%s:\n\n%s' % (author,last_msg))}" class="tooltip" href="${h.url('changeset_home',repo_name=name,revision=tip)}">${'r%s:%s' % (rev,h.short_id(tip))}</a></code> |
|
86 | <code><a title="${h.tooltip('%s:\n\n%s' % (author,last_msg))}" class="tooltip" href="${h.url('changeset_home',repo_name=name,revision=tip)}">${'r%s:%s' % (rev,h.short_id(tip))}</a></code> | |
87 | %else: |
|
87 | %else: | |
88 | ${_('No commits yet')} |
|
88 | ${_('No commits yet')} | |
89 | %endif |
|
89 | %endif | |
90 | </div> |
|
90 | </div> | |
91 | </%def> |
|
91 | </%def> | |
92 |
|
92 | |||
93 | <%def name="rss(name)"> |
|
93 | <%def name="rss(name)"> | |
94 | %if c.rhodecode_user.username != h.DEFAULT_USER: |
|
94 | %if c.rhodecode_user.username != h.DEFAULT_USER: | |
95 | <a title="${h.tooltip(_('Subscribe to %s rss feed')% name)}" href="${h.route_path('rss_feed_home', repo_name=name, _query=dict(auth_token=c.rhodecode_user.feed_token))}"><i class="icon-rss-sign"></i></a> |
|
95 | <a title="${h.tooltip(_('Subscribe to %s rss feed')% name)}" href="${h.route_path('rss_feed_home', repo_name=name, _query=dict(auth_token=c.rhodecode_user.feed_token))}"><i class="icon-rss-sign"></i></a> | |
96 | %else: |
|
96 | %else: | |
97 | <a title="${h.tooltip(_('Subscribe to %s rss feed')% name)}" href="${h.route_path('rss_feed_home', repo_name=name)}"><i class="icon-rss-sign"></i></a> |
|
97 | <a title="${h.tooltip(_('Subscribe to %s rss feed')% name)}" href="${h.route_path('rss_feed_home', repo_name=name)}"><i class="icon-rss-sign"></i></a> | |
98 | %endif |
|
98 | %endif | |
99 | </%def> |
|
99 | </%def> | |
100 |
|
100 | |||
101 | <%def name="atom(name)"> |
|
101 | <%def name="atom(name)"> | |
102 | %if c.rhodecode_user.username != h.DEFAULT_USER: |
|
102 | %if c.rhodecode_user.username != h.DEFAULT_USER: | |
103 | <a title="${h.tooltip(_('Subscribe to %s atom feed')% name)}" href="${h.route_path('atom_feed_home', repo_name=name, _query=dict(auth_token=c.rhodecode_user.feed_token))}"><i class="icon-rss-sign"></i></a> |
|
103 | <a title="${h.tooltip(_('Subscribe to %s atom feed')% name)}" href="${h.route_path('atom_feed_home', repo_name=name, _query=dict(auth_token=c.rhodecode_user.feed_token))}"><i class="icon-rss-sign"></i></a> | |
104 | %else: |
|
104 | %else: | |
105 | <a title="${h.tooltip(_('Subscribe to %s atom feed')% name)}" href="${h.route_path('atom_feed_home', repo_name=name)}"><i class="icon-rss-sign"></i></a> |
|
105 | <a title="${h.tooltip(_('Subscribe to %s atom feed')% name)}" href="${h.route_path('atom_feed_home', repo_name=name)}"><i class="icon-rss-sign"></i></a> | |
106 | %endif |
|
106 | %endif | |
107 | </%def> |
|
107 | </%def> | |
108 |
|
108 | |||
109 | <%def name="user_gravatar(email, size=16)"> |
|
109 | <%def name="user_gravatar(email, size=16)"> | |
110 | <div class="rc-user tooltip" title="${h.tooltip(h.author_string(email))}"> |
|
110 | <div class="rc-user tooltip" title="${h.tooltip(h.author_string(email))}"> | |
111 | ${base.gravatar(email, 16)} |
|
111 | ${base.gravatar(email, 16)} | |
112 | </div> |
|
112 | </div> | |
113 | </%def> |
|
113 | </%def> | |
114 |
|
114 | |||
115 | <%def name="repo_actions(repo_name, super_user=True)"> |
|
115 | <%def name="repo_actions(repo_name, super_user=True)"> | |
116 | <div> |
|
116 | <div> | |
117 | <div class="grid_edit"> |
|
117 | <div class="grid_edit"> | |
118 | <a href="${h.route_path('edit_repo',repo_name=repo_name)}" title="${_('Edit')}"> |
|
118 | <a href="${h.route_path('edit_repo',repo_name=repo_name)}" title="${_('Edit')}"> | |
119 | <i class="icon-pencil"></i>Edit</a> |
|
119 | <i class="icon-pencil"></i>Edit</a> | |
120 | </div> |
|
120 | </div> | |
121 | <div class="grid_delete"> |
|
121 | <div class="grid_delete"> | |
122 | ${h.secure_form(h.route_path('edit_repo_advanced_delete', repo_name=repo_name), method='POST')} |
|
122 | ${h.secure_form(h.route_path('edit_repo_advanced_delete', repo_name=repo_name), method='POST', request=request)} | |
123 | ${h.submit('remove_%s' % repo_name,_('Delete'),class_="btn btn-link btn-danger", |
|
123 | ${h.submit('remove_%s' % repo_name,_('Delete'),class_="btn btn-link btn-danger", | |
124 | onclick="return confirm('"+_('Confirm to delete this repository: %s') % repo_name+"');")} |
|
124 | onclick="return confirm('"+_('Confirm to delete this repository: %s') % repo_name+"');")} | |
125 | ${h.end_form()} |
|
125 | ${h.end_form()} | |
126 | </div> |
|
126 | </div> | |
127 | </div> |
|
127 | </div> | |
128 | </%def> |
|
128 | </%def> | |
129 |
|
129 | |||
130 | <%def name="repo_state(repo_state)"> |
|
130 | <%def name="repo_state(repo_state)"> | |
131 | <div> |
|
131 | <div> | |
132 | %if repo_state == 'repo_state_pending': |
|
132 | %if repo_state == 'repo_state_pending': | |
133 | <div class="tag tag4">${_('Creating')}</div> |
|
133 | <div class="tag tag4">${_('Creating')}</div> | |
134 | %elif repo_state == 'repo_state_created': |
|
134 | %elif repo_state == 'repo_state_created': | |
135 | <div class="tag tag1">${_('Created')}</div> |
|
135 | <div class="tag tag1">${_('Created')}</div> | |
136 | %else: |
|
136 | %else: | |
137 | <div class="tag alert2" title="${h.tooltip(repo_state)}">invalid</div> |
|
137 | <div class="tag alert2" title="${h.tooltip(repo_state)}">invalid</div> | |
138 | %endif |
|
138 | %endif | |
139 | </div> |
|
139 | </div> | |
140 | </%def> |
|
140 | </%def> | |
141 |
|
141 | |||
142 |
|
142 | |||
143 | ## REPO GROUP RENDERERS |
|
143 | ## REPO GROUP RENDERERS | |
144 | <%def name="quick_repo_group_menu(repo_group_name)"> |
|
144 | <%def name="quick_repo_group_menu(repo_group_name)"> | |
145 | <i class="pointer icon-more"></i> |
|
145 | <i class="pointer icon-more"></i> | |
146 | <div class="menu_items_container hidden"> |
|
146 | <div class="menu_items_container hidden"> | |
147 | <ul class="menu_items"> |
|
147 | <ul class="menu_items"> | |
148 | <li> |
|
148 | <li> | |
149 | <a href="${h.route_path('repo_group_home', repo_group_name=repo_group_name)}"> |
|
149 | <a href="${h.route_path('repo_group_home', repo_group_name=repo_group_name)}"> | |
150 | <span class="icon"> |
|
150 | <span class="icon"> | |
151 | <i class="icon-file-text"></i> |
|
151 | <i class="icon-file-text"></i> | |
152 | </span> |
|
152 | </span> | |
153 | <span>${_('Summary')}</span> |
|
153 | <span>${_('Summary')}</span> | |
154 | </a> |
|
154 | </a> | |
155 | </li> |
|
155 | </li> | |
156 |
|
156 | |||
157 | </ul> |
|
157 | </ul> | |
158 | </div> |
|
158 | </div> | |
159 | </%def> |
|
159 | </%def> | |
160 |
|
160 | |||
161 | <%def name="repo_group_name(repo_group_name, children_groups=None)"> |
|
161 | <%def name="repo_group_name(repo_group_name, children_groups=None)"> | |
162 | <div> |
|
162 | <div> | |
163 | <a href="${h.route_path('repo_group_home', repo_group_name=repo_group_name)}"> |
|
163 | <a href="${h.route_path('repo_group_home', repo_group_name=repo_group_name)}"> | |
164 | <i class="icon-folder-close" title="${_('Repository group')}"></i> |
|
164 | <i class="icon-folder-close" title="${_('Repository group')}"></i> | |
165 | %if children_groups: |
|
165 | %if children_groups: | |
166 | ${h.literal(' » '.join(children_groups))} |
|
166 | ${h.literal(' » '.join(children_groups))} | |
167 | %else: |
|
167 | %else: | |
168 | ${repo_group_name} |
|
168 | ${repo_group_name} | |
169 | %endif |
|
169 | %endif | |
170 | </a> |
|
170 | </a> | |
171 | </div> |
|
171 | </div> | |
172 | </%def> |
|
172 | </%def> | |
173 |
|
173 | |||
174 | <%def name="repo_group_desc(description)"> |
|
174 | <%def name="repo_group_desc(description)"> | |
175 | <div class="truncate-wrap">${description}</div> |
|
175 | <div class="truncate-wrap">${description}</div> | |
176 | </%def> |
|
176 | </%def> | |
177 |
|
177 | |||
178 | <%def name="repo_group_actions(repo_group_id, repo_group_name, gr_count)"> |
|
178 | <%def name="repo_group_actions(repo_group_id, repo_group_name, gr_count)"> | |
179 | <div class="grid_edit"> |
|
179 | <div class="grid_edit"> | |
180 | <a href="${h.url('edit_repo_group',group_name=repo_group_name)}" title="${_('Edit')}">Edit</a> |
|
180 | <a href="${h.url('edit_repo_group',group_name=repo_group_name)}" title="${_('Edit')}">Edit</a> | |
181 | </div> |
|
181 | </div> | |
182 | <div class="grid_delete"> |
|
182 | <div class="grid_delete"> | |
183 | ${h.secure_form(h.url('delete_repo_group', group_name=repo_group_name),method='delete')} |
|
183 | ${h.secure_form(h.url('delete_repo_group', group_name=repo_group_name),method='delete')} | |
184 | ${h.submit('remove_%s' % repo_group_name,_('Delete'),class_="btn btn-link btn-danger", |
|
184 | ${h.submit('remove_%s' % repo_group_name,_('Delete'),class_="btn btn-link btn-danger", | |
185 | onclick="return confirm('"+ungettext('Confirm to delete this group: %s with %s repository','Confirm to delete this group: %s with %s repositories',gr_count) % (repo_group_name, gr_count)+"');")} |
|
185 | onclick="return confirm('"+ungettext('Confirm to delete this group: %s with %s repository','Confirm to delete this group: %s with %s repositories',gr_count) % (repo_group_name, gr_count)+"');")} | |
186 | ${h.end_form()} |
|
186 | ${h.end_form()} | |
187 | </div> |
|
187 | </div> | |
188 | </%def> |
|
188 | </%def> | |
189 |
|
189 | |||
190 |
|
190 | |||
191 | <%def name="user_actions(user_id, username)"> |
|
191 | <%def name="user_actions(user_id, username)"> | |
192 | <div class="grid_edit"> |
|
192 | <div class="grid_edit"> | |
193 | <a href="${h.url('edit_user',user_id=user_id)}" title="${_('Edit')}"> |
|
193 | <a href="${h.url('edit_user',user_id=user_id)}" title="${_('Edit')}"> | |
194 | <i class="icon-pencil"></i>Edit</a> |
|
194 | <i class="icon-pencil"></i>Edit</a> | |
195 | </div> |
|
195 | </div> | |
196 | <div class="grid_delete"> |
|
196 | <div class="grid_delete"> | |
197 | ${h.secure_form(h.url('delete_user', user_id=user_id),method='delete')} |
|
197 | ${h.secure_form(h.url('delete_user', user_id=user_id),method='delete')} | |
198 | ${h.submit('remove_',_('Delete'),id="remove_user_%s" % user_id, class_="btn btn-link btn-danger", |
|
198 | ${h.submit('remove_',_('Delete'),id="remove_user_%s" % user_id, class_="btn btn-link btn-danger", | |
199 | onclick="return confirm('"+_('Confirm to delete this user: %s') % username+"');")} |
|
199 | onclick="return confirm('"+_('Confirm to delete this user: %s') % username+"');")} | |
200 | ${h.end_form()} |
|
200 | ${h.end_form()} | |
201 | </div> |
|
201 | </div> | |
202 | </%def> |
|
202 | </%def> | |
203 |
|
203 | |||
204 | <%def name="user_group_actions(user_group_id, user_group_name)"> |
|
204 | <%def name="user_group_actions(user_group_id, user_group_name)"> | |
205 | <div class="grid_edit"> |
|
205 | <div class="grid_edit"> | |
206 | <a href="${h.url('edit_users_group', user_group_id=user_group_id)}" title="${_('Edit')}">Edit</a> |
|
206 | <a href="${h.url('edit_users_group', user_group_id=user_group_id)}" title="${_('Edit')}">Edit</a> | |
207 | </div> |
|
207 | </div> | |
208 | <div class="grid_delete"> |
|
208 | <div class="grid_delete"> | |
209 | ${h.secure_form(h.url('delete_users_group', user_group_id=user_group_id),method='delete')} |
|
209 | ${h.secure_form(h.url('delete_users_group', user_group_id=user_group_id),method='delete')} | |
210 | ${h.submit('remove_',_('Delete'),id="remove_group_%s" % user_group_id, class_="btn btn-link btn-danger", |
|
210 | ${h.submit('remove_',_('Delete'),id="remove_group_%s" % user_group_id, class_="btn btn-link btn-danger", | |
211 | onclick="return confirm('"+_('Confirm to delete this user group: %s') % user_group_name+"');")} |
|
211 | onclick="return confirm('"+_('Confirm to delete this user group: %s') % user_group_name+"');")} | |
212 | ${h.end_form()} |
|
212 | ${h.end_form()} | |
213 | </div> |
|
213 | </div> | |
214 | </%def> |
|
214 | </%def> | |
215 |
|
215 | |||
216 |
|
216 | |||
217 | <%def name="user_name(user_id, username)"> |
|
217 | <%def name="user_name(user_id, username)"> | |
218 | ${h.link_to(h.person(username, 'username_or_name_or_email'), h.url('edit_user', user_id=user_id))} |
|
218 | ${h.link_to(h.person(username, 'username_or_name_or_email'), h.url('edit_user', user_id=user_id))} | |
219 | </%def> |
|
219 | </%def> | |
220 |
|
220 | |||
221 | <%def name="user_profile(username)"> |
|
221 | <%def name="user_profile(username)"> | |
222 | ${base.gravatar_with_user(username, 16)} |
|
222 | ${base.gravatar_with_user(username, 16)} | |
223 | </%def> |
|
223 | </%def> | |
224 |
|
224 | |||
225 | <%def name="user_group_name(user_group_id, user_group_name)"> |
|
225 | <%def name="user_group_name(user_group_id, user_group_name)"> | |
226 | <div> |
|
226 | <div> | |
227 | <a href="${h.url('edit_users_group', user_group_id=user_group_id)}"> |
|
227 | <a href="${h.url('edit_users_group', user_group_id=user_group_id)}"> | |
228 | <i class="icon-group" title="${_('User group')}"></i> ${user_group_name}</a> |
|
228 | <i class="icon-group" title="${_('User group')}"></i> ${user_group_name}</a> | |
229 | </div> |
|
229 | </div> | |
230 | </%def> |
|
230 | </%def> | |
231 |
|
231 | |||
232 |
|
232 | |||
233 | ## GISTS |
|
233 | ## GISTS | |
234 |
|
234 | |||
235 | <%def name="gist_gravatar(full_contact)"> |
|
235 | <%def name="gist_gravatar(full_contact)"> | |
236 | <div class="gist_gravatar"> |
|
236 | <div class="gist_gravatar"> | |
237 | ${base.gravatar(full_contact, 30)} |
|
237 | ${base.gravatar(full_contact, 30)} | |
238 | </div> |
|
238 | </div> | |
239 | </%def> |
|
239 | </%def> | |
240 |
|
240 | |||
241 | <%def name="gist_access_id(gist_access_id, full_contact)"> |
|
241 | <%def name="gist_access_id(gist_access_id, full_contact)"> | |
242 | <div> |
|
242 | <div> | |
243 | <b> |
|
243 | <b> | |
244 | <a href="${h.route_path('gist_show', gist_id=gist_access_id)}">gist: ${gist_access_id}</a> |
|
244 | <a href="${h.route_path('gist_show', gist_id=gist_access_id)}">gist: ${gist_access_id}</a> | |
245 | </b> |
|
245 | </b> | |
246 | </div> |
|
246 | </div> | |
247 | </%def> |
|
247 | </%def> | |
248 |
|
248 | |||
249 | <%def name="gist_author(full_contact, created_on, expires)"> |
|
249 | <%def name="gist_author(full_contact, created_on, expires)"> | |
250 | ${base.gravatar_with_user(full_contact, 16)} |
|
250 | ${base.gravatar_with_user(full_contact, 16)} | |
251 | </%def> |
|
251 | </%def> | |
252 |
|
252 | |||
253 |
|
253 | |||
254 | <%def name="gist_created(created_on)"> |
|
254 | <%def name="gist_created(created_on)"> | |
255 | <div class="created"> |
|
255 | <div class="created"> | |
256 | ${h.age_component(created_on, time_is_local=True)} |
|
256 | ${h.age_component(created_on, time_is_local=True)} | |
257 | </div> |
|
257 | </div> | |
258 | </%def> |
|
258 | </%def> | |
259 |
|
259 | |||
260 | <%def name="gist_expires(expires)"> |
|
260 | <%def name="gist_expires(expires)"> | |
261 | <div class="created"> |
|
261 | <div class="created"> | |
262 | %if expires == -1: |
|
262 | %if expires == -1: | |
263 | ${_('never')} |
|
263 | ${_('never')} | |
264 | %else: |
|
264 | %else: | |
265 | ${h.age_component(h.time_to_utcdatetime(expires))} |
|
265 | ${h.age_component(h.time_to_utcdatetime(expires))} | |
266 | %endif |
|
266 | %endif | |
267 | </div> |
|
267 | </div> | |
268 | </%def> |
|
268 | </%def> | |
269 |
|
269 | |||
270 | <%def name="gist_type(gist_type)"> |
|
270 | <%def name="gist_type(gist_type)"> | |
271 | %if gist_type != 'public': |
|
271 | %if gist_type != 'public': | |
272 | <div class="tag">${_('Private')}</div> |
|
272 | <div class="tag">${_('Private')}</div> | |
273 | %endif |
|
273 | %endif | |
274 | </%def> |
|
274 | </%def> | |
275 |
|
275 | |||
276 | <%def name="gist_description(gist_description)"> |
|
276 | <%def name="gist_description(gist_description)"> | |
277 | ${gist_description} |
|
277 | ${gist_description} | |
278 | </%def> |
|
278 | </%def> | |
279 |
|
279 | |||
280 |
|
280 | |||
281 | ## PULL REQUESTS GRID RENDERERS |
|
281 | ## PULL REQUESTS GRID RENDERERS | |
282 |
|
282 | |||
283 | <%def name="pullrequest_target_repo(repo_name)"> |
|
283 | <%def name="pullrequest_target_repo(repo_name)"> | |
284 | <div class="truncate"> |
|
284 | <div class="truncate"> | |
285 | ${h.link_to(repo_name,h.route_path('repo_summary',repo_name=repo_name))} |
|
285 | ${h.link_to(repo_name,h.route_path('repo_summary',repo_name=repo_name))} | |
286 | </div> |
|
286 | </div> | |
287 | </%def> |
|
287 | </%def> | |
288 | <%def name="pullrequest_status(status)"> |
|
288 | <%def name="pullrequest_status(status)"> | |
289 | <div class="${'flag_status %s' % status} pull-left"></div> |
|
289 | <div class="${'flag_status %s' % status} pull-left"></div> | |
290 | </%def> |
|
290 | </%def> | |
291 |
|
291 | |||
292 | <%def name="pullrequest_title(title, description)"> |
|
292 | <%def name="pullrequest_title(title, description)"> | |
293 | ${title} <br/> |
|
293 | ${title} <br/> | |
294 | ${h.shorter(description, 40)} |
|
294 | ${h.shorter(description, 40)} | |
295 | </%def> |
|
295 | </%def> | |
296 |
|
296 | |||
297 | <%def name="pullrequest_comments(comments_nr)"> |
|
297 | <%def name="pullrequest_comments(comments_nr)"> | |
298 | <i class="icon-comment"></i> ${comments_nr} |
|
298 | <i class="icon-comment"></i> ${comments_nr} | |
299 | </%def> |
|
299 | </%def> | |
300 |
|
300 | |||
301 | <%def name="pullrequest_name(pull_request_id, target_repo_name, short=False)"> |
|
301 | <%def name="pullrequest_name(pull_request_id, target_repo_name, short=False)"> | |
302 | <a href="${h.route_path('pullrequest_show',repo_name=target_repo_name,pull_request_id=pull_request_id)}"> |
|
302 | <a href="${h.route_path('pullrequest_show',repo_name=target_repo_name,pull_request_id=pull_request_id)}"> | |
303 | % if short: |
|
303 | % if short: | |
304 | #${pull_request_id} |
|
304 | #${pull_request_id} | |
305 | % else: |
|
305 | % else: | |
306 | ${_('Pull request #%(pr_number)s') % {'pr_number': pull_request_id,}} |
|
306 | ${_('Pull request #%(pr_number)s') % {'pr_number': pull_request_id,}} | |
307 | % endif |
|
307 | % endif | |
308 | </a> |
|
308 | </a> | |
309 | </%def> |
|
309 | </%def> | |
310 |
|
310 | |||
311 | <%def name="pullrequest_updated_on(updated_on)"> |
|
311 | <%def name="pullrequest_updated_on(updated_on)"> | |
312 | ${h.age_component(h.time_to_utcdatetime(updated_on))} |
|
312 | ${h.age_component(h.time_to_utcdatetime(updated_on))} | |
313 | </%def> |
|
313 | </%def> | |
314 |
|
314 | |||
315 | <%def name="pullrequest_author(full_contact)"> |
|
315 | <%def name="pullrequest_author(full_contact)"> | |
316 | ${base.gravatar_with_user(full_contact, 16)} |
|
316 | ${base.gravatar_with_user(full_contact, 16)} | |
317 | </%def> |
|
317 | </%def> |
@@ -1,100 +1,100 b'' | |||||
1 | <form |
|
1 | <form | |
2 | tal:define="style style|field.widget.style; |
|
2 | tal:define="style style|field.widget.style; | |
3 | css_class css_class|string:${field.widget.css_class or field.css_class or ''}; |
|
3 | css_class css_class|string:${field.widget.css_class or field.css_class or ''}; | |
4 | item_template item_template|field.widget.item_template; |
|
4 | item_template item_template|field.widget.item_template; | |
5 | autocomplete autocomplete|field.autocomplete; |
|
5 | autocomplete autocomplete|field.autocomplete; | |
6 | title title|field.title; |
|
6 | title title|field.title; | |
7 | errormsg errormsg|field.errormsg; |
|
7 | errormsg errormsg|field.errormsg; | |
8 | description description|field.description; |
|
8 | description description|field.description; | |
9 | buttons buttons|field.buttons; |
|
9 | buttons buttons|field.buttons; | |
10 | use_ajax use_ajax|field.use_ajax; |
|
10 | use_ajax use_ajax|field.use_ajax; | |
11 | ajax_options ajax_options|field.ajax_options; |
|
11 | ajax_options ajax_options|field.ajax_options; | |
12 | formid formid|field.formid; |
|
12 | formid formid|field.formid; | |
13 | action action|field.action or None; |
|
13 | action action|field.action or None; | |
14 | method method|field.method;" |
|
14 | method method|field.method;" | |
15 | tal:attributes="autocomplete autocomplete; |
|
15 | tal:attributes="autocomplete autocomplete; | |
16 | style style; |
|
16 | style style; | |
17 | class css_class; |
|
17 | class css_class; | |
18 | action action;" |
|
18 | action action;" | |
19 | id="${formid}" |
|
19 | id="${formid}" | |
20 | method="${method}" |
|
20 | method="${method}" | |
21 | enctype="multipart/form-data" |
|
21 | enctype="multipart/form-data" | |
22 | accept-charset="utf-8" |
|
22 | accept-charset="utf-8" | |
23 | i18n:domain="deform" |
|
23 | i18n:domain="deform" | |
24 | > |
|
24 | > | |
25 |
|
25 | |||
26 | <fieldset class="deform-form-fieldset"> |
|
26 | <fieldset class="deform-form-fieldset"> | |
27 |
|
27 | |||
28 | <legend tal:condition="title">${title}</legend> |
|
28 | <legend tal:condition="title">${title}</legend> | |
29 |
|
29 | |||
30 | <input type="hidden" name="${h.csrf_token_key}" value="${h.get_csrf_token()}" /> |
|
30 | <input type="hidden" name="${h.csrf_token_key}" value="${h.get_csrf_token(request.session)}" /> | |
31 | <input type="hidden" name="_charset_" /> |
|
31 | <input type="hidden" name="_charset_" /> | |
32 | <input type="hidden" name="__formid__" value="${formid}"/> |
|
32 | <input type="hidden" name="__formid__" value="${formid}"/> | |
33 |
|
33 | |||
34 | <!-- |
|
34 | <!-- | |
35 | <div class="alert alert-danger" tal:condition="field.error"> |
|
35 | <div class="alert alert-danger" tal:condition="field.error"> | |
36 | <div class="error-msg-lbl" i18n:translate="" |
|
36 | <div class="error-msg-lbl" i18n:translate="" | |
37 | >There was a problem with your submission</div> |
|
37 | >There was a problem with your submission</div> | |
38 | <div class="error-msg-detail" i18n:translate="" |
|
38 | <div class="error-msg-detail" i18n:translate="" | |
39 | >Errors have been highlighted below</div> |
|
39 | >Errors have been highlighted below</div> | |
40 | <p class="error-msg">${field.errormsg}</p> |
|
40 | <p class="error-msg">${field.errormsg}</p> | |
41 | </div> |
|
41 | </div> | |
42 | --> |
|
42 | --> | |
43 |
|
43 | |||
44 | <p class="section first" tal:condition="description"> |
|
44 | <p class="section first" tal:condition="description"> | |
45 | ${description} |
|
45 | ${description} | |
46 | </p> |
|
46 | </p> | |
47 |
|
47 | |||
48 | <div tal:repeat="child field" |
|
48 | <div tal:repeat="child field" | |
49 | tal:replace="structure child.render_template(item_template)"/> |
|
49 | tal:replace="structure child.render_template(item_template)"/> | |
50 |
|
50 | |||
51 | <div class="form-group"> |
|
51 | <div class="form-group"> | |
52 | <tal:loop tal:repeat="button buttons"> |
|
52 | <tal:loop tal:repeat="button buttons"> | |
53 | <button |
|
53 | <button | |
54 | tal:define="btn_disposition repeat.button.start and 'btn-primary' or (button.name == 'delete' and 'btn-danger' or 'btn-default'); |
|
54 | tal:define="btn_disposition repeat.button.start and 'btn-primary' or (button.name == 'delete' and 'btn-danger' or 'btn-default'); | |
55 | btn_icon button.icon|None" |
|
55 | btn_icon button.icon|None" | |
56 | tal:attributes="disabled button.disabled if button.disabled else None" |
|
56 | tal:attributes="disabled button.disabled if button.disabled else None" | |
57 | id="${formid+button.name}" |
|
57 | id="${formid+button.name}" | |
58 | name="${button.name}" |
|
58 | name="${button.name}" | |
59 | type="${button.type}" |
|
59 | type="${button.type}" | |
60 | class="btn ${button.css_class or btn_disposition}" |
|
60 | class="btn ${button.css_class or btn_disposition}" | |
61 | value="${button.value}"> |
|
61 | value="${button.value}"> | |
62 | <i tal:condition="btn_icon" class="${btn_icon}"> </i> |
|
62 | <i tal:condition="btn_icon" class="${btn_icon}"> </i> | |
63 | ${button.title} |
|
63 | ${button.title} | |
64 | </button> |
|
64 | </button> | |
65 | </tal:loop> |
|
65 | </tal:loop> | |
66 | </div> |
|
66 | </div> | |
67 |
|
67 | |||
68 | </fieldset> |
|
68 | </fieldset> | |
69 |
|
69 | |||
70 | <script type="text/javascript" tal:condition="use_ajax"> |
|
70 | <script type="text/javascript" tal:condition="use_ajax"> | |
71 | deform.addCallback( |
|
71 | deform.addCallback( | |
72 | '${formid}', |
|
72 | '${formid}', | |
73 | function(oid) { |
|
73 | function(oid) { | |
74 | var target = '#' + oid; |
|
74 | var target = '#' + oid; | |
75 | var options = { |
|
75 | var options = { | |
76 | target: target, |
|
76 | target: target, | |
77 | replaceTarget: true, |
|
77 | replaceTarget: true, | |
78 | success: function() { |
|
78 | success: function() { | |
79 | deform.processCallbacks(); |
|
79 | deform.processCallbacks(); | |
80 | deform.focusFirstInput(target); |
|
80 | deform.focusFirstInput(target); | |
81 | }, |
|
81 | }, | |
82 | beforeSerialize: function() { |
|
82 | beforeSerialize: function() { | |
83 | // See http://bit.ly/1agBs9Z (hack to fix tinymce-related ajax bug) |
|
83 | // See http://bit.ly/1agBs9Z (hack to fix tinymce-related ajax bug) | |
84 | if ('tinymce' in window) { |
|
84 | if ('tinymce' in window) { | |
85 | $(tinymce.get()).each( |
|
85 | $(tinymce.get()).each( | |
86 | function(i, el) { |
|
86 | function(i, el) { | |
87 | var content = el.getContent(); |
|
87 | var content = el.getContent(); | |
88 | var editor_input = document.getElementById(el.id); |
|
88 | var editor_input = document.getElementById(el.id); | |
89 | editor_input.value = content; |
|
89 | editor_input.value = content; | |
90 | }); |
|
90 | }); | |
91 | } |
|
91 | } | |
92 | } |
|
92 | } | |
93 | }; |
|
93 | }; | |
94 | var extra_options = ${ajax_options} || {}; |
|
94 | var extra_options = ${ajax_options} || {}; | |
95 | $('#' + oid).ajaxForm($.extend(options, extra_options)); |
|
95 | $('#' + oid).ajaxForm($.extend(options, extra_options)); | |
96 | } |
|
96 | } | |
97 | ); |
|
97 | ); | |
98 | </script> |
|
98 | </script> | |
99 |
|
99 | |||
100 | </form> No newline at end of file |
|
100 | </form> |
General Comments 0
You need to be logged in to leave comments.
Login now