author | sylvain.thenault@logilab.fr |
Wed, 18 Feb 2009 18:06:41 +0100 | |
branch | tls-sprint |
changeset 822 | e3d8db01f3f5 |
parent 820 | fcc11122ffe7 |
child 823 | cb8ccbef8fa5 |
permissions | -rw-r--r-- |
0 | 1 |
# -*- coding: utf-8 -*- |
2 |
"""default templates for CubicWeb web client |
|
3 |
||
4 |
:organization: Logilab |
|
580 | 5 |
:copyright: 2001-2009 LOGILAB S.A. (Paris, FRANCE), all rights reserved. |
0 | 6 |
:contact: http://www.logilab.fr/ -- mailto:contact@logilab.fr |
7 |
""" |
|
8 |
__docformat__ = "restructuredtext en" |
|
9 |
||
592
aae01d81c713
have navigation on top and bottom of search results
Arthur Lutz <arthur.lutz@logilab.fr>
parents:
432
diff
changeset
|
10 |
|
0 | 11 |
from logilab.mtconverter import html_escape |
12 |
||
13 |
from cubicweb import NoSelectableObject, ObjectNotFound |
|
822 | 14 |
from cubicweb.selectors import match_kwargs |
816
9cd49a910fce
kill Template class and 'templates' registry
sylvain.thenault@logilab.fr
parents:
811
diff
changeset
|
15 |
from cubicweb.view import View, MainTemplate, NOINDEX, NOFOLLOW |
811 | 16 |
from cubicweb.utils import make_uid, UStringIO |
0 | 17 |
from cubicweb.web.views.baseviews import vid_from_rset |
18 |
||
19 |
# main templates ############################################################## |
|
20 |
||
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
21 |
class LogInOutTemplate(MainTemplate): |
0 | 22 |
|
23 |
def call(self): |
|
24 |
self.set_request_content_type() |
|
25 |
w = self.w |
|
26 |
self.write_doctype() |
|
27 |
lang = self.req.lang |
|
28 |
self.template_header('text/html', self.req._('login_action')) |
|
29 |
w(u'<body>\n') |
|
30 |
self.content(w) |
|
31 |
w(u'</body>') |
|
32 |
||
33 |
def template_header(self, content_type, view=None, page_title='', additional_headers=()): |
|
34 |
w = self.whead |
|
35 |
# explictly close the <base> tag to avoid IE 6 bugs while browsing DOM |
|
36 |
w(u'<base href="%s"></base>' % html_escape(self.req.base_url())) |
|
37 |
w(u'<meta http-equiv="content-type" content="%s; charset=%s"/>\n' |
|
38 |
% (content_type, self.req.encoding)) |
|
39 |
w(NOINDEX) |
|
40 |
w(NOFOLLOW) |
|
41 |
w(u'\n'.join(additional_headers) + u'\n') |
|
819 | 42 |
self.wview('htmlheader', rset=self.rset) |
0 | 43 |
w(u'<title>%s</title>\n' % html_escape(page_title)) |
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
44 |
|
0 | 45 |
|
46 |
class LogInTemplate(LogInOutTemplate): |
|
47 |
id = 'login' |
|
48 |
title = 'log in' |
|
49 |
||
50 |
def content(self, w): |
|
819 | 51 |
self.wview('logform', rset=self.rset, id='loginBox', klass='') |
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
52 |
|
0 | 53 |
|
54 |
class LoggedOutTemplate(LogInOutTemplate): |
|
55 |
id = 'loggedout' |
|
56 |
title = 'logged out' |
|
57 |
||
58 |
def content(self, w): |
|
59 |
msg = self.req._('you have been logged out') |
|
60 |
w(u'<h1 class="noborder">%s</h1>\n' % msg) |
|
61 |
if self.config['anonymous-user']: |
|
62 |
indexurl = self.build_url('view', vid='index', __message=msg) |
|
63 |
w(u'<p><a href="%s">%s</a><p>' % ( |
|
64 |
html_escape(indexurl), |
|
65 |
self.req._('go back to the index page'))) |
|
66 |
||
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
67 |
|
0 | 68 |
class TheMainTemplate(MainTemplate): |
69 |
"""default main template : |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
70 |
|
0 | 71 |
- call header / footer templates |
72 |
- build result set |
|
73 |
- guess and call an appropriate view through the view manager |
|
74 |
""" |
|
75 |
id = 'main' |
|
76 |
||
77 |
def _select_view_and_rset(self): |
|
78 |
req = self.req |
|
79 |
if self.rset is None and not hasattr(req, '_rql_processed'): |
|
80 |
req._rql_processed = True |
|
81 |
rset = self.process_rql(req.form.get('rql')) |
|
82 |
else: |
|
83 |
rset = self.rset |
|
84 |
# handle special "method" param when necessary |
|
85 |
# XXX this should probably not be in the template (controller ?), however: |
|
86 |
# * we need to have the displayed rset |
|
87 |
# * we don't want to handle it in each view |
|
88 |
if rset and rset.rowcount == 1 and '__method' in req.form: |
|
89 |
entity = rset.get_entity(0, 0) |
|
90 |
try: |
|
91 |
method = getattr(entity, req.form.pop('__method')) |
|
92 |
method() |
|
93 |
except Exception, ex: |
|
94 |
self.exception('while handling __method') |
|
95 |
req.set_message(req._("error while handling __method: %s") % req._(ex)) |
|
96 |
vid = req.form.get('vid') or vid_from_rset(req, rset, self.schema) |
|
97 |
try: |
|
98 |
view = self.vreg.select_view(vid, req, rset) |
|
99 |
except ObjectNotFound: |
|
100 |
self.warning("the view %s could not be found", vid) |
|
101 |
req.set_message(req._("The view %s could not be found") % vid) |
|
102 |
vid = vid_from_rset(req, rset, self.schema) |
|
103 |
view = self.vreg.select_view(vid, req, rset) |
|
104 |
except NoSelectableObject: |
|
105 |
if rset: |
|
106 |
req.set_message(req._("The view %s can not be applied to this query") % vid) |
|
107 |
else: |
|
108 |
req.set_message(req._("You have no access to this view or it's not applyable to current data")) |
|
109 |
self.warning("the view %s can not be applied to this query", vid) |
|
110 |
vid = vid_from_rset(req, rset, self.schema) |
|
111 |
view = self.vreg.select_view(vid, req, rset) |
|
112 |
return view, rset |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
113 |
|
0 | 114 |
def call(self): |
115 |
view, rset = self._select_view_and_rset() |
|
116 |
req = self.req |
|
117 |
# update breadcrumps **before** validating cache, unless the view |
|
118 |
# specifies explicitly it should not be added to breadcrumb or the |
|
119 |
# view is a binary view |
|
120 |
if view.add_to_breadcrumbs and not view.binary: |
|
121 |
req.update_breadcrumbs() |
|
122 |
view.set_http_cache_headers() |
|
123 |
req.validate_cache() |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
124 |
with_templates = self.with_templates(view) |
818
dc45e3cc487c
set PageContentTemplate stream for binary views
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
811
diff
changeset
|
125 |
if with_templates: |
0 | 126 |
self.set_request_content_type() |
127 |
content_type = self.content_type |
|
128 |
self.template_header(content_type, view) |
|
818
dc45e3cc487c
set PageContentTemplate stream for binary views
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
811
diff
changeset
|
129 |
else: |
dc45e3cc487c
set PageContentTemplate stream for binary views
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
811
diff
changeset
|
130 |
view.set_request_content_type() |
dc45e3cc487c
set PageContentTemplate stream for binary views
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
811
diff
changeset
|
131 |
self.set_stream(templatable=False) |
819 | 132 |
self.wview('page-content', view=view, rset=rset) |
0 | 133 |
if with_templates: |
134 |
self.template_footer(view) |
|
135 |
||
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
136 |
def with_templates(self, view): |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
137 |
return (not view.binary and view.templatable and |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
138 |
not self.req.form.has_key('__notemplate')) |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
139 |
|
0 | 140 |
def process_rql(self, rql): |
141 |
"""execute rql if specified""" |
|
142 |
if rql: |
|
143 |
self.ensure_ro_rql(rql) |
|
144 |
if not isinstance(rql, unicode): |
|
145 |
rql = unicode(rql, self.req.encoding) |
|
146 |
pp = self.vreg.select_component('magicsearch', self.req) |
|
147 |
self.rset = pp.process_query(rql, self.req) |
|
148 |
return self.rset |
|
149 |
return None |
|
150 |
||
151 |
def template_header(self, content_type, view=None, page_title='', additional_headers=()): |
|
152 |
page_title = page_title or view.page_title() |
|
153 |
additional_headers = additional_headers or view.html_headers() |
|
154 |
self.template_html_header(content_type, page_title, additional_headers) |
|
155 |
self.template_body_header(view) |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
156 |
|
0 | 157 |
def template_html_header(self, content_type, page_title, additional_headers=()): |
158 |
w = self.whead |
|
159 |
lang = self.req.lang |
|
160 |
self.write_doctype() |
|
161 |
w(u'<base href="%s" />' % html_escape(self.req.base_url())) |
|
162 |
w(u'<meta http-equiv="content-type" content="%s; charset=%s"/>\n' |
|
163 |
% (content_type, self.req.encoding)) |
|
164 |
w(u'\n'.join(additional_headers) + u'\n') |
|
819 | 165 |
self.wview('htmlheader', rset=self.rset) |
0 | 166 |
if page_title: |
167 |
w(u'<title>%s</title>\n' % html_escape(page_title)) |
|
168 |
||
169 |
def template_body_header(self, view): |
|
170 |
w = self.w |
|
171 |
w(u'<body>\n') |
|
819 | 172 |
self.wview('header', rset=self.rset, view=view) |
0 | 173 |
w(u'<div id="page"><table width="100%" border="0" id="mainLayout"><tr>\n') |
174 |
self.nav_column(view, 'left') |
|
175 |
w(u'<td id="contentcol">\n') |
|
176 |
rqlcomp = self.vreg.select_component('rqlinput', self.req, self.rset) |
|
177 |
if rqlcomp: |
|
178 |
rqlcomp.dispatch(w=self.w, view=view) |
|
179 |
msgcomp = self.vreg.select_component('applmessages', self.req, self.rset) |
|
180 |
if msgcomp: |
|
181 |
msgcomp.dispatch(w=self.w) |
|
182 |
self.content_header(view) |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
183 |
|
0 | 184 |
def template_footer(self, view=None): |
185 |
self.content_footer(view) |
|
186 |
self.w(u'</td>\n') |
|
187 |
self.nav_column(view, 'right') |
|
188 |
self.w(u'</tr></table></div>\n') |
|
819 | 189 |
self.wview('footer', rset=self.rset) |
0 | 190 |
self.w(u'</body>') |
191 |
||
192 |
def nav_column(self, view, context): |
|
193 |
boxes = list(self.vreg.possible_vobjects('boxes', self.req, self.rset, |
|
194 |
view=view, context=context)) |
|
195 |
if boxes: |
|
196 |
self.w(u'<td class="navcol"><div class="navboxes">\n') |
|
197 |
for box in boxes: |
|
198 |
box.dispatch(w=self.w, view=view) |
|
199 |
self.w(u'</div></td>\n') |
|
200 |
||
201 |
def content_header(self, view=None): |
|
202 |
"""by default, display informal messages in content header""" |
|
819 | 203 |
self.wview('contentheader', rset=self.rset, view=view) |
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
204 |
|
0 | 205 |
def content_footer(self, view=None): |
819 | 206 |
self.wview('contentfooter', rset=self.rset, view=view) |
0 | 207 |
|
208 |
||
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
209 |
class PageContentTemplate(TheMainTemplate): |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
210 |
id = 'page-content' |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
211 |
|
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
212 |
def call(self, view=None, rset=None): |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
213 |
if view is None: |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
214 |
view, rset = self._select_view_and_rset() |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
215 |
with_templates = self.with_templates(view) |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
216 |
w = self.w |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
217 |
if with_templates: |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
218 |
w(u'<div id="pageContent">\n') |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
219 |
vtitle = self.req.form.get('vtitle') |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
220 |
if vtitle: |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
221 |
w(u'<h1 class="vtitle">%s</h1>\n' % html_escape(vtitle)) |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
222 |
# display entity type restriction component |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
223 |
etypefilter = self.vreg.select_component('etypenavigation', |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
224 |
self.req, self.rset) |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
225 |
if etypefilter and etypefilter.propval('visible'): |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
226 |
etypefilter.dispatch(w=w) |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
227 |
self.nav_html = UStringIO() |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
228 |
self.pagination(self.req, self.rset, self.nav_html.write, |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
229 |
not (view and view.need_navigation)) |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
230 |
w(_(self.nav_html.getvalue())) |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
231 |
w(u'<div id="contentmain">\n') |
818
dc45e3cc487c
set PageContentTemplate stream for binary views
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
811
diff
changeset
|
232 |
else: |
dc45e3cc487c
set PageContentTemplate stream for binary views
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
811
diff
changeset
|
233 |
self.set_stream(templatable=False) |
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
234 |
if view.binary: |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
235 |
# have to replace our unicode stream using view's binary stream |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
236 |
view.dispatch() |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
237 |
assert self._stream, 'duh, template used as a sub-view ?? (%s)' % self._stream |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
238 |
self._stream = view._stream |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
239 |
else: |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
240 |
view.dispatch(w=w) |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
241 |
if with_templates: |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
242 |
w(u'</div>\n') # close id=contentmain |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
243 |
w(_(self.nav_html.getvalue())) |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
244 |
w(u'</div>\n') # closes id=pageContent |
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
245 |
|
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
246 |
|
0 | 247 |
class ErrorTemplate(TheMainTemplate): |
248 |
"""fallback template if an internal error occured during displaying the |
|
249 |
main template. This template may be called for authentication error, |
|
250 |
which means that req.cnx and req.user may not be set. |
|
251 |
""" |
|
252 |
id = 'error' |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
253 |
|
0 | 254 |
def call(self): |
255 |
"""display an unexpected error""" |
|
256 |
self.set_request_content_type() |
|
257 |
self.req.reset_headers() |
|
258 |
view = self.vreg.select_view('error', self.req, self.rset) |
|
259 |
self.template_header(self.content_type, view, self.req._('an error occured'), |
|
260 |
[NOINDEX, NOFOLLOW]) |
|
261 |
view.dispatch(w=self.w) |
|
262 |
self.template_footer(view) |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
263 |
|
0 | 264 |
def template_header(self, content_type, view=None, page_title='', additional_headers=()): |
265 |
w = self.whead |
|
266 |
lang = self.req.lang |
|
267 |
self.write_doctype() |
|
268 |
w(u'<meta http-equiv="content-type" content="%s; charset=%s"/>\n' |
|
269 |
% (content_type, self.req.encoding)) |
|
270 |
w(u'\n'.join(additional_headers)) |
|
819 | 271 |
self.wview('htmlheader', rset=self.rset) |
0 | 272 |
w(u'<title>%s</title>\n' % html_escape(page_title)) |
273 |
self.w(u'<body>\n') |
|
274 |
||
275 |
def template_footer(self, view=None): |
|
276 |
self.w(u'</body>') |
|
277 |
||
278 |
||
279 |
class SimpleMainTemplate(TheMainTemplate): |
|
280 |
||
281 |
id = 'main-no-top' |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
282 |
|
163
451a3e35dbcb
repairing error from preceding commit
Laure Bourgois <Laure.Bourgois@logilab.fr>
parents:
158
diff
changeset
|
283 |
def template_header(self, content_type, view=None, page_title='', additional_headers=()): |
451a3e35dbcb
repairing error from preceding commit
Laure Bourgois <Laure.Bourgois@logilab.fr>
parents:
158
diff
changeset
|
284 |
page_title = page_title or view.page_title() |
0 | 285 |
additional_headers = additional_headers or view.html_headers() |
286 |
whead = self.whead |
|
287 |
lang = self.req.lang |
|
288 |
self.write_doctype() |
|
289 |
whead(u'<meta http-equiv="content-type" content="%s; charset=%s"/>\n' |
|
290 |
% (content_type, self.req.encoding)) |
|
291 |
whead(u'\n'.join(additional_headers) + u'\n') |
|
819 | 292 |
self.wview('htmlheader', rset=self.rset) |
0 | 293 |
w = self.w |
294 |
w(u'<title>%s</title>\n' % html_escape(page_title)) |
|
295 |
w(u'<body>\n') |
|
296 |
w(u'<div id="page">') |
|
297 |
w(u'<table width="100%" height="100%" border="0"><tr>\n') |
|
298 |
w(u'<td class="navcol">\n') |
|
299 |
self.topleft_header() |
|
300 |
boxes = list(self.vreg.possible_vobjects('boxes', self.req, self.rset, |
|
301 |
view=view, context='left')) |
|
302 |
if boxes: |
|
303 |
w(u'<div class="navboxes">\n') |
|
304 |
for box in boxes: |
|
305 |
box.dispatch(w=w) |
|
306 |
self.w(u'</div>\n') |
|
307 |
w(u'</td>') |
|
308 |
w(u'<td id="contentcol" rowspan="2">') |
|
309 |
w(u'<div id="pageContent">\n') |
|
310 |
vtitle = self.req.form.get('vtitle') |
|
311 |
if vtitle: |
|
432 | 312 |
w(u'<h1 class="vtitle">%s</h1>' % html_escape(vtitle)) |
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
313 |
|
0 | 314 |
def topleft_header(self): |
315 |
self.w(u'<table id="header"><tr>\n') |
|
316 |
self.w(u'<td>') |
|
317 |
self.vreg.select_component('logo', self.req, self.rset).dispatch(w=self.w) |
|
318 |
self.w(u'</td>\n') |
|
319 |
self.w(u'</tr></table>\n') |
|
320 |
||
321 |
# page parts templates ######################################################## |
|
322 |
||
816
9cd49a910fce
kill Template class and 'templates' registry
sylvain.thenault@logilab.fr
parents:
811
diff
changeset
|
323 |
class HTMLHeader(View): |
0 | 324 |
"""default html headers""" |
325 |
id = 'htmlheader' |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
326 |
|
0 | 327 |
def call(self, **kwargs): |
328 |
self.favicon() |
|
329 |
self.stylesheets() |
|
330 |
self.javascripts() |
|
331 |
self.alternates() |
|
332 |
self.pageid() |
|
333 |
||
334 |
def favicon(self): |
|
335 |
favicon = self.req.external_resource('FAVICON', None) |
|
336 |
if favicon: |
|
337 |
self.whead(u'<link rel="shortcut icon" href="%s"/>\n' % favicon) |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
338 |
|
0 | 339 |
def stylesheets(self): |
340 |
req = self.req |
|
341 |
add_css = req.add_css |
|
342 |
for css in req.external_resource('STYLESHEETS'): |
|
343 |
add_css(css, localfile=False) |
|
344 |
for css in req.external_resource('STYLESHEETS_PRINT'): |
|
345 |
add_css(css, u'print', localfile=False) |
|
346 |
for css in req.external_resource('IE_STYLESHEETS'): |
|
347 |
add_css(css, localfile=False, ieonly=True) |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
348 |
|
0 | 349 |
def javascripts(self): |
350 |
for jscript in self.req.external_resource('JAVASCRIPTS'): |
|
351 |
self.req.add_js(jscript, localfile=False) |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
352 |
|
0 | 353 |
def alternates(self): |
142
0425ee84cfa6
add selector to test if result set is an object (for rss feed component)
Laure Bourgois <Laure.Bourgois@logilab.fr>
parents:
29
diff
changeset
|
354 |
urlgetter = self.vreg.select_component('rss_feed_url', self.req, self.rset) |
0425ee84cfa6
add selector to test if result set is an object (for rss feed component)
Laure Bourgois <Laure.Bourgois@logilab.fr>
parents:
29
diff
changeset
|
355 |
if urlgetter is not None: |
0425ee84cfa6
add selector to test if result set is an object (for rss feed component)
Laure Bourgois <Laure.Bourgois@logilab.fr>
parents:
29
diff
changeset
|
356 |
url = urlgetter.feed_url() |
0 | 357 |
self.whead(u'<link rel="alternate" type="application/rss+xml" title="RSS feed" href="%s"/>\n' |
144 | 358 |
% html_escape(url)) |
0 | 359 |
|
360 |
def pageid(self): |
|
361 |
req = self.req |
|
362 |
pid = make_uid(id(req)) |
|
363 |
req.pageid = pid |
|
364 |
req.html_headers.define_var('pageid', pid); |
|
365 |
||
366 |
||
816
9cd49a910fce
kill Template class and 'templates' registry
sylvain.thenault@logilab.fr
parents:
811
diff
changeset
|
367 |
class HTMLPageHeader(View): |
0 | 368 |
"""default html page header""" |
369 |
id = 'header' |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
370 |
|
0 | 371 |
def call(self, view, **kwargs): |
372 |
self.main_header(view) |
|
373 |
self.w(u''' |
|
374 |
<div id="stateheader">''') |
|
375 |
self.state_header() |
|
376 |
self.w(u''' |
|
377 |
</div> |
|
378 |
''') |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
379 |
|
0 | 380 |
def main_header(self, view): |
381 |
"""build the top menu with authentification info and the rql box""" |
|
382 |
self.w(u'<table id="header"><tr>\n') |
|
383 |
self.w(u'<td id="firstcolumn">') |
|
384 |
self.vreg.select_component('logo', self.req, self.rset).dispatch(w=self.w) |
|
385 |
self.w(u'</td>\n') |
|
386 |
# appliname and breadcrumbs |
|
387 |
self.w(u'<td id="headtext">') |
|
388 |
comp = self.vreg.select_component('appliname', self.req, self.rset) |
|
389 |
if comp and comp.propval('visible'): |
|
390 |
comp.dispatch(w=self.w) |
|
391 |
comp = self.vreg.select_component('breadcrumbs', self.req, self.rset, view=view) |
|
392 |
if comp and comp.propval('visible'): |
|
393 |
comp.dispatch(w=self.w, view=view) |
|
394 |
self.w(u'</td>') |
|
395 |
# logged user and help |
|
396 |
self.w(u'<td>\n') |
|
397 |
comp = self.vreg.select_component('loggeduserlink', self.req, self.rset) |
|
398 |
comp.dispatch(w=self.w) |
|
399 |
self.w(u'</td><td>') |
|
400 |
helpcomp = self.vreg.select_component('help', self.req, self.rset) |
|
401 |
if helpcomp: # may not be available if Card is not defined in the schema |
|
402 |
helpcomp.dispatch(w=self.w) |
|
403 |
self.w(u'</td>') |
|
404 |
# lastcolumn |
|
405 |
self.w(u'<td id="lastcolumn">') |
|
406 |
self.w(u'</td>\n') |
|
407 |
self.w(u'</tr></table>\n') |
|
819 | 408 |
self.wview('logform', rset=self.rset, id='popupLoginBox', klass='hidden', |
409 |
title=False, message=False) |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
410 |
|
0 | 411 |
def state_header(self): |
412 |
state = self.req.search_state |
|
413 |
if state[0] == 'normal': |
|
414 |
return |
|
415 |
_ = self.req._ |
|
416 |
value = self.view('oneline', self.req.eid_rset(state[1][1])) |
|
417 |
msg = ' '.join((_("searching for"), |
|
418 |
display_name(self.req, state[1][3]), |
|
419 |
_("to associate with"), value, |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
420 |
_("by relation"), '"', |
0 | 421 |
display_name(self.req, state[1][2], state[1][0]), |
422 |
'"')) |
|
423 |
return self.w(u'<div class="stateMessage">%s</div>' % msg) |
|
424 |
||
425 |
||
426 |
||
816
9cd49a910fce
kill Template class and 'templates' registry
sylvain.thenault@logilab.fr
parents:
811
diff
changeset
|
427 |
class HTMLPageFooter(View): |
0 | 428 |
"""default html page footer: include logo if any, and close the HTML body |
429 |
""" |
|
430 |
id = 'footer' |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
431 |
|
0 | 432 |
def call(self, **kwargs): |
433 |
req = self.req |
|
434 |
self.w(u'<div class="footer">') |
|
435 |
# XXX Take object from the registry if in there? would be |
|
436 |
# better anyway |
|
437 |
from cubicweb.web.views.wdoc import ChangeLogView |
|
438 |
self.w(u'<a href="%s">%s</a> | ' % (req.build_url('changelog'), |
|
439 |
req._(ChangeLogView.title).lower())) |
|
440 |
self.w(u'<a href="%s">%s</a> | ' % (req.build_url('doc/about'), |
|
441 |
req._('about this site'))) |
|
580 | 442 |
self.w(u'© 2001-2009 <a href="http://www.logilab.fr">Logilab S.A.</a>') |
0 | 443 |
self.w(u'</div>') |
444 |
||
445 |
||
816
9cd49a910fce
kill Template class and 'templates' registry
sylvain.thenault@logilab.fr
parents:
811
diff
changeset
|
446 |
class HTMLContentHeader(View): |
0 | 447 |
"""default html page content header: |
448 |
* include message component if selectable for this request |
|
449 |
* include selectable content navigation components |
|
450 |
""" |
|
451 |
id = 'contentheader' |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
452 |
|
0 | 453 |
def call(self, view, **kwargs): |
454 |
"""by default, display informal messages in content header""" |
|
455 |
components = self.vreg.possible_vobjects('contentnavigation', |
|
456 |
self.req, self.rset, |
|
457 |
view=view, context='navtop') |
|
458 |
if components: |
|
459 |
self.w(u'<div id="contentheader">') |
|
460 |
for comp in components: |
|
461 |
comp.dispatch(w=self.w, view=view) |
|
462 |
self.w(u'</div><div class="clear"></div>') |
|
463 |
||
464 |
||
816
9cd49a910fce
kill Template class and 'templates' registry
sylvain.thenault@logilab.fr
parents:
811
diff
changeset
|
465 |
class HTMLContentFooter(View): |
0 | 466 |
"""default html page content footer: include selectable content navigation |
467 |
components |
|
468 |
""" |
|
469 |
id = 'contentfooter' |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
470 |
|
0 | 471 |
def call(self, view, **kwargs): |
472 |
components = self.vreg.possible_vobjects('contentnavigation', |
|
473 |
self.req, self.rset, |
|
474 |
view=view, context='navbottom') |
|
475 |
if components: |
|
476 |
self.w(u'<div id="contentfooter">') |
|
477 |
for comp in components: |
|
478 |
comp.dispatch(w=self.w, view=view) |
|
479 |
self.w(u'</div>') |
|
480 |
||
481 |
||
816
9cd49a910fce
kill Template class and 'templates' registry
sylvain.thenault@logilab.fr
parents:
811
diff
changeset
|
482 |
class LogFormTemplate(View): |
0 | 483 |
id = 'logform' |
822 | 484 |
__select__ = match_kwargs('id', 'title') |
485 |
||
0 | 486 |
title = 'log in' |
822 | 487 |
|
0 | 488 |
def call(self, id, klass, title=True, message=True): |
489 |
self.req.add_css('cubicweb.login.css') |
|
490 |
self.w(u'<div id="%s" class="%s">' % (id, klass)) |
|
491 |
if title: |
|
492 |
self.w(u'<div id="loginTitle">%s</div>' |
|
493 |
% self.req.property_value('ui.site-title')) |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
494 |
self.w(u'<div id="loginContent">\n') |
0 | 495 |
|
496 |
if message: |
|
497 |
self.display_message() |
|
498 |
if self.config['auth-mode'] == 'http': |
|
499 |
# HTTP authentication |
|
500 |
pass |
|
501 |
else: |
|
502 |
# Cookie authentication |
|
503 |
self.login_form(id) |
|
504 |
self.w(u'</div></div>\n') |
|
505 |
||
506 |
def display_message(self): |
|
507 |
message = self.req.message |
|
508 |
if message: |
|
509 |
self.w(u'<div class="simpleMessage">%s</div>\n' % message) |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
510 |
|
0 | 511 |
def login_form(self, id): |
512 |
_ = self.req._ |
|
513 |
self.w(u'<form method="post" action="%s" id="login_form">\n' |
|
514 |
% html_escape(login_form_url(self.config, self.req))) |
|
515 |
self.w(u'<table>\n') |
|
516 |
self.w(u'<tr>\n') |
|
637
a78ffe83fadc
change label for new registered users
Arthur Lutz <arthur.lutz@logilab.fr>
parents:
609
diff
changeset
|
517 |
self.w(u'<td><label for="__login">%s</label></td>' % _('login or email')) |
0 | 518 |
self.w(u'<td><input name="__login" id="__login" class="data" type="text" /></td>') |
519 |
self.w(u'</tr><tr>\n') |
|
520 |
self.w(u'<td><label for="__password" >%s</label></td>' % _('password')) |
|
521 |
self.w(u'<td><input name="__password" id="__password" class="data" type="password" /></td>\n') |
|
522 |
self.w(u'</tr><tr>\n') |
|
523 |
self.w(u'<td> </td><td><input type="submit" class="loginButton right" value="%s" />\n</td>' % _('log in')) |
|
524 |
self.w(u'</tr>\n') |
|
525 |
self.w(u'</table>\n') |
|
526 |
self.w(u'</form>\n') |
|
29
7d14f1eadded
fix focus problems on login inputs
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
16
diff
changeset
|
527 |
self.req.html_headers.add_onload('jQuery("#__login:visible").focus()') |
0 | 528 |
|
641
ed668804af37
split main template into main template + page-content template
Adrien Di Mascio <Adrien.DiMascio@logilab.fr>
parents:
609
diff
changeset
|
529 |
|
0 | 530 |
def login_form_url(config, req): |
531 |
if req.https: |
|
532 |
return req.url() |
|
533 |
if config.get('https-url'): |
|
534 |
return req.url().replace(req.base_url(), config['https-url']) |
|
535 |
return req.url() |
|
536 |