PageRenderTime 28ms CodeModel.GetById 5ms RepoModel.GetById 0ms app.codeStats 0ms

/lib/pear/Spreadsheet/Excel/Writer/BIFFwriter.php

http://github.com/moodle/moodle
PHP | 238 lines | 91 code | 21 blank | 126 comment | 10 complexity | 4fb4a6aaa6a0887f7049f45d2a3436d0 MD5 | raw file
Possible License(s): MIT, AGPL-3.0, MPL-2.0-no-copyleft-exception, LGPL-3.0, GPL-3.0, Apache-2.0, LGPL-2.1, BSD-3-Clause
  1. <?php
  2. /*
  3. * Module written/ported by Xavier Noguer <xnoguer@php.net>
  4. *
  5. * The majority of this is _NOT_ my code. I simply ported it from the
  6. * PERL Spreadsheet::WriteExcel module.
  7. *
  8. * The author of the Spreadsheet::WriteExcel module is John McNamara
  9. * <jmcnamara@cpan.org>
  10. *
  11. * I _DO_ maintain this code, and John McNamara has nothing to do with the
  12. * porting of this code to PHP. Any questions directly related to this
  13. * class library should be directed to me.
  14. *
  15. * License Information:
  16. *
  17. * Spreadsheet_Excel_Writer: A library for generating Excel Spreadsheets
  18. * Copyright (c) 2002-2003 Xavier Noguer xnoguer@php.net
  19. *
  20. * This library is free software; you can redistribute it and/or
  21. * modify it under the terms of the GNU Lesser General Public
  22. * License as published by the Free Software Foundation; either
  23. * version 2.1 of the License, or (at your option) any later version.
  24. *
  25. * This library is distributed in the hope that it will be useful,
  26. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  27. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  28. * Lesser General Public License for more details.
  29. *
  30. * You should have received a copy of the GNU Lesser General Public
  31. * License along with this library; if not, write to the Free Software
  32. * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
  33. */
  34. require_once 'PEAR.php';
  35. /**
  36. * Class for writing Excel BIFF records.
  37. *
  38. * From "MICROSOFT EXCEL BINARY FILE FORMAT" by Mark O'Brien (Microsoft Corporation):
  39. *
  40. * BIFF (BInary File Format) is the file format in which Excel documents are
  41. * saved on disk. A BIFF file is a complete description of an Excel document.
  42. * BIFF files consist of sequences of variable-length records. There are many
  43. * different types of BIFF records. For example, one record type describes a
  44. * formula entered into a cell; one describes the size and location of a
  45. * window into a document; another describes a picture format.
  46. *
  47. * @author Xavier Noguer <xnoguer@php.net>
  48. * @category FileFormats
  49. * @package Spreadsheet_Excel_Writer
  50. */
  51. class Spreadsheet_Excel_Writer_BIFFwriter extends PEAR
  52. {
  53. /**
  54. * The BIFF/Excel version (5).
  55. * @var integer
  56. */
  57. var $_BIFF_version = 0x0500;
  58. /**
  59. * The byte order of this architecture. 0 => little endian, 1 => big endian
  60. * @var integer
  61. */
  62. var $_byte_order;
  63. /**
  64. * The string containing the data of the BIFF stream
  65. * @var string
  66. */
  67. var $_data;
  68. /**
  69. * The size of the data in bytes. Should be the same as strlen($this->_data)
  70. * @var integer
  71. */
  72. var $_datasize;
  73. /**
  74. * The maximun length for a BIFF record. See _addContinue()
  75. * @var integer
  76. * @see _addContinue()
  77. */
  78. var $_limit;
  79. /**
  80. * Constructor
  81. *
  82. * @access public
  83. */
  84. function Spreadsheet_Excel_Writer_BIFFwriter()
  85. {
  86. $this->_byte_order = '';
  87. $this->_data = '';
  88. $this->_datasize = 0;
  89. $this->_limit = 2080;
  90. // Set the byte order
  91. $this->_setByteOrder();
  92. }
  93. /**
  94. * Determine the byte order and store it as class data to avoid
  95. * recalculating it for each call to new().
  96. *
  97. * @access private
  98. */
  99. function _setByteOrder()
  100. {
  101. // Check if "pack" gives the required IEEE 64bit float
  102. $teststr = pack("d", 1.2345);
  103. $number = pack("C8", 0x8D, 0x97, 0x6E, 0x12, 0x83, 0xC0, 0xF3, 0x3F);
  104. if ($number == $teststr) {
  105. $byte_order = 0; // Little Endian
  106. } elseif ($number == strrev($teststr)){
  107. $byte_order = 1; // Big Endian
  108. } else {
  109. // Give up. I'll fix this in a later version.
  110. return $this->raiseError("Required floating point format ".
  111. "not supported on this platform.");
  112. }
  113. $this->_byte_order = $byte_order;
  114. }
  115. /**
  116. * General storage function
  117. *
  118. * @param string $data binary data to prepend
  119. * @access private
  120. */
  121. function _prepend($data)
  122. {
  123. if (strlen($data) > $this->_limit) {
  124. $data = $this->_addContinue($data);
  125. }
  126. $this->_data = $data.$this->_data;
  127. $this->_datasize += strlen($data);
  128. }
  129. /**
  130. * General storage function
  131. *
  132. * @param string $data binary data to append
  133. * @access private
  134. */
  135. function _append($data)
  136. {
  137. if (strlen($data) > $this->_limit) {
  138. $data = $this->_addContinue($data);
  139. }
  140. $this->_data = $this->_data.$data;
  141. $this->_datasize += strlen($data);
  142. }
  143. /**
  144. * Writes Excel BOF record to indicate the beginning of a stream or
  145. * sub-stream in the BIFF file.
  146. *
  147. * @param integer $type Type of BIFF file to write: 0x0005 Workbook,
  148. * 0x0010 Worksheet.
  149. * @access private
  150. */
  151. function _storeBof($type)
  152. {
  153. $record = 0x0809; // Record identifier
  154. // According to the SDK $build and $year should be set to zero.
  155. // However, this throws a warning in Excel 5. So, use magic numbers.
  156. if ($this->_BIFF_version == 0x0500) {
  157. $length = 0x0008;
  158. $unknown = '';
  159. $build = 0x096C;
  160. $year = 0x07C9;
  161. } elseif ($this->_BIFF_version == 0x0600) {
  162. $length = 0x0010;
  163. $unknown = pack("VV", 0x00000041, 0x00000006); //unknown last 8 bytes for BIFF8
  164. $build = 0x0DBB;
  165. $year = 0x07CC;
  166. }
  167. $version = $this->_BIFF_version;
  168. $header = pack("vv", $record, $length);
  169. $data = pack("vvvv", $version, $type, $build, $year);
  170. $this->_prepend($header . $data . $unknown);
  171. }
  172. /**
  173. * Writes Excel EOF record to indicate the end of a BIFF stream.
  174. *
  175. * @access private
  176. */
  177. function _storeEof()
  178. {
  179. $record = 0x000A; // Record identifier
  180. $length = 0x0000; // Number of bytes to follow
  181. $header = pack("vv", $record, $length);
  182. $this->_append($header);
  183. }
  184. /**
  185. * Excel limits the size of BIFF records. In Excel 5 the limit is 2084 bytes. In
  186. * Excel 97 the limit is 8228 bytes. Records that are longer than these limits
  187. * must be split up into CONTINUE blocks.
  188. *
  189. * This function takes a long BIFF record and inserts CONTINUE records as
  190. * necessary.
  191. *
  192. * @param string $data The original binary data to be written
  193. * @return string A very convenient string of continue blocks
  194. * @access private
  195. */
  196. function _addContinue($data)
  197. {
  198. $limit = $this->_limit;
  199. $record = 0x003C; // Record identifier
  200. // The first 2080/8224 bytes remain intact. However, we have to change
  201. // the length field of the record.
  202. $tmp = substr($data, 0, 2).pack("v", $limit-4).substr($data, 4, $limit - 4);
  203. $header = pack("vv", $record, $limit); // Headers for continue records
  204. // Retrieve chunks of 2080/8224 bytes +4 for the header.
  205. $data_length = strlen($data);
  206. for ($i = $limit; $i < ($data_length - $limit); $i += $limit) {
  207. $tmp .= $header;
  208. $tmp .= substr($data, $i, $limit);
  209. }
  210. // Retrieve the last chunk of data
  211. $header = pack("vv", $record, strlen($data) - $i);
  212. $tmp .= $header;
  213. $tmp .= substr($data, $i, strlen($data) - $i);
  214. return $tmp;
  215. }
  216. }
  217. ?>