267 lines
		
	
	
	
		
			9.6 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
			
		
		
	
	
			267 lines
		
	
	
	
		
			9.6 KiB
		
	
	
	
		
			Python
		
	
	
		
			Executable file
		
	
	
	
	
| #
 | |
| #  Copyright (c) 2012 Helmut Merz helmutm@cy55.de
 | |
| #
 | |
| #  This program is free software; you can redistribute it and/or modify
 | |
| #  it under the terms of the GNU General Public License as published by
 | |
| #  the Free Software Foundation; either version 2 of the License, or
 | |
| #  (at your option) any later version.
 | |
| #
 | |
| #  This program is distributed in the hope that it will be useful,
 | |
| #  but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| #  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 | |
| #  GNU General Public License for more details.
 | |
| #
 | |
| #  You should have received a copy of the GNU General Public License
 | |
| #  along with this program; if not, write to the Free Software
 | |
| #  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
 | |
| #
 | |
| 
 | |
| """
 | |
| Report result sets and related classes.
 | |
| """
 | |
| 
 | |
| from copy import copy
 | |
| from zope.cachedescriptors.property import Lazy
 | |
| 
 | |
| from cybertools.composer.interfaces import IInstance
 | |
| from cybertools.composer.report.base import BaseQueryCriteria
 | |
| from cybertools.util.jeep import Jeep
 | |
| 
 | |
| 
 | |
| class BaseRow(object):
 | |
| 
 | |
|     def __init__(self, context, parent):
 | |
|         self.context = context
 | |
|         self.parent = parent
 | |
|         self.data = {}
 | |
|         self.sequenceNumber = 0
 | |
|         self.cssClass = 'row'
 | |
| 
 | |
|     def getRawValue(self, attr):
 | |
|         return self.data.get(attr)
 | |
| 
 | |
| 
 | |
| class TotalsRow(BaseRow):
 | |
| 
 | |
|     pass
 | |
| 
 | |
| 
 | |
| class Row(BaseRow):
 | |
| 
 | |
|     attributeHandlers = {}
 | |
|     cssClass = u''
 | |
| 
 | |
|     def getRawValue(self, attr):
 | |
|         return self.attributeHandlers.get(
 | |
|                         attr, self.getContextAttr)(self, attr)
 | |
| 
 | |
|     @staticmethod
 | |
|     def getContextAttr(obj, attr):
 | |
|         return getattr(obj.context, attr)
 | |
| 
 | |
|     def getGroupFields(self):
 | |
|         return [self.getRawValue(f.name) for f in
 | |
|                     self.parent.context.fields if 'group' in f.executionSteps]
 | |
| 
 | |
|     @Lazy
 | |
|     def displayedColumns(self):
 | |
|         return self.parent.context.getActiveOutputFields()
 | |
| 
 | |
|     @Lazy
 | |
|     def allColumns(self):
 | |
|         return self.parent.context.getAllFields()
 | |
| 
 | |
|     def useRowProperty(self, attr):
 | |
|         return getattr(self, attr)
 | |
| 
 | |
| 
 | |
| class GroupHeaderRow(BaseRow):
 | |
| 
 | |
|     def getRawValue(self, attr):
 | |
|         return self.data.get(attr, u'')
 | |
| 
 | |
|     @Lazy
 | |
|     def displayedColumns(self):
 | |
|         fields = self.parent.context.getActiveOutputFields()
 | |
|         for col in self.headerColumns:
 | |
|             for idx, f in enumerate(fields):
 | |
|                 if f.name == col.name:
 | |
|                     fields[idx] = col
 | |
|         return fields
 | |
| 
 | |
| 
 | |
| class SubTotalsRow(BaseRow):
 | |
| 
 | |
|     def getRawValue(self, attr):
 | |
|         return self.data.get(attr, u'')
 | |
| 
 | |
|     @Lazy
 | |
|     def displayedColumns(self):
 | |
|         fields = self.parent.context.getActiveOutputFields()
 | |
|         if not self.subTotalsGroupColumns:
 | |
|             return fields
 | |
|         for col in self.subTotalsGroupColumns:
 | |
|             for idx, f in enumerate(fields):
 | |
|                 if f.name == col.name:
 | |
|                     fields[idx] = col
 | |
|         return fields
 | |
| 
 | |
| 
 | |
| class ResultSet(object):
 | |
| 
 | |
|     def __init__(self, context, data,
 | |
|                  rowFactory=Row, headerRowFactory=GroupHeaderRow,
 | |
|                  sortCriteria=None, queryCriteria=BaseQueryCriteria(),
 | |
|                  limits=None):
 | |
|         self.context = context  # the report or report instance
 | |
|         self.data = data
 | |
|         self.rowFactory = rowFactory
 | |
|         self.headerRowFactory = headerRowFactory
 | |
|         self.sortCriteria = sortCriteria
 | |
|         self.queryCriteria = queryCriteria
 | |
|         self.limits = limits
 | |
|         self.totals = TotalsRow(None, self)
 | |
| 
 | |
|     def getHeaderRow(self, row, columns):
 | |
|         headerRow = self.headerRowFactory(None, self)
 | |
|         headerRow.cssClass = 'headerRow'
 | |
|         headerRow.headerColumns = []
 | |
|         for c in columns:
 | |
|             if c.output:
 | |
|                 headerRow.data[c.output] = c.getRawValue(row)
 | |
|                 headerColumn = copy(c)
 | |
|                 headerColumn.__name__ = c.output
 | |
|                 headerColumn.cssClass = c.cssClass
 | |
|                 headerRow.headerColumns.append(headerColumn)
 | |
|                 if headerColumn.groupHeaderColspan is not None:
 | |
|                     colNames = [col.name for col in self.displayedColumns]
 | |
|                     nextColumns = [col for col in self.displayedColumns
 | |
|                                    if colNames.index(col.name) > colNames.index(c.output)]
 | |
|                     for i in range(headerColumn.groupHeaderColspan - 1):
 | |
|                         nextColumn = copy(nextColumns[i])
 | |
|                         nextColumn.groupHeaderHidden = True
 | |
|                         headerRow.headerColumns.append(nextColumn)
 | |
|         return headerRow
 | |
| 
 | |
|     def getSubTotalsRow(self, gf, row, columns, values):
 | |
|         if not gf.name in ','.join([','.join(c.totals) for c in columns]).split(','):
 | |
|             return None
 | |
|         subTotalsRow = SubTotalsRow(None, self)
 | |
|         subTotalsRow.subTotalsGroupColumns = []
 | |
|         subTotalsRow.cssClass = 'subTotalsRow'
 | |
|         for idx, c in enumerate(columns):
 | |
|             subTotalsRow.data[c.name] = values[idx]
 | |
|         if gf in self.subTotalsGroupColumns:
 | |
|             if gf.totalsDescription is None:
 | |
|                 display = gf.getDisplayValue(row)
 | |
|                 if isinstance(display, dict):
 | |
|                     display = display.get('title')
 | |
|                 subTotalsRow.data[gf.output] = u'SUMME: ' + display
 | |
|             else:
 | |
|                 v = gf.totalsDescription.getDisplayValue(row)
 | |
|                 if v is None:
 | |
|                     v = u''
 | |
|                 subTotalsRow.data[gf.totalsDescription.output] = v
 | |
|             if gf.groupHeaderColspan is not None:
 | |
|                 colNames = [col.name for col in self.displayedColumns]
 | |
|                 for col in self.displayedColumns:
 | |
|                     sCol = copy(col)
 | |
|                     if colNames.index(col.name) > colNames.index(gf.output):
 | |
|                         if colNames.index(col.name) < colNames.index(gf.output) + gf.groupHeaderColspan:
 | |
|                             sCol.groupHeaderHidden = True
 | |
|                     if col.name == gf.output:
 | |
|                         sCol.groupHeaderColspan = gf.groupHeaderColspan
 | |
|                     subTotalsRow.subTotalsGroupColumns.append(sCol)
 | |
|         return subTotalsRow
 | |
| 
 | |
|     def getResult(self):
 | |
|         result = [self.rowFactory(item, self) for item in self.data]
 | |
|         result = [row for row in result if self.queryCriteria.check(row)]
 | |
|         if self.sortCriteria:
 | |
|             result.sort(key=lambda x:
 | |
|                         [f.getSortValue(x) for f in self.sortCriteria])
 | |
|         if self.groupColumns:
 | |
|             res = []
 | |
|             groupValues = [None for f in self.groupColumns]
 | |
|             subTotals = [[0.0 for f in stc] for stc in self.subTotalsColumns]
 | |
|             lastRow = None
 | |
|             for row in result:
 | |
|                 subTotalsRows = []
 | |
|                 headerRows = []
 | |
|                 for idx, f in enumerate(self.groupColumns):
 | |
|                     value = f.getRawValue(row)
 | |
|                     if value != groupValues[idx]:
 | |
|                         # TODO: loop through all lower-level fields
 | |
|                         # for j, f in enumerate(self.groupColumns)[idx:]:
 | |
|                         #     # use idx+j for correct indexing
 | |
|                         groupValues[idx] = value
 | |
|                         headerRows.append(self.getHeaderRow(row, (f,) + f.outputWith))
 | |
|                         if lastRow is not None and f.getDisplayValue(lastRow):
 | |
|                             subTr = self.getSubTotalsRow(f, lastRow,
 | |
|                                 self.subTotalsColumns[idx], subTotals[idx])
 | |
|                             if subTr is not None:
 | |
|                                 subTotalsRows.append(subTr)
 | |
|                             subTotals[idx] = [0.0 for f in self.subTotalsColumns[idx]]
 | |
|                 for subTotalsRow in reversed(subTotalsRows):
 | |
|                     res.append(subTotalsRow)
 | |
|                 for headerRow in headerRows:
 | |
|                     res.append(headerRow)
 | |
|                 res.append(row)
 | |
|                 for idx, sc in enumerate(self.subTotalsColumns):
 | |
|                     for idx2, f in enumerate(sc):
 | |
|                         subTotals[idx][idx2] += f.getValue(row,
 | |
|                                                            ignoreTotals=True)
 | |
|                 lastRow = row
 | |
|             if lastRow is not None:
 | |
|                 subTotalsRows = []
 | |
|                 for idx, f in enumerate(self.groupColumns):
 | |
|                     if f.getDisplayValue(lastRow):
 | |
|                         subTr = self.getSubTotalsRow(f, lastRow,
 | |
|                             self.subTotalsColumns[idx], subTotals[idx])
 | |
|                         if subTr is not None:
 | |
|                             subTotalsRows.append(subTr)
 | |
|                 for subTotalsRow in reversed(subTotalsRows):
 | |
|                     res.append(subTotalsRow)
 | |
|             result = res
 | |
|         if self.limits:
 | |
|             start, stop = self.limits
 | |
|             result = result[start:stop]
 | |
|         number = 0
 | |
|         for idx, row in enumerate(result):
 | |
|             if not isinstance(row, (GroupHeaderRow, SubTotalsRow)):
 | |
|                 row.sequenceNumber = number + 1
 | |
|                 number += 1
 | |
|         return result
 | |
| 
 | |
|     @Lazy
 | |
|     def result(self):
 | |
|         return self.getResult()
 | |
| 
 | |
|     def __iter__(self):
 | |
|         return iter(self.result)
 | |
| 
 | |
|     def first(self):
 | |
|         if len(self.result) > 0:
 | |
|             return self.result[0]
 | |
|         return self.rowFactory(None, self)
 | |
| 
 | |
|     @Lazy
 | |
|     def displayedColumns(self):
 | |
|         return Jeep(self.context.getActiveOutputFields())
 | |
| 
 | |
|     @Lazy
 | |
|     def groupColumns(self):
 | |
|         return self.context.getGroupFields()
 | |
| 
 | |
|     @Lazy
 | |
|     def subTotalsColumns(self):
 | |
|         return self.context.getSubTotalsFields()
 | |
| 
 | |
|     @Lazy
 | |
|     def subTotalsGroupColumns(self):
 | |
|         return self.context.getSubTotalsGroupFields()
 | |
| 
 | |
| 
 | |
|     def getOutputColumnsForField(self, f):
 | |
|         return self.context.getOutputFieldsForField(f)
 | |
| 
 | 
