pivottable.inc.php 6.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185
  1. <?php
  2. /**
  3. * @version V4.50 6 July 2004 (c) 2000-2005 John Lim (jlim@natsoft.com.my). All rights reserved.
  4. * Released under both BSD license and Lesser GPL library license.
  5. * Whenever there is any discrepancy between the two licenses,
  6. * the BSD license will take precedence.
  7. *
  8. * Set tabs to 4 for best viewing.
  9. *
  10. * Latest version is available at http://php.weblogs.com
  11. *
  12. * Requires PHP4.01pl2 or later because it uses include_once
  13. */
  14. /*
  15. * Concept from daniel.lucazeau@ajornet.com.
  16. *
  17. * @param db Adodb database connection
  18. * @param tables List of tables to join
  19. * @rowfields List of fields to display on each row
  20. * @colfield Pivot field to slice and display in columns, if we want to calculate
  21. * ranges, we pass in an array (see example2)
  22. * @where Where clause. Optional.
  23. * @aggfield This is the field to sum. Optional.
  24. * Since 2.3.1, if you can use your own aggregate function
  25. * instead of SUM, eg. $sumfield = 'AVG(fieldname)';
  26. * @sumlabel Prefix to display in sum columns. Optional.
  27. * @aggfn Aggregate function to use (could be AVG, SUM, COUNT)
  28. * @showcount Show count of records
  29. *
  30. * @returns Sql generated
  31. */
  32. function PivotTableSQL($db,$tables,$rowfields,$colfield, $where=false,
  33. $aggfield = false,$sumlabel='Sum ',$aggfn ='SUM', $showcount = true)
  34. {
  35. if ($aggfield) $hidecnt = true;
  36. else $hidecnt = false;
  37. $iif = strpos($db->databaseType,'access') !== false;
  38. // note - vfp still doesn' work even with IIF enabled || $db->databaseType == 'vfp';
  39. //$hidecnt = false;
  40. if ($where) $where = "\nWHERE $where";
  41. if (!is_array($colfield)) $colarr = $db->GetCol("select distinct $colfield from $tables $where order by 1");
  42. if (!$aggfield) $hidecnt = false;
  43. $sel = "$rowfields, ";
  44. if (is_array($colfield)) {
  45. foreach ($colfield as $k => $v) {
  46. $k = trim($k);
  47. if (!$hidecnt) {
  48. $sel .= $iif ?
  49. "\n\t$aggfn(IIF($v,1,0)) AS \"$k\", "
  50. :
  51. "\n\t$aggfn(CASE WHEN $v THEN 1 ELSE 0 END) AS \"$k\", ";
  52. }
  53. if ($aggfield) {
  54. $sel .= $iif ?
  55. "\n\t$aggfn(IIF($v,$aggfield,0)) AS \"$sumlabel$k\", "
  56. :
  57. "\n\t$aggfn(CASE WHEN $v THEN $aggfield ELSE 0 END) AS \"$sumlabel$k\", ";
  58. }
  59. }
  60. } else {
  61. foreach ($colarr as $v) {
  62. if (!is_numeric($v)) $vq = $db->qstr($v);
  63. else $vq = $v;
  64. $v = trim($v);
  65. if (strlen($v) == 0 ) $v = 'null';
  66. if (!$hidecnt) {
  67. $sel .= $iif ?
  68. "\n\t$aggfn(IIF($colfield=$vq,1,0)) AS \"$v\", "
  69. :
  70. "\n\t$aggfn(CASE WHEN $colfield=$vq THEN 1 ELSE 0 END) AS \"$v\", ";
  71. }
  72. if ($aggfield) {
  73. if ($hidecnt) $label = $v;
  74. else $label = "{$v}_$aggfield";
  75. $sel .= $iif ?
  76. "\n\t$aggfn(IIF($colfield=$vq,$aggfield,0)) AS \"$label\", "
  77. :
  78. "\n\t$aggfn(CASE WHEN $colfield=$vq THEN $aggfield ELSE 0 END) AS \"$label\", ";
  79. }
  80. }
  81. }
  82. if ($aggfield && $aggfield != '1'){
  83. $agg = "$aggfn($aggfield)";
  84. $sel .= "\n\t$agg as \"$sumlabel$aggfield\", ";
  85. }
  86. if ($showcount)
  87. $sel .= "\n\tSUM(1) as Total";
  88. else
  89. $sel = substr($sel,0,strlen($sel)-2);
  90. $sql = "SELECT $sel \nFROM $tables $where \nGROUP BY $rowfields";
  91. return $sql;
  92. }
  93. /* EXAMPLES USING MS NORTHWIND DATABASE */
  94. if (0) {
  95. # example1
  96. #
  97. # Query the main "product" table
  98. # Set the rows to CompanyName and QuantityPerUnit
  99. # and the columns to the Categories
  100. # and define the joins to link to lookup tables
  101. # "categories" and "suppliers"
  102. #
  103. $sql = PivotTableSQL(
  104. $gDB, # adodb connection
  105. 'products p ,categories c ,suppliers s', # tables
  106. 'CompanyName,QuantityPerUnit', # row fields
  107. 'CategoryName', # column fields
  108. 'p.CategoryID = c.CategoryID and s.SupplierID= p.SupplierID' # joins/where
  109. );
  110. print "<pre>$sql";
  111. $rs = $gDB->Execute($sql);
  112. rs2html($rs);
  113. /*
  114. Generated SQL:
  115. SELECT CompanyName,QuantityPerUnit,
  116. SUM(CASE WHEN CategoryName='Beverages' THEN 1 ELSE 0 END) AS "Beverages",
  117. SUM(CASE WHEN CategoryName='Condiments' THEN 1 ELSE 0 END) AS "Condiments",
  118. SUM(CASE WHEN CategoryName='Confections' THEN 1 ELSE 0 END) AS "Confections",
  119. SUM(CASE WHEN CategoryName='Dairy Products' THEN 1 ELSE 0 END) AS "Dairy Products",
  120. SUM(CASE WHEN CategoryName='Grains/Cereals' THEN 1 ELSE 0 END) AS "Grains/Cereals",
  121. SUM(CASE WHEN CategoryName='Meat/Poultry' THEN 1 ELSE 0 END) AS "Meat/Poultry",
  122. SUM(CASE WHEN CategoryName='Produce' THEN 1 ELSE 0 END) AS "Produce",
  123. SUM(CASE WHEN CategoryName='Seafood' THEN 1 ELSE 0 END) AS "Seafood",
  124. SUM(1) as Total
  125. FROM products p ,categories c ,suppliers s WHERE p.CategoryID = c.CategoryID and s.SupplierID= p.SupplierID
  126. GROUP BY CompanyName,QuantityPerUnit
  127. */
  128. //=====================================================================
  129. # example2
  130. #
  131. # Query the main "product" table
  132. # Set the rows to CompanyName and QuantityPerUnit
  133. # and the columns to the UnitsInStock for diiferent ranges
  134. # and define the joins to link to lookup tables
  135. # "categories" and "suppliers"
  136. #
  137. $sql = PivotTableSQL(
  138. $gDB, # adodb connection
  139. 'products p ,categories c ,suppliers s', # tables
  140. 'CompanyName,QuantityPerUnit', # row fields
  141. # column ranges
  142. array(
  143. ' 0 ' => 'UnitsInStock <= 0',
  144. "1 to 5" => '0 < UnitsInStock and UnitsInStock <= 5',
  145. "6 to 10" => '5 < UnitsInStock and UnitsInStock <= 10',
  146. "11 to 15" => '10 < UnitsInStock and UnitsInStock <= 15',
  147. "16+" =>'15 < UnitsInStock'
  148. ),
  149. ' p.CategoryID = c.CategoryID and s.SupplierID= p.SupplierID', # joins/where
  150. 'UnitsInStock', # sum this field
  151. 'Sum' # sum label prefix
  152. );
  153. print "<pre>$sql";
  154. $rs = $gDB->Execute($sql);
  155. rs2html($rs);
  156. /*
  157. Generated SQL:
  158. SELECT CompanyName,QuantityPerUnit,
  159. SUM(CASE WHEN UnitsInStock <= 0 THEN UnitsInStock ELSE 0 END) AS "Sum 0 ",
  160. SUM(CASE WHEN 0 < UnitsInStock and UnitsInStock <= 5 THEN UnitsInStock ELSE 0 END) AS "Sum 1 to 5",
  161. SUM(CASE WHEN 5 < UnitsInStock and UnitsInStock <= 10 THEN UnitsInStock ELSE 0 END) AS "Sum 6 to 10",
  162. SUM(CASE WHEN 10 < UnitsInStock and UnitsInStock <= 15 THEN UnitsInStock ELSE 0 END) AS "Sum 11 to 15",
  163. SUM(CASE WHEN 15 < UnitsInStock THEN UnitsInStock ELSE 0 END) AS "Sum 16+",
  164. SUM(UnitsInStock) AS "Sum UnitsInStock",
  165. SUM(1) as Total
  166. FROM products p ,categories c ,suppliers s WHERE p.CategoryID = c.CategoryID and s.SupplierID= p.SupplierID
  167. GROUP BY CompanyName,QuantityPerUnit
  168. */
  169. }
  170. ?>