VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/testmanager/webui/wuicontentbase.py@ 106061

Last change on this file since 106061 was 106061, checked in by vboxsync, 4 months ago

Copyright year updates by scm.

  • Property svn:eol-style set to native
  • Property svn:keywords set to Author Date Id Revision
File size: 53.9 KB
Line 
1# -*- coding: utf-8 -*-
2# $Id: wuicontentbase.py 106061 2024-09-16 14:03:52Z vboxsync $
3
4"""
5Test Manager Web-UI - Content Base Classes.
6"""
7
8__copyright__ = \
9"""
10Copyright (C) 2012-2024 Oracle and/or its affiliates.
11
12This file is part of VirtualBox base platform packages, as
13available from https://www.virtualbox.org.
14
15This program is free software; you can redistribute it and/or
16modify it under the terms of the GNU General Public License
17as published by the Free Software Foundation, in version 3 of the
18License.
19
20This program is distributed in the hope that it will be useful, but
21WITHOUT ANY WARRANTY; without even the implied warranty of
22MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
23General Public License for more details.
24
25You should have received a copy of the GNU General Public License
26along with this program; if not, see <https://www.gnu.org/licenses>.
27
28The contents of this file may alternatively be used under the terms
29of the Common Development and Distribution License Version 1.0
30(CDDL), a copy of it is provided in the "COPYING.CDDL" file included
31in the VirtualBox distribution, in which case the provisions of the
32CDDL are applicable instead of those of the GPL.
33
34You may elect to license modified versions of this file under the
35terms and conditions of either the GPL or the CDDL or both.
36
37SPDX-License-Identifier: GPL-3.0-only OR CDDL-1.0
38"""
39__version__ = "$Revision: 106061 $"
40
41
42# Standard python imports.
43import copy;
44import sys;
45
46# Validation Kit imports.
47from common import utils, webutils;
48from testmanager import config;
49from testmanager.webui.wuibase import WuiDispatcherBase, WuiException;
50from testmanager.webui.wuihlpform import WuiHlpForm;
51from testmanager.core import db;
52from testmanager.core.base import AttributeChangeEntryPre;
53
54# Python 3 hacks:
55if sys.version_info[0] >= 3:
56 unicode = str; # pylint: disable=redefined-builtin,invalid-name
57
58
59class WuiHtmlBase(object): # pylint: disable=too-few-public-methods
60 """
61 Base class for HTML objects.
62 """
63
64 def __init__(self):
65 """Dummy init to shut up pylint."""
66 pass; # pylint: disable=unnecessary-pass
67
68 def toHtml(self):
69
70 """
71 Must be overridden by sub-classes.
72 """
73 assert False;
74 return '';
75
76 def __str__(self):
77 """ String representation is HTML, simplifying formatting and such. """
78 return self.toHtml();
79
80
81class WuiLinkBase(WuiHtmlBase): # pylint: disable=too-few-public-methods
82 """
83 For passing links from WuiListContentBase._formatListEntry.
84 """
85
86 def __init__(self, sName, sUrlBase, dParams = None, sConfirm = None, sTitle = None,
87 sFragmentId = None, fBracketed = True, sExtraAttrs = ''):
88 WuiHtmlBase.__init__(self);
89 self.sName = sName
90 self.sUrl = sUrlBase
91 self.sConfirm = sConfirm;
92 self.sTitle = sTitle;
93 self.fBracketed = fBracketed;
94 self.sExtraAttrs = sExtraAttrs;
95
96 if dParams:
97 # Do some massaging of None arguments.
98 dParams = dict(dParams);
99 for sKey in dParams:
100 if dParams[sKey] is None:
101 dParams[sKey] = '';
102 self.sUrl += '?' + webutils.encodeUrlParams(dParams);
103
104 if sFragmentId is not None:
105 self.sUrl += '#' + sFragmentId;
106
107 def setBracketed(self, fBracketed):
108 """Changes the bracketing style."""
109 self.fBracketed = fBracketed;
110 return True;
111
112 def toHtml(self):
113 """
114 Returns a simple HTML anchor element.
115 """
116 sExtraAttrs = self.sExtraAttrs;
117 if self.sConfirm is not None:
118 sExtraAttrs += 'onclick=\'return confirm("%s");\' ' % (webutils.escapeAttr(self.sConfirm),);
119 if self.sTitle is not None:
120 sExtraAttrs += 'title="%s" ' % (webutils.escapeAttr(self.sTitle),);
121 if sExtraAttrs and sExtraAttrs[-1] != ' ':
122 sExtraAttrs += ' ';
123
124 sFmt = '[<a %shref="%s">%s</a>]';
125 if not self.fBracketed:
126 sFmt = '<a %shref="%s">%s</a>';
127 return sFmt % (sExtraAttrs, webutils.escapeAttr(self.sUrl), webutils.escapeElem(self.sName));
128
129 @staticmethod
130 def estimateStringWidth(sString):
131 """
132 Takes a string and estimate it's width so the caller can pad with
133 U+2002 before tab in a title text. This is very very rough.
134 """
135 cchWidth = 0;
136 for ch in sString:
137 if ch.isupper() or ch in u'wm\u2007\u2003\u2001\u3000':
138 cchWidth += 2;
139 else:
140 cchWidth += 1;
141 return cchWidth;
142
143 @staticmethod
144 def getStringWidthPaddingEx(cchWidth, cchMaxWidth):
145 """ Works with estiamteStringWidth(). """
146 if cchWidth + 2 <= cchMaxWidth:
147 return u'\u2002' * ((cchMaxWidth - cchWidth) * 2 // 3)
148 return u'';
149
150 @staticmethod
151 def getStringWidthPadding(sString, cchMaxWidth):
152 """ Works with estiamteStringWidth(). """
153 return WuiLinkBase.getStringWidthPaddingEx(WuiLinkBase.estimateStringWidth(sString), cchMaxWidth);
154
155 @staticmethod
156 def padStringToWidth(sString, cchMaxWidth):
157 """ Works with estimateStringWidth. """
158 cchWidth = WuiLinkBase.estimateStringWidth(sString);
159 if cchWidth < cchMaxWidth:
160 return sString + WuiLinkBase.getStringWidthPaddingEx(cchWidth, cchMaxWidth);
161 return sString;
162
163
164class WuiTmLink(WuiLinkBase): # pylint: disable=too-few-public-methods
165 """ Local link to the test manager. """
166
167 kdDbgParams = [];
168
169 def __init__(self, sName, sUrlBase, dParams = None, sConfirm = None, sTitle = None,
170 sFragmentId = None, fBracketed = True):
171
172 # Add debug parameters if necessary.
173 if self.kdDbgParams:
174 if not dParams:
175 dParams = dict(self.kdDbgParams);
176 else:
177 dParams = dict(dParams);
178 for sKey in self.kdDbgParams:
179 if sKey not in dParams:
180 dParams[sKey] = self.kdDbgParams[sKey];
181
182 WuiLinkBase.__init__(self, sName, sUrlBase, dParams, sConfirm, sTitle, sFragmentId, fBracketed);
183
184
185class WuiAdminLink(WuiTmLink): # pylint: disable=too-few-public-methods
186 """ Local link to the test manager's admin portion. """
187
188 def __init__(self, sName, sAction, tsEffectiveDate = None, dParams = None, sConfirm = None, sTitle = None,
189 sFragmentId = None, fBracketed = True):
190 from testmanager.webui.wuiadmin import WuiAdmin;
191 if not dParams:
192 dParams = {};
193 else:
194 dParams = dict(dParams);
195 if sAction is not None:
196 dParams[WuiAdmin.ksParamAction] = sAction;
197 if tsEffectiveDate is not None:
198 dParams[WuiAdmin.ksParamEffectiveDate] = tsEffectiveDate;
199 WuiTmLink.__init__(self, sName, WuiAdmin.ksScriptName, dParams = dParams, sConfirm = sConfirm, sTitle = sTitle,
200 sFragmentId = sFragmentId, fBracketed = fBracketed);
201
202class WuiMainLink(WuiTmLink): # pylint: disable=too-few-public-methods
203 """ Local link to the test manager's main portion. """
204
205 def __init__(self, sName, sAction, dParams = None, sConfirm = None, sTitle = None, sFragmentId = None, fBracketed = True):
206 if not dParams:
207 dParams = {};
208 else:
209 dParams = dict(dParams);
210 from testmanager.webui.wuimain import WuiMain;
211 if sAction is not None:
212 dParams[WuiMain.ksParamAction] = sAction;
213 WuiTmLink.__init__(self, sName, WuiMain.ksScriptName, dParams = dParams, sConfirm = sConfirm, sTitle = sTitle,
214 sFragmentId = sFragmentId, fBracketed = fBracketed);
215
216class WuiSvnLink(WuiLinkBase): # pylint: disable=too-few-public-methods
217 """
218 For linking to a SVN revision.
219 """
220 def __init__(self, iRevision, sName = None, fBracketed = True, sExtraAttrs = ''):
221 if sName is None:
222 sName = 'r%s' % (iRevision,);
223 WuiLinkBase.__init__(self, sName, config.g_ksTracLogUrlPrefix, { 'rev': iRevision,},
224 fBracketed = fBracketed, sExtraAttrs = sExtraAttrs);
225
226class WuiSvnLinkWithTooltip(WuiSvnLink): # pylint: disable=too-few-public-methods
227 """
228 For linking to a SVN revision with changelog tooltip.
229 """
230 def __init__(self, iRevision, sRepository, sName = None, fBracketed = True):
231 sExtraAttrs = ' onmouseover="return svnHistoryTooltipShow(event,\'%s\',%s);" onmouseout="return tooltipHide();"' \
232 % ( sRepository, iRevision, );
233 WuiSvnLink.__init__(self, iRevision, sName = sName, fBracketed = fBracketed, sExtraAttrs = sExtraAttrs);
234
235class WuiBuildLogLink(WuiLinkBase):
236 """
237 For linking to a build log.
238 """
239 def __init__(self, sUrl, sName = None, fBracketed = True):
240 assert sUrl;
241 if sName is None:
242 sName = 'Build log';
243 if not webutils.hasSchema(sUrl):
244 WuiLinkBase.__init__(self, sName, config.g_ksBuildLogUrlPrefix + sUrl, fBracketed = fBracketed);
245 else:
246 WuiLinkBase.__init__(self, sName, sUrl, fBracketed = fBracketed);
247
248class WuiRawHtml(WuiHtmlBase): # pylint: disable=too-few-public-methods
249 """
250 For passing raw html from WuiListContentBase._formatListEntry.
251 """
252 def __init__(self, sHtml):
253 self.sHtml = sHtml;
254 WuiHtmlBase.__init__(self);
255
256 def toHtml(self):
257 return self.sHtml;
258
259class WuiHtmlKeeper(WuiHtmlBase): # pylint: disable=too-few-public-methods
260 """
261 For keeping a list of elements, concatenating their toHtml output together.
262 """
263 def __init__(self, aoInitial = None, sSep = ' '):
264 WuiHtmlBase.__init__(self);
265 self.sSep = sSep;
266 self.aoKept = [];
267 if aoInitial is not None:
268 if isinstance(aoInitial, WuiHtmlBase):
269 self.aoKept.append(aoInitial);
270 else:
271 self.aoKept.extend(aoInitial);
272
273 def append(self, oObject):
274 """ Appends one objects. """
275 self.aoKept.append(oObject);
276
277 def extend(self, aoObjects):
278 """ Appends a list of objects. """
279 self.aoKept.extend(aoObjects);
280
281 def toHtml(self):
282 return self.sSep.join(oObj.toHtml() for oObj in self.aoKept);
283
284class WuiSpanText(WuiRawHtml): # pylint: disable=too-few-public-methods
285 """
286 Outputs the given text within a span of the given CSS class.
287 """
288 def __init__(self, sSpanClass, sText, sTitle = None):
289 if sTitle is None:
290 WuiRawHtml.__init__(self,
291 u'<span class="%s">%s</span>'
292 % ( webutils.escapeAttr(sSpanClass), webutils.escapeElem(sText),));
293 else:
294 WuiRawHtml.__init__(self,
295 u'<span class="%s" title="%s">%s</span>'
296 % ( webutils.escapeAttr(sSpanClass), webutils.escapeAttr(sTitle), webutils.escapeElem(sText),));
297
298class WuiElementText(WuiRawHtml): # pylint: disable=too-few-public-methods
299 """
300 Outputs the given element text.
301 """
302 def __init__(self, sText):
303 WuiRawHtml.__init__(self, webutils.escapeElem(sText));
304
305
306class WuiContentBase(object): # pylint: disable=too-few-public-methods
307 """
308 Base for the content classes.
309 """
310
311 ## The text/symbol for a very short add link.
312 ksShortAddLink = u'\u2795'
313 ## HTML hex entity string for ksShortAddLink.
314 ksShortAddLinkHtml = '&#x2795;;'
315 ## The text/symbol for a very short edit link.
316 ksShortEditLink = u'\u270D'
317 ## HTML hex entity string for ksShortDetailsLink.
318 ksShortEditLinkHtml = '&#x270d;'
319 ## The text/symbol for a very short details link.
320 ksShortDetailsLink = u'\U0001f6c8\ufe0e'
321 ## HTML hex entity string for ksShortDetailsLink.
322 ksShortDetailsLinkHtml = '&#x1f6c8;;&#xfe0e;'
323 ## The text/symbol for a very short change log / details / previous page link.
324 ksShortChangeLogLink = u'\u2397'
325 ## HTML hex entity string for ksShortDetailsLink.
326 ksShortChangeLogLinkHtml = '&#x2397;'
327 ## The text/symbol for a very short reports link.
328 ksShortReportLink = u'\U0001f4ca\ufe0e'
329 ## HTML hex entity string for ksShortReportLink.
330 ksShortReportLinkHtml = '&#x1f4ca;&#xfe0e;'
331 ## The text/symbol for a very short test results link.
332 ksShortTestResultsLink = u'\U0001f5d0\ufe0e'
333
334
335 def __init__(self, fnDPrint = None, oDisp = None):
336 self._oDisp = oDisp; # WuiDispatcherBase.
337 self._fnDPrint = fnDPrint;
338 if fnDPrint is None and oDisp is not None:
339 self._fnDPrint = oDisp.dprint;
340
341 def dprint(self, sText):
342 """ Debug printing. """
343 if self._fnDPrint:
344 self._fnDPrint(sText);
345
346 @staticmethod
347 def formatTsShort(oTs):
348 """
349 Formats a timestamp (db rep) into a short form.
350 """
351 oTsZulu = db.dbTimestampToZuluDatetime(oTs);
352 sTs = oTsZulu.strftime('%Y-%m-%d %H:%M:%SZ');
353 return unicode(sTs).replace('-', u'\u2011').replace(' ', u'\u00a0');
354
355 def getNowTs(self):
356 """ Gets a database compatible current timestamp from python. See db.dbTimestampPythonNow(). """
357 return db.dbTimestampPythonNow();
358
359 def formatIntervalShort(self, oInterval):
360 """
361 Formats an interval (db rep) into a short form.
362 """
363 # default formatting for negative intervals.
364 if oInterval.days < 0:
365 return str(oInterval);
366
367 # Figure the hour, min and sec counts.
368 cHours = oInterval.seconds // 3600;
369 cMinutes = (oInterval.seconds % 3600) // 60;
370 cSeconds = oInterval.seconds - cHours * 3600 - cMinutes * 60;
371
372 # Tailor formatting to the interval length.
373 if oInterval.days > 0:
374 if oInterval.days > 1:
375 return '%d days, %d:%02d:%02d' % (oInterval.days, cHours, cMinutes, cSeconds);
376 return '1 day, %d:%02d:%02d' % (cHours, cMinutes, cSeconds);
377 if cMinutes > 0 or cSeconds >= 30 or cHours > 0:
378 return '%d:%02d:%02d' % (cHours, cMinutes, cSeconds);
379 if cSeconds >= 10:
380 return '%d.%ds' % (cSeconds, oInterval.microseconds // 100000);
381 if cSeconds > 0:
382 return '%d.%02ds' % (cSeconds, oInterval.microseconds // 10000);
383 return '%d ms' % (oInterval.microseconds // 1000,);
384
385 @staticmethod
386 def genericPageWalker(iCurItem, cItems, sHrefFmt, cWidth = 11, iBase = 1, sItemName = 'page'):
387 """
388 Generic page walker generator.
389
390 sHrefFmt has three %s sequences:
391 1. The first is the page number link parameter (0-based).
392 2. The title text, iBase-based number or text.
393 3. The link text, iBase-based number or text.
394 """
395
396 # Calc display range.
397 iStart = 0 if iCurItem - cWidth // 2 <= cWidth // 4 else iCurItem - cWidth // 2;
398 iEnd = iStart + cWidth;
399 if iEnd > cItems:
400 iEnd = cItems;
401 if cItems > cWidth:
402 iStart = cItems - cWidth;
403
404 sHtml = u'';
405
406 # Previous page (using << >> because &laquo; and &raquo are too tiny).
407 if iCurItem > 0:
408 sHtml += '%s&nbsp;&nbsp;' % sHrefFmt % (iCurItem - 1, 'previous ' + sItemName, '&lt;&lt;');
409 else:
410 sHtml += '&lt;&lt;&nbsp;&nbsp;';
411
412 # 1 2 3 4...
413 if iStart > 0:
414 sHtml += '%s&nbsp; ... &nbsp;\n' % (sHrefFmt % (0, 'first %s' % (sItemName,), 0 + iBase),);
415
416 sHtml += '&nbsp;\n'.join(sHrefFmt % (i, '%s %d' % (sItemName, i + iBase), i + iBase) if i != iCurItem
417 else unicode(i + iBase)
418 for i in range(iStart, iEnd));
419 if iEnd < cItems:
420 sHtml += '&nbsp; ... &nbsp;%s\n' % (sHrefFmt % (cItems - 1, 'last %s' % (sItemName,), cItems - 1 + iBase));
421
422 # Next page.
423 if iCurItem + 1 < cItems:
424 sHtml += '&nbsp;&nbsp;%s' % sHrefFmt % (iCurItem + 1, 'next ' + sItemName, '&gt;&gt;');
425 else:
426 sHtml += '&nbsp;&nbsp;&gt;&gt;';
427
428 return sHtml;
429
430class WuiSingleContentBase(WuiContentBase): # pylint: disable=too-few-public-methods
431 """
432 Base for the content classes working on a single data object (oData).
433 """
434 def __init__(self, oData, oDisp = None, fnDPrint = None):
435 WuiContentBase.__init__(self, oDisp = oDisp, fnDPrint = fnDPrint);
436 self._oData = oData; # Usually ModelDataBase.
437
438
439class WuiFormContentBase(WuiSingleContentBase): # pylint: disable=too-few-public-methods
440 """
441 Base class for simple input form content classes (single data object).
442 """
443
444 ## @name Form mode.
445 ## @{
446 ksMode_Add = 'add';
447 ksMode_Edit = 'edit';
448 ksMode_Show = 'show';
449 ## @}
450
451 ## Default action mappings.
452 kdSubmitActionMappings = {
453 ksMode_Add: 'AddPost',
454 ksMode_Edit: 'EditPost',
455 };
456
457 def __init__(self, oData, sMode, sCoreName, oDisp, sTitle, sId = None, fEditable = True, sSubmitAction = None):
458 WuiSingleContentBase.__init__(self, copy.copy(oData), oDisp);
459 assert sMode in [self.ksMode_Add, self.ksMode_Edit, self.ksMode_Show];
460 assert len(sTitle) > 1;
461 assert sId is None or sId;
462
463 self._sMode = sMode;
464 self._sCoreName = sCoreName;
465 self._sActionBase = 'ksAction' + sCoreName;
466 self._sTitle = sTitle;
467 self._sId = sId if sId is not None else (type(oData).__name__.lower() + 'form');
468 self._fEditable = fEditable and (oDisp is None or not oDisp.isReadOnlyUser())
469 self._sSubmitAction = sSubmitAction;
470 if sSubmitAction is None and sMode != self.ksMode_Show:
471 self._sSubmitAction = getattr(oDisp, self._sActionBase + self.kdSubmitActionMappings[sMode]);
472 self._sRedirectTo = None;
473
474
475 def _populateForm(self, oForm, oData):
476 """
477 Populates the form. oData has parameter NULL values.
478 This must be reimplemented by the child.
479 """
480 _ = oForm; _ = oData;
481 raise Exception('Reimplement me!');
482
483 def _generatePostFormContent(self, oData):
484 """
485 Generate optional content that comes below the form.
486 Returns a list of tuples, where the first tuple element is the title
487 and the second the content. I.e. similar to show() output.
488 """
489 _ = oData;
490 return [];
491
492 @staticmethod
493 def _calcChangeLogEntryLinks(aoEntries, iEntry):
494 """
495 Returns an array of links to go with the change log entry.
496 """
497 _ = aoEntries; _ = iEntry;
498 ## @todo detect deletion and recreation.
499 ## @todo view details link.
500 ## @todo restore link (need new action)
501 ## @todo clone link.
502 return [];
503
504 @staticmethod
505 def _guessChangeLogEntryDescription(aoEntries, iEntry):
506 """
507 Guesses the action + author that caused the change log entry.
508 Returns descriptive string.
509 """
510 oEntry = aoEntries[iEntry];
511
512 # Figure the author of the change.
513 if oEntry.sAuthor is not None:
514 sAuthor = '%s (#%s)' % (oEntry.sAuthor, oEntry.uidAuthor,);
515 elif oEntry.uidAuthor is not None:
516 sAuthor = '#%d (??)' % (oEntry.uidAuthor,);
517 else:
518 sAuthor = None;
519
520 # Figure the action.
521 if oEntry.oOldRaw is None:
522 if sAuthor is None:
523 return 'Created by batch job.';
524 return 'Created by %s.' % (sAuthor,);
525
526 if sAuthor is None:
527 return 'Automatically updated.'
528 return 'Modified by %s.' % (sAuthor,);
529
530 @staticmethod
531 def formatChangeLogEntry(aoEntries, iEntry, sUrl, dParams):
532 """
533 Formats one change log entry into one or more HTML table rows.
534
535 The sUrl and dParams arguments are used to construct links to historical
536 data using the tsEffective value. If no links wanted, they'll both be None.
537
538 Note! The parameters are given as array + index in case someone wishes
539 to access adjacent entries later in order to generate better
540 change descriptions.
541 """
542 oEntry = aoEntries[iEntry];
543
544 # Turn the effective date into a URL if we can:
545 if sUrl:
546 dParams[WuiDispatcherBase.ksParamEffectiveDate] = oEntry.tsEffective;
547 sEffective = WuiLinkBase(WuiFormContentBase.formatTsShort(oEntry.tsEffective), sUrl,
548 dParams, fBracketed = False).toHtml();
549 else:
550 sEffective = webutils.escapeElem(WuiFormContentBase.formatTsShort(oEntry.tsEffective))
551
552 # The primary row.
553 sRowClass = 'tmodd' if (iEntry + 1) & 1 else 'tmeven';
554 sContent = ' <tr class="%s">\n' \
555 ' <td rowspan="%d">%s</td>\n' \
556 ' <td rowspan="%d">%s</td>\n' \
557 ' <td colspan="3">%s%s</td>\n' \
558 ' </tr>\n' \
559 % ( sRowClass,
560 len(oEntry.aoChanges) + 1, sEffective,
561 len(oEntry.aoChanges) + 1, webutils.escapeElem(WuiFormContentBase.formatTsShort(oEntry.tsExpire)),
562 WuiFormContentBase._guessChangeLogEntryDescription(aoEntries, iEntry),
563 ' '.join(oLink.toHtml() for oLink in WuiFormContentBase._calcChangeLogEntryLinks(aoEntries, iEntry)),);
564
565 # Additional rows for each changed attribute.
566 j = 0;
567 for oChange in oEntry.aoChanges:
568 if isinstance(oChange, AttributeChangeEntryPre):
569 sContent += ' <tr class="%s%s"><td>%s</td>'\
570 '<td><div class="tdpre">%s%s%s</div></td>' \
571 '<td><div class="tdpre">%s%s%s</div></td></tr>\n' \
572 % ( sRowClass, 'odd' if j & 1 else 'even',
573 webutils.escapeElem(oChange.sAttr),
574 '<pre>' if oChange.sOldText else '',
575 webutils.escapeElem(oChange.sOldText),
576 '</pre>' if oChange.sOldText else '',
577 '<pre>' if oChange.sNewText else '',
578 webutils.escapeElem(oChange.sNewText),
579 '</pre>' if oChange.sNewText else '', );
580 else:
581 sContent += ' <tr class="%s%s"><td>%s</td><td>%s</td><td>%s</td></tr>\n' \
582 % ( sRowClass, 'odd' if j & 1 else 'even',
583 webutils.escapeElem(oChange.sAttr),
584 webutils.escapeElem(oChange.sOldText),
585 webutils.escapeElem(oChange.sNewText), );
586 j += 1;
587
588 return sContent;
589
590 def _showChangeLogNavi(self, fMoreEntries, iPageNo, cEntriesPerPage, tsNow, sWhere):
591 """
592 Returns the HTML for the change log navigator.
593 Note! See also _generateNavigation.
594 """
595 sNavigation = '<div class="tmlistnav-%s">\n' % sWhere;
596 sNavigation += ' <table class="tmlistnavtab">\n' \
597 ' <tr>\n';
598 dParams = self._oDisp.getParameters();
599 dParams[WuiDispatcherBase.ksParamChangeLogEntriesPerPage] = cEntriesPerPage;
600 dParams[WuiDispatcherBase.ksParamChangeLogPageNo] = iPageNo;
601 if tsNow is not None:
602 dParams[WuiDispatcherBase.ksParamEffectiveDate] = tsNow;
603
604 # Prev and combo box in one cell. Both inside the form for formatting reasons.
605 sNavigation += ' <td>\n' \
606 ' <form name="ChangeLogEntriesPerPageForm" method="GET">\n'
607
608 # Prev
609 if iPageNo > 0:
610 dParams[WuiDispatcherBase.ksParamChangeLogPageNo] = iPageNo - 1;
611 sNavigation += '<a href="?%s#tmchangelog">Previous</a>\n' \
612 % (webutils.encodeUrlParams(dParams),);
613 dParams[WuiDispatcherBase.ksParamChangeLogPageNo] = iPageNo;
614 else:
615 sNavigation += 'Previous\n';
616
617 # Entries per page selector.
618 del dParams[WuiDispatcherBase.ksParamChangeLogEntriesPerPage];
619 sNavigation += '&nbsp; &nbsp;\n' \
620 ' <select name="%s" onchange="window.location=\'?%s&%s=\' + ' \
621 'this.options[this.selectedIndex].value + \'#tmchangelog\';" ' \
622 'title="Max change log entries per page">\n' \
623 % (WuiDispatcherBase.ksParamChangeLogEntriesPerPage,
624 webutils.encodeUrlParams(dParams),
625 WuiDispatcherBase.ksParamChangeLogEntriesPerPage);
626 dParams[WuiDispatcherBase.ksParamChangeLogEntriesPerPage] = cEntriesPerPage;
627
628 for iEntriesPerPage in [2, 4, 8, 16, 32, 64, 128, 256, 384, 512, 768, 1024, 1536, 2048, 3072, 4096, 8192]:
629 sNavigation += ' <option value="%d" %s>%d entries per page</option>\n' \
630 % ( iEntriesPerPage,
631 'selected="selected"' if iEntriesPerPage == cEntriesPerPage else '',
632 iEntriesPerPage );
633 sNavigation += ' </select>\n';
634
635 # End of cell (and form).
636 sNavigation += ' </form>\n' \
637 ' </td>\n';
638
639 # Next
640 if fMoreEntries:
641 dParams[WuiDispatcherBase.ksParamChangeLogPageNo] = iPageNo + 1;
642 sNavigation += ' <td><a href="?%s#tmchangelog">Next</a></td>\n' \
643 % (webutils.encodeUrlParams(dParams),);
644 else:
645 sNavigation += ' <td>Next</td>\n';
646
647 sNavigation += ' </tr>\n' \
648 ' </table>\n' \
649 '</div>\n';
650 return sNavigation;
651
652 def setRedirectTo(self, sRedirectTo):
653 """
654 For setting the hidden redirect-to field.
655 """
656 self._sRedirectTo = sRedirectTo;
657 return True;
658
659 def showChangeLog(self, aoEntries, fMoreEntries, iPageNo, cEntriesPerPage, tsNow, fShowNavigation = True):
660 """
661 Render the change log, returning raw HTML.
662 aoEntries is an array of ChangeLogEntry.
663 """
664 sContent = '\n' \
665 '<hr>\n' \
666 '<div id="tmchangelog">\n' \
667 ' <h3>Change Log </h3>\n';
668 if fShowNavigation:
669 sContent += self._showChangeLogNavi(fMoreEntries, iPageNo, cEntriesPerPage, tsNow, 'top');
670 sContent += ' <table class="tmtable tmchangelog">\n' \
671 ' <thead class="tmheader">' \
672 ' <tr>' \
673 ' <th rowspan="2">When</th>\n' \
674 ' <th rowspan="2">Expire (excl)</th>\n' \
675 ' <th colspan="3">Changes</th>\n' \
676 ' </tr>\n' \
677 ' <tr>\n' \
678 ' <th>Attribute</th>\n' \
679 ' <th>Old value</th>\n' \
680 ' <th>New value</th>\n' \
681 ' </tr>\n' \
682 ' </thead>\n' \
683 ' <tbody>\n';
684
685 if self._sMode == self.ksMode_Show:
686 sUrl = self._oDisp.getUrlNoParams();
687 dParams = self._oDisp.getParameters();
688 else:
689 sUrl = None;
690 dParams = None;
691
692 for iEntry, _ in enumerate(aoEntries):
693 sContent += self.formatChangeLogEntry(aoEntries, iEntry, sUrl, dParams);
694
695 sContent += ' </tbody>\n' \
696 ' </table>\n';
697 if fShowNavigation and len(aoEntries) >= 8:
698 sContent += self._showChangeLogNavi(fMoreEntries, iPageNo, cEntriesPerPage, tsNow, 'bottom');
699 sContent += '</div>\n\n';
700 return sContent;
701
702 def _generateTopRowFormActions(self, oData):
703 """
704 Returns a list of WuiTmLinks.
705 """
706 aoActions = [];
707 if self._sMode == self.ksMode_Show and self._fEditable:
708 # Remove _idGen and effective date since we're always editing the current data,
709 # and make sure the primary ID is present. Also remove change log stuff.
710 dParams = self._oDisp.getParameters();
711 if hasattr(oData, 'ksIdGenAttr'):
712 sIdGenParam = getattr(oData, 'ksParam_' + oData.ksIdGenAttr);
713 if sIdGenParam in dParams:
714 del dParams[sIdGenParam];
715 for sParam in [ WuiDispatcherBase.ksParamEffectiveDate, ] + list(WuiDispatcherBase.kasChangeLogParams):
716 if sParam in dParams:
717 del dParams[sParam];
718 dParams[getattr(oData, 'ksParam_' + oData.ksIdAttr)] = getattr(oData, oData.ksIdAttr);
719
720 dParams[WuiDispatcherBase.ksParamAction] = getattr(self._oDisp, self._sActionBase + 'Edit');
721 aoActions.append(WuiTmLink('Edit', '', dParams));
722
723 # Add clone operation if available. This uses the same data selection as for showing details. No change log.
724 if hasattr(self._oDisp, self._sActionBase + 'Clone'):
725 dParams = self._oDisp.getParameters();
726 for sParam in WuiDispatcherBase.kasChangeLogParams:
727 if sParam in dParams:
728 del dParams[sParam];
729 dParams[WuiDispatcherBase.ksParamAction] = getattr(self._oDisp, self._sActionBase + 'Clone');
730 aoActions.append(WuiTmLink('Clone', '', dParams));
731
732 elif self._sMode == self.ksMode_Edit:
733 # Details views the details at a given time, so we need either idGen or an effecive date + regular id.
734 dParams = {};
735 if hasattr(oData, 'ksIdGenAttr'):
736 sIdGenParam = getattr(oData, 'ksParam_' + oData.ksIdGenAttr);
737 dParams[sIdGenParam] = getattr(oData, oData.ksIdGenAttr);
738 elif hasattr(oData, 'tsEffective'):
739 dParams[WuiDispatcherBase.ksParamEffectiveDate] = oData.tsEffective;
740 dParams[getattr(oData, 'ksParam_' + oData.ksIdAttr)] = getattr(oData, oData.ksIdAttr);
741 dParams[WuiDispatcherBase.ksParamAction] = getattr(self._oDisp, self._sActionBase + 'Details');
742 aoActions.append(WuiTmLink('Details', '', dParams));
743
744 # Add delete operation if available.
745 if hasattr(self._oDisp, self._sActionBase + 'DoRemove'):
746 dParams = self._oDisp.getParameters();
747 dParams[WuiDispatcherBase.ksParamAction] = getattr(self._oDisp, self._sActionBase + 'DoRemove');
748 dParams[getattr(oData, 'ksParam_' + oData.ksIdAttr)] = getattr(oData, oData.ksIdAttr);
749 aoActions.append(WuiTmLink('Delete', '', dParams, sConfirm = "Are you absolutely sure?"));
750
751 return aoActions;
752
753 def showForm(self, dErrors = None, sErrorMsg = None):
754 """
755 Render the form.
756 """
757 oForm = WuiHlpForm(self._sId,
758 '?' + webutils.encodeUrlParams({WuiDispatcherBase.ksParamAction: self._sSubmitAction}),
759 dErrors if dErrors is not None else {},
760 fReadOnly = self._sMode == self.ksMode_Show);
761
762 self._oData.convertToParamNull();
763
764 # If form cannot be constructed due to some reason we
765 # need to show this reason
766 try:
767 self._populateForm(oForm, self._oData);
768 if self._sRedirectTo is not None:
769 oForm.addTextHidden(self._oDisp.ksParamRedirectTo, self._sRedirectTo);
770 except WuiException as oXcpt:
771 sContent = unicode(oXcpt)
772 else:
773 sContent = oForm.finalize();
774
775 # Add any post form content.
776 atPostFormContent = self._generatePostFormContent(self._oData);
777 if atPostFormContent:
778 for iSection, tSection in enumerate(atPostFormContent):
779 (sSectionTitle, sSectionContent) = tSection;
780 sContent += u'<div id="postform-%d" class="tmformpostsection">\n' % (iSection,);
781 if sSectionTitle:
782 sContent += '<h3 class="tmformpostheader">%s</h3>\n' % (webutils.escapeElem(sSectionTitle),);
783 sContent += u' <div id="postform-%d-content" class="tmformpostcontent">\n' % (iSection,);
784 sContent += sSectionContent;
785 sContent += u' </div>\n' \
786 u'</div>\n';
787
788 # Add action to the top.
789 aoActions = self._generateTopRowFormActions(self._oData);
790 if aoActions:
791 sActionLinks = '<p>%s</p>' % (' '.join(unicode(oLink) for oLink in aoActions));
792 sContent = sActionLinks + sContent;
793
794 # Add error info to the top.
795 if sErrorMsg is not None:
796 sContent = '<p class="tmerrormsg">' + webutils.escapeElem(sErrorMsg) + '</p>\n' + sContent;
797
798 return (self._sTitle, sContent);
799
800 def getListOfItems(self, asListItems = tuple(), asSelectedItems = tuple()):
801 """
802 Format generic list which should be used by HTML form
803 """
804 aoRet = []
805 for sListItem in asListItems:
806 fEnabled = sListItem in asSelectedItems;
807 aoRet.append((sListItem, fEnabled, sListItem))
808 return aoRet
809
810
811class WuiListContentBase(WuiContentBase):
812 """
813 Base for the list content classes.
814 """
815
816 def __init__(self, aoEntries, iPage, cItemsPerPage, tsEffectiveDate, sTitle, # pylint: disable=too-many-arguments
817 sId = None, fnDPrint = None, oDisp = None, aiSelectedSortColumns = None, fTimeNavigation = True):
818 WuiContentBase.__init__(self, fnDPrint = fnDPrint, oDisp = oDisp);
819 self._aoEntries = aoEntries; ## @todo should replace this with a Logic object and define methods for querying.
820 self._iPage = iPage;
821 self._cItemsPerPage = cItemsPerPage;
822 self._tsEffectiveDate = tsEffectiveDate;
823 self._fTimeNavigation = fTimeNavigation;
824 self._sTitle = sTitle; assert len(sTitle) > 1;
825 if sId is None:
826 sId = sTitle.strip().replace(' ', '').lower();
827 assert sId.strip();
828 self._sId = sId;
829 self._asColumnHeaders = [];
830 self._asColumnAttribs = [];
831 self._aaiColumnSorting = []; ##< list of list of integers
832 self._aiSelectedSortColumns = aiSelectedSortColumns; ##< list of integers
833
834 def _formatCommentCell(self, sComment, cMaxLines = 3, cchMaxLine = 63):
835 """
836 Helper functions for formatting comment cell.
837 Returns None or WuiRawHtml instance.
838 """
839 # Nothing to do for empty comments.
840 if sComment is None:
841 return None;
842 sComment = sComment.strip();
843 if not sComment:
844 return None;
845
846 # Restrict the text if necessary, making the whole text available thru mouse-over.
847 ## @todo this would be better done by java script or smth, so it could automatically adjust to the table size.
848 if len(sComment) > cchMaxLine or sComment.count('\n') >= cMaxLines:
849 sShortHtml = '';
850 for iLine, sLine in enumerate(sComment.split('\n')):
851 if iLine >= cMaxLines:
852 break;
853 if iLine > 0:
854 sShortHtml += '<br>\n';
855 if len(sLine) > cchMaxLine:
856 sShortHtml += webutils.escapeElem(sLine[:(cchMaxLine - 3)]);
857 sShortHtml += '...';
858 else:
859 sShortHtml += webutils.escapeElem(sLine);
860 return WuiRawHtml('<span class="tmcomment" title="%s">%s</span>' % (webutils.escapeAttr(sComment), sShortHtml,));
861
862 return WuiRawHtml('<span class="tmcomment">%s</span>' % (webutils.escapeElem(sComment).replace('\n', '<br>'),));
863
864 def _formatListEntry(self, iEntry):
865 """
866 Formats the specified list entry as a list of column values.
867 Returns HTML for a table row.
868
869 The child class really need to override this!
870 """
871 # ASSUMES ModelDataBase children.
872 asRet = [];
873 for sAttr in self._aoEntries[0].getDataAttributes():
874 asRet.append(getattr(self._aoEntries[iEntry], sAttr));
875 return asRet;
876
877 def _formatListEntryHtml(self, iEntry):
878 """
879 Formats the specified list entry as HTML.
880 Returns HTML for a table row.
881
882 The child class can override this to
883 """
884 if (iEntry + 1) & 1:
885 sRow = u' <tr class="tmodd">\n';
886 else:
887 sRow = u' <tr class="tmeven">\n';
888
889 aoValues = self._formatListEntry(iEntry);
890 assert len(aoValues) == len(self._asColumnHeaders), '%s vs %s' % (len(aoValues), len(self._asColumnHeaders));
891
892 for i, oValue in enumerate(aoValues):
893 if i < len(self._asColumnAttribs) and self._asColumnAttribs[i]:
894 sRow += u' <td ' + self._asColumnAttribs[i] + '>';
895 else:
896 sRow += u' <td>';
897
898 if isinstance(oValue, WuiHtmlBase):
899 sRow += oValue.toHtml();
900 elif isinstance(oValue, list):
901 if oValue:
902 for oElement in oValue:
903 if isinstance(oElement, WuiHtmlBase):
904 sRow += oElement.toHtml();
905 elif db.isDbTimestamp(oElement):
906 sRow += webutils.escapeElem(self.formatTsShort(oElement));
907 else:
908 sRow += webutils.escapeElem(unicode(oElement));
909 sRow += ' ';
910 elif db.isDbTimestamp(oValue):
911 sRow += webutils.escapeElem(self.formatTsShort(oValue));
912 elif db.isDbInterval(oValue):
913 sRow += webutils.escapeElem(self.formatIntervalShort(oValue));
914 elif oValue is not None:
915 sRow += webutils.escapeElem(unicode(oValue));
916
917 sRow += u'</td>\n';
918
919 return sRow + u' </tr>\n';
920
921 @staticmethod
922 def generateTimeNavigationComboBox(sWhere, dParams, tsEffective):
923 """
924 Generates the HTML for the xxxx ago combo box form.
925 """
926 sNavigation = '<form name="TmTimeNavCombo-%s" method="GET">\n' % (sWhere,);
927 sNavigation += ' <select name="%s" onchange="window.location=' % (WuiDispatcherBase.ksParamEffectiveDate);
928 sNavigation += '\'?%s&%s=\' + ' % (webutils.encodeUrlParams(dParams), WuiDispatcherBase.ksParamEffectiveDate)
929 sNavigation += 'this.options[this.selectedIndex].value;" title="Effective date">\n';
930
931 aoWayBackPoints = [
932 ('+0000-00-00 00:00:00.00', 'Now', ' title="Present Day. Present Time."'), # lain :)
933
934 ('-0000-00-00 01:00:00.00', '1 hour ago', ''),
935 ('-0000-00-00 02:00:00.00', '2 hours ago', ''),
936 ('-0000-00-00 03:00:00.00', '3 hours ago', ''),
937
938 ('-0000-00-01 00:00:00.00', '1 day ago', ''),
939 ('-0000-00-02 00:00:00.00', '2 days ago', ''),
940 ('-0000-00-03 00:00:00.00', '3 days ago', ''),
941
942 ('-0000-00-07 00:00:00.00', '1 week ago', ''),
943 ('-0000-00-14 00:00:00.00', '2 weeks ago', ''),
944 ('-0000-00-21 00:00:00.00', '3 weeks ago', ''),
945
946 ('-0000-01-00 00:00:00.00', '1 month ago', ''),
947 ('-0000-02-00 00:00:00.00', '2 months ago', ''),
948 ('-0000-03-00 00:00:00.00', '3 months ago', ''),
949 ('-0000-04-00 00:00:00.00', '4 months ago', ''),
950 ('-0000-05-00 00:00:00.00', '5 months ago', ''),
951 ('-0000-06-00 00:00:00.00', 'Half a year ago', ''),
952
953 ('-0001-00-00 00:00:00.00', '1 year ago', ''),
954 ]
955 fSelected = False;
956 for sTimestamp, sWayBackPointCaption, sExtraAttrs in aoWayBackPoints:
957 if sTimestamp == tsEffective:
958 fSelected = True;
959 sNavigation += ' <option value="%s"%s%s>%s</option>\n' \
960 % (webutils.quoteUrl(sTimestamp),
961 ' selected="selected"' if sTimestamp == tsEffective else '',
962 sExtraAttrs, sWayBackPointCaption, );
963 if not fSelected and tsEffective != '':
964 sNavigation += ' <option value="%s" selected>%s</option>\n' \
965 % (webutils.quoteUrl(tsEffective), WuiContentBase.formatTsShort(tsEffective))
966
967 sNavigation += ' </select>\n' \
968 '</form>\n';
969 return sNavigation;
970
971 @staticmethod
972 def generateTimeNavigationDateTime(sWhere, dParams, sNow):
973 """
974 Generates HTML for a form with date + time input fields.
975
976 Note! Modifies dParams!
977 """
978
979 #
980 # Date + time input fields. We use a java script helper to combine the two
981 # into a hidden field as there is no portable datetime input field type.
982 #
983 sNavigation = '<form method="get" action="?" onchange="timeNavigationUpdateHiddenEffDate(this,\'%s\')">' % (sWhere,);
984 if sNow is None:
985 sNow = utils.getIsoTimestamp();
986 else:
987 sNow = utils.normalizeIsoTimestampToZulu(sNow);
988 asSplit = sNow.split('T');
989 sNavigation += ' <input type="date" value="%s" id="EffDate%s"/> ' % (asSplit[0], sWhere, );
990 sNavigation += ' <input type="time" value="%s" id="EffTime%s"/> ' % (asSplit[1][:8], sWhere,);
991 sNavigation += ' <input type="hidden" name="%s" value="%s" id="EffDateTime%s"/>' \
992 % (WuiDispatcherBase.ksParamEffectiveDate, webutils.escapeAttr(sNow), sWhere);
993 for sKey in dParams:
994 sNavigation += ' <input type="hidden" name="%s" value="%s"/>' \
995 % (webutils.escapeAttr(sKey), webutils.escapeAttrToStr(dParams[sKey]));
996 sNavigation += ' <input type="submit" value="Set"/>\n' \
997 '</form>\n';
998 return sNavigation;
999
1000 ## @todo move to better place! WuiMain uses it.
1001 @staticmethod
1002 def generateTimeNavigation(sWhere, dParams, tsEffectiveAbs, sPreamble = '', sPostamble = '', fKeepPageNo = False):
1003 """
1004 Returns HTML for time navigation.
1005
1006 Note! Modifies dParams!
1007 Note! Views without a need for a timescale just stubs this method.
1008 """
1009 sNavigation = '<div class="tmtimenav-%s tmtimenav">%s' % (sWhere, sPreamble,);
1010
1011 #
1012 # Prepare the URL parameters.
1013 #
1014 if WuiDispatcherBase.ksParamPageNo in dParams: # Forget about page No when changing a period
1015 del dParams[WuiDispatcherBase.ksParamPageNo]
1016 if not fKeepPageNo and WuiDispatcherBase.ksParamEffectiveDate in dParams:
1017 tsEffectiveParam = dParams[WuiDispatcherBase.ksParamEffectiveDate];
1018 del dParams[WuiDispatcherBase.ksParamEffectiveDate];
1019 else:
1020 tsEffectiveParam = ''
1021
1022 #
1023 # Generate the individual parts.
1024 #
1025 sNavigation += WuiListContentBase.generateTimeNavigationDateTime(sWhere, dParams, tsEffectiveAbs);
1026 sNavigation += WuiListContentBase.generateTimeNavigationComboBox(sWhere, dParams, tsEffectiveParam);
1027
1028 sNavigation += '%s</div>' % (sPostamble,);
1029 return sNavigation;
1030
1031 def _generateTimeNavigation(self, sWhere, sPreamble = '', sPostamble = ''):
1032 """
1033 Returns HTML for time navigation.
1034
1035 Note! Views without a need for a timescale just stubs this method.
1036 """
1037 return self.generateTimeNavigation(sWhere, self._oDisp.getParameters(), self._oDisp.getEffectiveDateParam(),
1038 sPreamble, sPostamble)
1039
1040 @staticmethod
1041 def generateItemPerPageSelector(sWhere, dParams, cCurItemsPerPage):
1042 """
1043 Generate HTML code for items per page selector.
1044 Note! Modifies dParams!
1045 """
1046
1047 # Drop the current page count parameter.
1048 if WuiDispatcherBase.ksParamItemsPerPage in dParams:
1049 del dParams[WuiDispatcherBase.ksParamItemsPerPage];
1050
1051 # Remove the current page number.
1052 if WuiDispatcherBase.ksParamPageNo in dParams:
1053 del dParams[WuiDispatcherBase.ksParamPageNo];
1054
1055 sHtmlItemsPerPageSelector = '<form name="TmItemsPerPageForm-%s" method="GET" class="tmitemsperpage-%s tmitemsperpage">\n'\
1056 ' <select name="%s" onchange="window.location=\'?%s&%s=\' + ' \
1057 'this.options[this.selectedIndex].value;" title="Max items per page">\n' \
1058 % (sWhere, WuiDispatcherBase.ksParamItemsPerPage, sWhere,
1059 webutils.encodeUrlParams(dParams),
1060 WuiDispatcherBase.ksParamItemsPerPage)
1061
1062 acItemsPerPage = [16, 32, 64, 128, 256, 384, 512, 768, 1024, 1536, 2048, 3072, 4096];
1063 for cItemsPerPage in acItemsPerPage:
1064 sHtmlItemsPerPageSelector += ' <option value="%d" %s>%d per page</option>\n' \
1065 % (cItemsPerPage,
1066 'selected="selected"' if cItemsPerPage == cCurItemsPerPage else '',
1067 cItemsPerPage)
1068 sHtmlItemsPerPageSelector += ' </select>\n' \
1069 '</form>\n';
1070
1071 return sHtmlItemsPerPageSelector
1072
1073
1074 def _generateNavigation(self, sWhere):
1075 """
1076 Return HTML for navigation.
1077 """
1078
1079 #
1080 # ASSUMES the dispatcher/controller code fetches one entry more than
1081 # needed to fill the page to indicate further records.
1082 #
1083 sNavigation = '<div class="tmlistnav-%s">\n' % sWhere;
1084 sNavigation += ' <table class="tmlistnavtab">\n' \
1085 ' <tr>\n';
1086 dParams = self._oDisp.getParameters();
1087 dParams[WuiDispatcherBase.ksParamItemsPerPage] = self._cItemsPerPage;
1088 dParams[WuiDispatcherBase.ksParamPageNo] = self._iPage;
1089 if self._tsEffectiveDate is not None:
1090 dParams[WuiDispatcherBase.ksParamEffectiveDate] = self._tsEffectiveDate;
1091
1092 # Prev
1093 if self._iPage > 0:
1094 dParams[WuiDispatcherBase.ksParamPageNo] = self._iPage - 1;
1095 sNavigation += ' <td align="left"><a href="?%s">Previous</a></td>\n' % (webutils.encodeUrlParams(dParams),);
1096 else:
1097 sNavigation += ' <td></td>\n';
1098
1099 # Time scale.
1100 if self._fTimeNavigation:
1101 sNavigation += '<td align="center" class="tmtimenav">';
1102 sNavigation += self._generateTimeNavigation(sWhere);
1103 sNavigation += '</td>';
1104
1105 # page count and next.
1106 sNavigation += '<td align="right" class="tmnextanditemsperpage">\n';
1107
1108 if len(self._aoEntries) > self._cItemsPerPage:
1109 dParams[WuiDispatcherBase.ksParamPageNo] = self._iPage + 1;
1110 sNavigation += ' <a href="?%s">Next</a>\n' % (webutils.encodeUrlParams(dParams),);
1111 sNavigation += self.generateItemPerPageSelector(sWhere, dParams, self._cItemsPerPage);
1112 sNavigation += '</td>\n';
1113 sNavigation += ' </tr>\n' \
1114 ' </table>\n' \
1115 '</div>\n';
1116 return sNavigation;
1117
1118 def _checkSortingByColumnAscending(self, aiColumns):
1119 """
1120 Checks if we're sorting by this column.
1121
1122 Returns 0 if not sorting by this, negative if descending, positive if ascending. The
1123 value indicates the priority (nearer to 0 is higher).
1124 """
1125 if len(aiColumns) <= len(self._aiSelectedSortColumns):
1126 aiColumns = list(aiColumns);
1127 aiNegColumns = list([-i for i in aiColumns]); # pylint: disable=consider-using-generator
1128 i = 0;
1129 while i + len(aiColumns) <= len(self._aiSelectedSortColumns):
1130 aiSub = list(self._aiSelectedSortColumns[i : i + len(aiColumns)]);
1131 if aiSub == aiColumns:
1132 return 1 + i;
1133 if aiSub == aiNegColumns:
1134 return -1 - i;
1135 i += 1;
1136 return 0;
1137
1138 def _generateTableHeaders(self):
1139 """
1140 Generate table headers.
1141 Returns raw html string.
1142 Overridable.
1143 """
1144
1145 sHtml = ' <thead class="tmheader"><tr>';
1146 for iHeader, oHeader in enumerate(self._asColumnHeaders):
1147 if isinstance(oHeader, WuiHtmlBase):
1148 sHtml += '<th>' + oHeader.toHtml() + '</th>';
1149 elif iHeader < len(self._aaiColumnSorting) and self._aaiColumnSorting[iHeader] is not None:
1150 sHtml += '<th>'
1151 iSorting = self._checkSortingByColumnAscending(self._aaiColumnSorting[iHeader]);
1152 if iSorting > 0:
1153 sDirection = '&nbsp;&#x25b4;' if iSorting == 1 else '<small>&nbsp;&#x25b5;</small>';
1154 sSortParams = ','.join([str(-i) for i in self._aaiColumnSorting[iHeader]]);
1155 else:
1156 sDirection = '';
1157 if iSorting < 0:
1158 sDirection = '&nbsp;&#x25be;' if iSorting == -1 else '<small>&nbsp;&#x25bf;</small>'
1159 sSortParams = ','.join([str(i) for i in self._aaiColumnSorting[iHeader]]);
1160 sHtml += '<a href="javascript:ahrefActionSortByColumns(\'%s\',[%s]);">' \
1161 % (WuiDispatcherBase.ksParamSortColumns, sSortParams);
1162 sHtml += webutils.escapeElem(oHeader) + '</a>' + sDirection + '</th>';
1163 else:
1164 sHtml += '<th>' + webutils.escapeElem(oHeader) + '</th>';
1165 sHtml += '</tr><thead>\n';
1166 return sHtml
1167
1168 def _generateTable(self):
1169 """
1170 show worker that just generates the table.
1171 """
1172
1173 #
1174 # Create a table.
1175 # If no colum headers are provided, fall back on database field
1176 # names, ASSUMING that the entries are ModelDataBase children.
1177 # Note! the cellspacing is for IE8.
1178 #
1179 sPageBody = '<table class="tmtable" id="' + self._sId + '" cellspacing="0">\n';
1180
1181 if not self._asColumnHeaders:
1182 self._asColumnHeaders = self._aoEntries[0].getDataAttributes();
1183
1184 sPageBody += self._generateTableHeaders();
1185
1186 #
1187 # Format the body and close the table.
1188 #
1189 sPageBody += ' <tbody>\n';
1190 for iEntry in range(min(len(self._aoEntries), self._cItemsPerPage)):
1191 sPageBody += self._formatListEntryHtml(iEntry);
1192 sPageBody += ' </tbody>\n' \
1193 '</table>\n';
1194 return sPageBody;
1195
1196 def _composeTitle(self):
1197 """Composes the title string (return value)."""
1198 sTitle = self._sTitle;
1199 if self._iPage != 0:
1200 sTitle += ' (page ' + unicode(self._iPage + 1) + ')'
1201 if self._tsEffectiveDate is not None:
1202 sTitle += ' as per ' + unicode(self.formatTsShort(self._tsEffectiveDate));
1203 return sTitle;
1204
1205
1206 def show(self, fShowNavigation = True):
1207 """
1208 Displays the list.
1209 Returns (Title, HTML) on success, raises exception on error.
1210 """
1211
1212 sPageBody = ''
1213 if fShowNavigation:
1214 sPageBody += self._generateNavigation('top');
1215
1216 if self._aoEntries:
1217 sPageBody += self._generateTable();
1218 if fShowNavigation:
1219 sPageBody += self._generateNavigation('bottom');
1220 else:
1221 sPageBody += '<p>No entries.</p>'
1222
1223 return (self._composeTitle(), sPageBody);
1224
1225
1226class WuiListContentWithActionBase(WuiListContentBase):
1227 """
1228 Base for the list content with action classes.
1229 """
1230
1231 def __init__(self, aoEntries, iPage, cItemsPerPage, tsEffectiveDate, sTitle, # pylint: disable=too-many-arguments
1232 sId = None, fnDPrint = None, oDisp = None, aiSelectedSortColumns = None):
1233 WuiListContentBase.__init__(self, aoEntries, iPage, cItemsPerPage, tsEffectiveDate, sTitle, sId = sId,
1234 fnDPrint = fnDPrint, oDisp = oDisp, aiSelectedSortColumns = aiSelectedSortColumns);
1235 self._aoActions = None; # List of [ oValue, sText, sHover ] provided by the child class.
1236 self._sAction = None; # Set by the child class.
1237 self._sCheckboxName = None; # Set by the child class.
1238 self._asColumnHeaders = [ WuiRawHtml('<input type="checkbox" onClick="toggle%s(this)">'
1239 % ('' if sId is None else sId)), ];
1240 self._asColumnAttribs = [ 'align="center"', ];
1241 self._aaiColumnSorting = [ None, ];
1242
1243 def _getCheckBoxColumn(self, iEntry, sValue):
1244 """
1245 Used by _formatListEntry implementations, returns a WuiRawHtmlBase object.
1246 """
1247 _ = iEntry;
1248 return WuiRawHtml('<input type="checkbox" name="%s" value="%s">'
1249 % (webutils.escapeAttr(self._sCheckboxName), webutils.escapeAttr(unicode(sValue))));
1250
1251 def show(self, fShowNavigation=True):
1252 """
1253 Displays the list.
1254 Returns (Title, HTML) on success, raises exception on error.
1255 """
1256 assert self._aoActions is not None;
1257 assert self._sAction is not None;
1258
1259 sPageBody = '<script language="JavaScript">\n' \
1260 'function toggle%s(oSource) {\n' \
1261 ' aoCheckboxes = document.getElementsByName(\'%s\');\n' \
1262 ' for(var i in aoCheckboxes)\n' \
1263 ' aoCheckboxes[i].checked = oSource.checked;\n' \
1264 '}\n' \
1265 '</script>\n' \
1266 % ('' if self._sId is None else self._sId, self._sCheckboxName,);
1267 if fShowNavigation:
1268 sPageBody += self._generateNavigation('top');
1269 if self._aoEntries:
1270
1271 sPageBody += '<form action="?%s" method="post" class="tmlistactionform">\n' \
1272 % (webutils.encodeUrlParams({WuiDispatcherBase.ksParamAction: self._sAction,}),);
1273 sPageBody += self._generateTable();
1274
1275 sPageBody += ' <label>Actions</label>\n' \
1276 ' <select name="%s" id="%s-action-combo" class="tmlistactionform-combo">\n' \
1277 % (webutils.escapeAttr(WuiDispatcherBase.ksParamListAction), webutils.escapeAttr(self._sId),);
1278 for oValue, sText, _ in self._aoActions:
1279 sPageBody += ' <option value="%s">%s</option>\n' \
1280 % (webutils.escapeAttr(unicode(oValue)), webutils.escapeElem(sText), );
1281 sPageBody += ' </select>\n';
1282 sPageBody += ' <input type="submit"></input>\n';
1283 sPageBody += '</form>\n';
1284 if fShowNavigation:
1285 sPageBody += self._generateNavigation('bottom');
1286 else:
1287 sPageBody += '<p>No entries.</p>'
1288
1289 return (self._composeTitle(), sPageBody);
1290
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette