📁 PHP Dosya Yöneticisi
/
/
home
/
demodesigncom
/
b2be-ticaretscripti.demodesign.com.tr
/
boss_old
/
excel
📝
SimpleXLSX.php
← Geri Dön
<?php /** * SimpleXLSX php class v0.8.9 * MS Excel 2007 workbooks reader * * Copyright (c) 2012 - 2019 SimpleXLSX * * @category SimpleXLSX * @package SimpleXLSX * @copyright Copyright (c) 2012 - 2019 SimpleXLSX (https://github.com/shuchkin/simplexlsx/) * @license MIT * @version 0.8.9 */ /** Examples * * Example 1: * if ( $xlsx = SimpleXLSX::parse('book.xlsx') ) { * print_r( $xlsx->rows() ); * } else { * echo SimpleXLSX::parseError(); * } * * Example 2: html table * if ( $xlsx = SimpleXLSX::parse('book.xlsx') ) { * echo $xlsx->toHTML(); * } else { * echo SimpleXLSX::parseError(); * } * * Example 3: rowsEx * $xlsx = SimpleXLSX::parse('book.xlsx'); * print_r( $xlsx->rowsEx() ); * * Example 4: select worksheet * $xlsx = SimpleXLSX::parse('book.xlsx'); * print_r( $xlsx->rows(1) ); // second worksheet * * Example 5: IDs and worksheet names * $xlsx = SimpleXLSX::parse('book.xlsx'); * print_r( $xlsx->sheetNames() ); // array( 0 => 'Sheet 1', 1 => 'Catalog' ); * * Example 6: get sheet name by index * $xlsx = SimpleXLSX::parse('book.xlsx'); * echo 'Sheet Name 2 = '.$xlsx->sheetName(1); * * Example 7: getCell (very slow) * echo $xlsx->getCell(1,'D12'); // reads D12 cell from second sheet * * Example 8: read data * if ( $xlsx = SimpleXLSX::parse( file_get_contents('http://www.example.com/example.xlsx'), true) ) { * $dim = $xlsx->dimension(1); * $num_cols = $dim[0]; * $num_rows = $dim[1]; * echo $xlsx->sheetName(1).':'.$num_cols.'x'.$num_rows; * } else { * echo SimpleXLSX::parseError(); * } * * Example 9: old style * $xlsx = new SimpleXLSX('book.xlsx'); * if ( $xlsx->success() ) { * print_r( $xlsx->rows() ); * } else { * echo 'xlsx error: '.$xlsx->error(); * } */ /** Changelog * v0.8.9 (2019-08-15) fixed release version * v0.8.8 (2019-06-19) removed list( $x, $y ), added bool $xlsx->skipEmptyRows, $xlsx->parseFile( $filename ), $xlsx->parseData( $data ), release 0.8.8 * v0.8.7 (2019-04-18) empty rows fixed * v0.8.6 (2019-04-16) 1900/1904 bug fixed * v0.8.5 (2019-03-07) SimpleXLSX::ParseErrno(), $xlsx->errno() returns error code * v0.8.4 (2019-02-14) detect datetime values, mb_string.func_overload=2 support .!. Bitrix * v0.8.3 (2018-11-14) getCell - fixed empty cells and rows, safe now, but very slow * v0.8.2 (2018-11-09) fix empty cells and rows in rows() and rowsEx(), added setDateTimeFormat( $see_php_date_func ) * v0.8.1 (2018-11-07) rename simplexlsx.php to SimpleXLSX.php, rename parse_error to parseError fix _columnIndex, add ->toHTML(), GNU to MIT license * v0.7.13 (2018-06-18) get sheet indexes bug fix * v0.7.12 (2018-06-17) $worksheet_id to $worksheetIndex, sheet numeration started 0 * v0.7.11 (2018-04-25) rowsEx(), added row index "r" to cell info * v0.7.10 (2018-04-21) fixed getCell, returns NULL if not exits * v0.7.9 (2018-01-15) fixed sheetNames() (namespaced or not namespaced attr) * v0.7.8 (2018-01-15) remove namespace prefixes (hardcoded) * v0.7.7 (2017-10-02) XML External Entity (XXE) Prevention (<!ENTITY xxe SYSTEM "file: ///etc/passwd" >]>) * v0.7.6 (2017-09-26) if worksheet_id === 0 (default) then detect first sheet (for LibreOffice capabilities) * v0.7.5 (2017-09-10) ->getCell() - fixed * v0.7.4 (2017-08-22) ::parseError() - get last error in "static style" * v0.7.3 (2017-08-14) ->_parse fixed relations reader, added ->getCell( sheet_id, address, format ) for direct cell reading * v0.7.2 (2017-05-13) ::parse( $filename ) helper method * v0.7.1 (2017-03-29) License added * v0.6.11 (2016-07-27) fixed timestamp() * v0.6.10 (2016-06-10) fixed search entries (UPPERCASE) * v0.6.9 (2015-04-12) $xlsx->datetimeFormat to force dates out * v0.6.8 (2013-10-13) fixed dimension() where 1 row only, fixed rowsEx() empty cells indexes (Daniel Stastka) * v0.6.7 (2013-08-10) fixed unzip (mac), added $debug param to _constructor to display errors * v0.6.6 (2013-06-03) +entryExists(), * v0.6.5 (2013-03-18) fixed sheetName() * v0.6.4 (2013-03-13) rowsEx(), _parse(): fixed date column type & format detection * v0.6.3 (2013-03-13) rowsEx(): fixed formulas, added date type 'd', added format 'format' * dimension(): fixed empty sheet dimension * + sheetNames() - returns array( sheet_id => sheet_name, sheet_id2 => sheet_name2 ...) * v0.6.2 (2012-10-04) fixed empty cells, rowsEx() returns type and formulas now * v0.6.1 (2012-09-14) removed "raise exception" and fixed _unzip * v0.6 (2012-09-13) success(), error(), __constructor( $filename, $is_data = false ) * v0.5.1 (2012-09-13) sheetName() fixed * v0.5 (2012-09-12) sheetName() * v0.4 sheets(), sheetsCount(), unixstamp( $excelDateTime ) * v0.3 - fixed empty cells (Gonzo patch) */ /** @noinspection PhpUndefinedFieldInspection */ /** @noinspection PhpComposerExtensionStubsInspection */ /** @noinspection MultiAssignmentUsageInspection */ class SimpleXLSX { // Don't remove this string! Created by Sergey Shuchkin sergey.shuchkin@gmail.com const SCHEMA_REL_OFFICEDOCUMENT = 'http://schemas.openxmlformats.org/officeDocument/2006/relationships/officeDocument'; const SCHEMA_REL_SHAREDSTRINGS = 'http://schemas.openxmlformats.org/officeDocument/2006/relationships/sharedStrings'; const SCHEMA_REL_WORKSHEET = 'http://schemas.openxmlformats.org/officeDocument/2006/relationships/worksheet'; const SCHEMA_REL_STYLES = 'http://schemas.openxmlformats.org/officeDocument/2006/relationships/styles'; public static $CF = array( // Cell formats 0 => 'General', 1 => '0', 2 => '0.00', 3 => '#,##0', 4 => '#,##0.00', 9 => '0%', 10 => '0.00%', 11 => '0.00E+00', 12 => '# ?/?', 13 => '# ??/??', 14 => 'mm-dd-yy', 15 => 'd-mmm-yy', 16 => 'd-mmm', 17 => 'mmm-yy', 18 => 'h:mm AM/PM', 19 => 'h:mm:ss AM/PM', 20 => 'h:mm', 21 => 'h:mm:ss', 22 => 'm/d/yy h:mm', 37 => '#,##0 ;(#,##0)', 38 => '#,##0 ;[Red](#,##0)', 39 => '#,##0.00;(#,##0.00)', 40 => '#,##0.00;[Red](#,##0.00)', 44 => '_("$"* #,##0.00_);_("$"* \(#,##0.00\);_("$"* "-"??_);_(@_)', 45 => 'mm:ss', 46 => '[h]:mm:ss', 47 => 'mmss.0', 48 => '##0.0E+0', 49 => '@', 27 => '[$-404]e/m/d', 30 => 'm/d/yy', 36 => '[$-404]e/m/d', 50 => '[$-404]e/m/d', 57 => '[$-404]e/m/d', 59 => 't0', 60 => 't0.00', 61 => 't#,##0', 62 => 't#,##0.00', 67 => 't0%', 68 => 't0.00%', 69 => 't# ?/?', 70 => 't# ??/??', ); public $cellFormats = array(); public $datetimeFormat = 'Y-m-d H:i:s'; public $skipEmptyRows = false; public $debug; /* @var SimpleXMLElement $workbook */ private $workbook; /* @var SimpleXMLElement[] $sheets */ private $sheets; private $sheetNames = array(); // scheme private $styles; private $hyperlinks; /* @var array[] $package */ private $package; private $datasec; private $sharedstrings; private $date1904 = 0; /* private $date_formats = array( 0xe => "d/m/Y", 0xf => "d-M-Y", 0x10 => "d-M", 0x11 => "M-Y", 0x12 => "h:i a", 0x13 => "h:i:s a", 0x14 => "H:i", 0x15 => "H:i:s", 0x16 => "d/m/Y H:i", 0x2d => "i:s", 0x2e => "H:i:s", 0x2f => "i:s.S" ); private $number_formats = array( 0x1 => "%1.0f", // "0" 0x2 => "%1.2f", // "0.00", 0x3 => "%1.0f", //"#,##0", 0x4 => "%1.2f", //"#,##0.00", 0x5 => "%1.0f", //"$#,##0;($#,##0)", 0x6 => '$%1.0f', //"$#,##0;($#,##0)", 0x7 => '$%1.2f', //"$#,##0.00;($#,##0.00)", 0x8 => '$%1.2f', //"$#,##0.00;($#,##0.00)", 0x9 => '%1.0f%%', //"0%" 0xa => '%1.2f%%', //"0.00%" 0xb => '%1.2f', //"0.00E00", 0x25 => '%1.0f', //"#,##0;(#,##0)", 0x26 => '%1.0f', //"#,##0;(#,##0)", 0x27 => '%1.2f', //"#,##0.00;(#,##0.00)", 0x28 => '%1.2f', //"#,##0.00;(#,##0.00)", 0x29 => '%1.0f', //"#,##0;(#,##0)", 0x2a => '$%1.0f', //"$#,##0;($#,##0)", 0x2b => '%1.2f', //"#,##0.00;(#,##0.00)", 0x2c => '$%1.2f', //"$#,##0.00;($#,##0.00)", 0x30 => '%1.0f'); //"##0.0E0"; // }}} */ private $errno = 0; private $error = false; public function __construct( $filename = null, $is_data = null, $debug = null ) { if ( $debug !== null ) { $this->debug = $debug; } $this->package = array( 'filename' => '', 'mtime' => 0, 'size' => 0, 'comment' => '', 'entries' => array() ); if ( $filename && $this->_unzip( $filename, $is_data )) { $this->_parse(); } } public function parseFile( $filename ) { if ( $this->_unzip( $filename )) { return $this->_parse(); } return false; } public function parseData( $data ) { if ( $this->_unzip($data, true )) { return $this->_parse(); } return false; } private function _unzip( $filename, $is_data = false ) { // Clear current file $this->datasec = array(); if ( $is_data ) { $this->package['filename'] = 'default.xlsx'; $this->package['mtime'] = time(); $this->package['size'] = $this->_strlen( $filename ); $vZ = $filename; } else { if ( ! is_readable( $filename ) ) { $this->error( 1, 'File not found ' . $filename ); return false; } // Package information $this->package['filename'] = $filename; $this->package['mtime'] = filemtime( $filename ); $this->package['size'] = filesize( $filename ); // Read file $vZ = file_get_contents( $filename ); } // Cut end of central directory /* $aE = explode("\x50\x4b\x05\x06", $vZ); if (count($aE) == 1) { $this->error('Unknown format'); return false; } */ if ( ( $pcd = $this->_strrpos( $vZ, "\x50\x4b\x05\x06" ) ) === false ) { $this->error( 2, 'Unknown archive format' ); return false; } $aE = array( 0 => $this->_substr( $vZ, 0, $pcd ), 1 => $this->_substr( $vZ, $pcd + 3 ) ); // Normal way $aP = unpack( 'x16/v1CL', $aE[1] ); $this->package['comment'] = $this->_substr( $aE[1], 18, $aP['CL'] ); // Translates end of line from other operating systems $this->package['comment'] = strtr( $this->package['comment'], array( "\r\n" => "\n", "\r" => "\n" ) ); // Cut the entries from the central directory $aE = explode( "\x50\x4b\x01\x02", $vZ ); // Explode to each part $aE = explode( "\x50\x4b\x03\x04", $aE[0] ); // Shift out spanning signature or empty entry array_shift( $aE ); // Loop through the entries foreach ( $aE as $vZ ) { $aI = array(); $aI['E'] = 0; $aI['EM'] = ''; // Retrieving local file header information // $aP = unpack('v1VN/v1GPF/v1CM/v1FT/v1FD/V1CRC/V1CS/V1UCS/v1FNL', $vZ); $aP = unpack( 'v1VN/v1GPF/v1CM/v1FT/v1FD/V1CRC/V1CS/V1UCS/v1FNL/v1EFL', $vZ ); // Check if data is encrypted // $bE = ($aP['GPF'] && 0x0001) ? TRUE : FALSE; $bE = false; $nF = $aP['FNL']; $mF = $aP['EFL']; // Special case : value block after the compressed data if ( $aP['GPF'] & 0x0008 ) { $aP1 = unpack( 'V1CRC/V1CS/V1UCS', $this->_substr( $vZ, - 12 ) ); $aP['CRC'] = $aP1['CRC']; $aP['CS'] = $aP1['CS']; $aP['UCS'] = $aP1['UCS']; // 2013-08-10 $vZ = $this->_substr( $vZ, 0, - 12 ); if ( $this->_substr( $vZ, - 4 ) === "\x50\x4b\x07\x08" ) { $vZ = $this->_substr( $vZ, 0, - 4 ); } } // Getting stored filename $aI['N'] = $this->_substr( $vZ, 26, $nF ); if ( $this->_substr( $aI['N'], - 1 ) === '/' ) { // is a directory entry - will be skipped continue; } // Truncate full filename in path and filename $aI['P'] = dirname( $aI['N'] ); $aI['P'] = $aI['P'] === '.' ? '' : $aI['P']; $aI['N'] = basename( $aI['N'] ); $vZ = $this->_substr( $vZ, 26 + $nF + $mF ); if ( $this->_strlen( $vZ ) !== (int) $aP['CS'] ) { // check only if availabled $aI['E'] = 1; $aI['EM'] = 'Compressed size is not equal with the value in header information.'; } else if ( $bE ) { $aI['E'] = 5; $aI['EM'] = 'File is encrypted, which is not supported from this class.'; } else { switch ( $aP['CM'] ) { case 0: // Stored // Here is nothing to do, the file ist flat. break; case 8: // Deflated $vZ = gzinflate( $vZ ); break; case 12: // BZIP2 if ( extension_loaded( 'bz2' ) ) { /** @noinspection PhpComposerExtensionStubsInspection */ $vZ = bzdecompress( $vZ ); } else { $aI['E'] = 7; $aI['EM'] = 'PHP BZIP2 extension not available.'; } break; default: $aI['E'] = 6; $aI['EM'] = "De-/Compression method {$aP['CM']} is not supported."; } if ( ! $aI['E'] ) { if ( $vZ === false ) { $aI['E'] = 2; $aI['EM'] = 'Decompression of data failed.'; } else if ( $this->_strlen( $vZ ) !== (int) $aP['UCS'] ) { $aI['E'] = 3; $aI['EM'] = 'Uncompressed size is not equal with the value in header information.'; } else if ( crc32( $vZ ) !== $aP['CRC'] ) { $aI['E'] = 4; $aI['EM'] = 'CRC32 checksum is not equal with the value in header information.'; } } } $aI['D'] = $vZ; // DOS to UNIX timestamp $aI['T'] = mktime( ( $aP['FT'] & 0xf800 ) >> 11, ( $aP['FT'] & 0x07e0 ) >> 5, ( $aP['FT'] & 0x001f ) << 1, ( $aP['FD'] & 0x01e0 ) >> 5, $aP['FD'] & 0x001f, ( ( $aP['FD'] & 0xfe00 ) >> 9 ) + 1980 ); //$this->Entries[] = &new SimpleUnzipEntry($aI); $this->package['entries'][] = array( 'data' => $aI['D'], 'error' => $aI['E'], 'error_msg' => $aI['EM'], 'name' => $aI['N'], 'path' => $aI['P'], 'time' => $aI['T'] ); } // end for each entries return true; } // sheets numeration: 1,2,3.... public function error( $num = null, $str = null ) { if ( $num ) { $this->errno = $num; $this->error = $str; if ( $this->debug ) { trigger_error( __CLASS__ . ': ' . $this->error, E_USER_WARNING ); } } return $this->error; } public function errno() { return $this->errno; } private function _parse() { // Document data holders $this->sharedstrings = array(); $this->sheets = array(); // $this->styles = array(); // Read relations and search for officeDocument if ( $relations = $this->getEntryXML( '_rels/.rels' ) ) { foreach ( $relations->Relationship as $rel ) { $rel_type = trim( (string) $rel['Type'] ); $rel_target = trim( (string) $rel['Target'] ); if ( $rel_type === self::SCHEMA_REL_OFFICEDOCUMENT && $this->workbook = $this->getEntryXML( $rel_target ) ) { $index_rId = array(); // [0 => rId1] $index = 0; foreach ( $this->workbook->sheets->sheet as $s ) { /* @var SimpleXMLElement $s */ $this->sheetNames[ $index ] = (string) $s['name']; $index_rId[ $index ] = (string) $s['id']; $index++; } if ( (int) $this->workbook->workbookPr['date1904'] === 1 ) { $this->date1904 = 1; } // print_r( $index_rId ); if ( $workbookRelations = $this->getEntryXML( dirname( $rel_target ) . '/_rels/workbook.xml.rels' ) ) { // Loop relations for workbook and extract sheets... foreach ( $workbookRelations->Relationship as $workbookRelation ) { $wrel_type = trim( (string) $workbookRelation['Type'] ); $wrel_path = dirname( trim( (string) $rel['Target'] ) ) . '/' . trim( (string) $workbookRelation['Target'] ); if ( ! $this->entryExists( $wrel_path ) ) { continue; } if ( $wrel_type === self::SCHEMA_REL_WORKSHEET ) { // Sheets if ( $sheet = $this->getEntryXML( $wrel_path ) ) { $index = array_search( (string) $workbookRelation['Id'], $index_rId, false ); $this->sheets[ $index ] = $sheet; } } else if ( $wrel_type === self::SCHEMA_REL_SHAREDSTRINGS ) { if ( $sharedStrings = $this->getEntryXML( $wrel_path ) ) { foreach ( $sharedStrings->si as $val ) { if ( isset( $val->t ) ) { $this->sharedstrings[] = (string) $val->t; } elseif ( isset( $val->r ) ) { $this->sharedstrings[] = $this->_parseRichText( $val ); } } } } else if ( $wrel_type === self::SCHEMA_REL_STYLES ) { $this->styles = $this->getEntryXML( $wrel_path ); $nf = array(); if ( $this->styles->numFmts->numFmt !== null ) { foreach ( $this->styles->numFmts->numFmt as $v ) { $nf[ (int) $v['numFmtId'] ] = (string) $v['formatCode']; } } if ( $this->styles->cellXfs->xf !== null ) { foreach ( $this->styles->cellXfs->xf as $v ) { $v = (array) $v->attributes(); $v['format'] = ''; if ( isset( $v['@attributes']['numFmtId'] ) ) { $v = $v['@attributes']; $fid = (int) $v['numFmtId']; if ( isset( self::$CF[ $fid ] ) ) { $v['format'] = self::$CF[ $fid ]; } else if ( isset( $nf[ $fid ] ) ) { $v['format'] = $nf[ $fid ]; } } $this->cellFormats[] = $v; } } } } break; } } } } if ( count( $this->sheets ) ) { // Sort sheets ksort( $this->sheets ); return true; } return false; } /* * @param string $name Filename in archive * @return SimpleXMLElement|bool */ public function getEntryXML( $name ) { if ( $entry_xml = $this->getEntryData( $name ) ) { // dirty remove namespace prefixes and empty rows $entry_xml = preg_replace('/xmlns[^=]*="[^"]*"/i','', $entry_xml ); // remove namespaces $entry_xml = preg_replace('/[a-zA-Z0-9]+:([a-zA-Z0-9]+="[^"]+")/','$1$2', $entry_xml ); // remove namespaced attrs $entry_xml = preg_replace('/<[a-zA-Z0-9]+:([^>]+)>/', '<$1>', $entry_xml); // fix namespaced openned tags $entry_xml = preg_replace('/<\/[a-zA-Z0-9]+:([^>]+)>/', '</$1>', $entry_xml); // fix namespaced closed tags if ( $this->skipEmptyRows && strpos($name, '/sheet') ) { $entry_xml = preg_replace( '/<row[^>]+>\s*?(<c[^\/]+\/>)+\s*?<\/row>/', '', $entry_xml,-1, $cnt ); // remove empty rows if ( $cnt ) { $entry_xml = preg_replace('/<dimension[^\/]+\/>/', '', $entry_xml); } } // echo '<pre>'.$name."\r\n".htmlspecialchars( $entry_xml ).'</pre>'. // XML External Entity (XXE) Prevention $_old = libxml_disable_entity_loader(); $entry_xmlobj = simplexml_load_string( $entry_xml ); // echo '<pre>'.print_r( $entry_xmlobj, true).'</pre>'; libxml_disable_entity_loader($_old); if ( $entry_xmlobj ) { return $entry_xmlobj; } $e = libxml_get_last_error(); $this->error( 3, 'XML-entry ' . $name.' parser error '.$e->message.' line '.$e->line ); } else { $this->error( 4, 'XML-entry not found ' . $name ); } return false; } public function getEntryData( $name ) { $dir = $this->_strtoupper( dirname( $name ) ); $name = $this->_strtoupper( basename( $name ) ); foreach ( $this->package['entries'] as $entry ) { if ( $this->_strtoupper( $entry['path'] ) === $dir && $this->_strtoupper( $entry['name'] ) === $name ) { return $entry['data']; } } $this->error( 5, 'Entry not found '.$name ); return false; } public function entryExists( $name ) { // 0.6.6 $dir = $this->_strtoupper( dirname( $name ) ); $name = $this->_strtoupper( basename( $name ) ); foreach ( $this->package['entries'] as $entry ) { if ( $this->_strtoupper( $entry['path'] ) === $dir && $this->_strtoupper( $entry['name'] ) === $name ) { return true; } } return false; } private function _parseRichText( $is = null ) { $value = array(); if ( isset( $is->t ) ) { $value[] = (string) $is->t; } else if ( isset($is->r ) ) { foreach ( $is->r as $run ) { $value[] = (string) $run->t; } } return implode( '', $value ); } public static function parse( $filename, $is_data = false, $debug = false, $skip_empty_rows = false ) { $xlsx = new self(); $xlsx->debug = $debug; $xlsx->skipEmptyRows = $skip_empty_rows; $is_data ? $xlsx->parseData( $filename ) : $xlsx->parseFile( $filename ); if ( $xlsx->success() ) { return $xlsx; } self::parseError( $xlsx->error() ); self::parseErrno( $xlsx->errno() ); return false; } public static function parseError( $set = false ) { static $error = false; return $set ? $error = $set : $error; } public static function parseErrno( $set = false ) { static $errno = false; return $set ? $errno = $set : $errno; } public function success() { return ! $this->error; } public function rows( $worksheetIndex = 0 ) { if ( ( $ws = $this->worksheet( $worksheetIndex ) ) === false ) { return false; } $dim = $this->dimension( $worksheetIndex ); $numCols = $dim[0]; $numRows = $dim[1]; $emptyRow = array(); for( $i = 0; $i < $numCols; $i++) { $emptyRow[] = ''; } $rows = array(); for( $i = 0; $i < $numRows; $i++) { $rows[] = $emptyRow; } $curR = 0; /* @var SimpleXMLElement $ws */ foreach ( $ws->sheetData->row as $row ) { $curC = 0; foreach ( $row->c as $c ) { // detect skipped cols $idx = $this->getIndex( (string) $c['r'] ); $x = $idx[0]; $y = $idx[1]; if ( $x > -1 ) { $curC = $x; $curR = $y; } $rows[ $curR ][ $curC ] = $this->value( $c ); $curC++; } $curR ++; } return $rows; } public function rowsEx( $worksheetIndex = 0 ) { if ( ( $ws = $this->worksheet( $worksheetIndex ) ) === false ) { return false; } $rows = array(); $dim = $this->dimension( $worksheetIndex ); $numCols = $dim[0]; $numRows = $dim[1]; /** @noinspection ForeachInvariantsInspection */ for ( $y = 0; $y < $numRows; $y++ ) { /** @noinspection ForeachInvariantsInspection */ for ( $x = 0; $x < $numCols; $x++ ) { // 0.6.8 $c = ''; for ( $k = $x; $k >= 0; $k = (int) ( $k / 26 ) - 1 ) { $c = chr( $k % 26 + 65 ) . $c; } $rows[ $y ][ $x ] = array( 'type' => '', 'name' => $c . ( $y + 1 ), 'value' => '', 'href' => '', 'f' => '', 'format' => '', 'r' => $y ); } } $curR = 0; /* @var SimpleXMLElement $ws */ foreach ( $ws->sheetData->row as $row ) { $r_idx = (int) $row['r']; $curC = 0; foreach ( $row->c as $c ) { $r = (string) $c['r']; $t = (string) $c['t']; $s = (int) $c['s']; $idx = $this->getIndex( $r ); $x = $idx[0]; $y = $idx[1]; if ( $x > -1 ) { $curC = $x; $curR = $y; } if ( $s > 0 && isset( $this->cellFormats[ $s ] ) ) { $format = $this->cellFormats[ $s ]['format']; } else { $format = ''; } $rows[ $curR ][ $curC ] = array( 'type' => $t, 'name' => (string) $c['r'], 'value' => $this->value( $c ), 'href' => $this->href( $c ), 'f' => (string) $c->f, 'format' => $format, 'r' => $r_idx ); $curC++; } $curR ++; } return $rows; } public function toHTML( $worksheetIndex = 0 ) { $s = '<table class=excel>'; foreach( $this->rows( $worksheetIndex ) as $r ) { $s .= '<tr>'; foreach ( $r as $c ) { $s .= '<td nowrap>'.( $c === '' ? ' ' : htmlspecialchars( $c, ENT_QUOTES )).'</td>'; } $s .= "</tr>\r\n"; } $s .= '</table>'; return $s; } public function worksheet( $worksheetIndex = 0 ) { if ( isset( $this->sheets[ $worksheetIndex ] ) ) { $ws = $this->sheets[ $worksheetIndex ]; if ( isset( $ws->hyperlinks ) ) { $this->hyperlinks = array(); foreach ( $ws->hyperlinks->hyperlink as $hyperlink ) { $this->hyperlinks[ (string) $hyperlink['ref'] ] = (string) $hyperlink['display']; } } return $ws; } $this->error( 6, 'Worksheet not found ' . $worksheetIndex ); return false; } /** * returns [numCols,numRows] of worksheet * * @param int $worksheetIndex * * @return array */ public function dimension( $worksheetIndex = 0 ) { if ( ( $ws = $this->worksheet( $worksheetIndex ) ) === false ) { return array(0,0); } /* @var SimpleXMLElement $ws */ $ref = (string) $ws->dimension['ref']; if ( $this->_strpos( $ref, ':' ) !== false ) { $d = explode( ':', $ref ); $idx = $this->getIndex( $d[1] ); return array( $idx[0] + 1, $idx[1] + 1 ); } if ( $ref !== '' ) { // 0.6.8 $index = $this->getIndex( $ref ); return array( $index[0] + 1, $index[1] + 1 ); } // slow method $maxC = $maxR = 0; foreach ( $ws->sheetData->row as $row ) { foreach ( $row->c as $c ) { $idx = $this->getIndex( (string) $c['r'] ); $x = $idx[0]; $y = $idx[1]; if ( $x > 0 ) { if ( $x > $maxC ) { $maxC = $x; } if ( $y > $maxR ) { $maxR = $y; } } } } return array( $maxC+1, $maxR+1 ); } public function getIndex( $cell = 'A1' ) { if ( preg_match( '/([A-Z]+)(\d+)/', $cell, $m ) ) { $col = $m[1]; $row = $m[2]; $colLen = $this->_strlen( $col ); $index = 0; for ( $i = $colLen - 1; $i >= 0; $i -- ) { /** @noinspection PowerOperatorCanBeUsedInspection */ $index += ( ord( $col[$i] ) - 64 ) * pow( 26, $colLen - $i - 1 ); } return array( $index - 1, $row - 1 ); } // $this->error( 'Invalid cell index ' . $cell ); return array(-1,-1); } public function value( $cell ) { // Determine data type $dataType = (string) $cell['t']; if ( !$dataType ) { // number $s = (int) $cell['s']; if ( $s > 0 && isset( $this->cellFormats[ $s ] ) ) { $format = $this->cellFormats[ $s ]['format']; if ( strpos( $format, 'm') !== false ) { $dataType = 'd'; } } } $value = ''; switch ( $dataType ) { case 's': // Value is a shared string if ( (string) $cell->v !== '' ) { $value = $this->sharedstrings[ (int) $cell->v ]; } break; case 'b': // Value is boolean $value = (string) $cell->v; if ( $value === '0' ) { $value = false; } else if ( $value === '1' ) { $value = true; } else { $value = (bool) $cell->v; } break; case 'inlineStr': // Value is rich text inline $value = $this->_parseRichText( $cell->is ); break; case 'e': // Value is an error message if ( (string) $cell->v !== '' ) { $value = (string) $cell->v; } break; case 'd': // Value is a date $value = $this->datetimeFormat ? gmdate( $this->datetimeFormat, $this->unixstamp( (float) $cell->v ) ) : (float) $cell->v; break; default: // Value is a string $value = (string) $cell->v; // Check for numeric values if ( is_numeric( $value ) && $dataType !== 's' ) { /** @noinspection TypeUnsafeComparisonInspection */ if ( $value == (int) $value ) { $value = (int) $value; } /** @noinspection TypeUnsafeComparisonInspection */ elseif ( $value == (float) $value ) { $value = (float) $value; } } } return $value; } public function unixstamp( $excelDateTime ) { $d = floor( $excelDateTime ); // days since 1900 or 1904 $t = $excelDateTime - $d; if ( $this->date1904 ) { /** @noinspection SummerTimeUnsafeTimeManipulationInspection */ $d += 1462; } /** @noinspection SummerTimeUnsafeTimeManipulationInspection */ $t = ( abs( $d ) > 0 ) ? ( $d - 25569 ) * 86400 + round( $t * 86400 ) : round( $t * 86400 ); return (int) $t; } /** * Returns cell value * VERY SLOW! Use ->rows() or ->rowsEx() * * @param int $worksheetIndex * @param string|array $cell ref or coords, D12 or [3,12] * * @return mixed Returns NULL if not found */ public function getCell( $worksheetIndex = 0, $cell = 'A1' ) { if (($ws = $this->worksheet( $worksheetIndex)) === false) { return false; } $idx = is_array( $cell ) ? $cell : $this->getIndex( (string) $cell ); $C = $idx[0]; $R = $idx[1]; $curR = 0; /* @var SimpleXMLElement $ws */ foreach ( $ws->sheetData->row as $row ) { $curC = 0; foreach ( $row->c as $c ) { // detect skipped cols $idx = $this->getIndex( (string) $c['r'] ); $x = $idx[0]; $y = $idx[1]; if ( $x > 0 ) { $curC = $x; $curR = $y; } if ( $curR === $R && $curC === $C ) { return $this->value( $c ); } if ( $curR > $R ){ return null; } $curC++; } $curR ++; } return null; } public function href( $cell ) { return isset( $this->hyperlinks[ (string) $cell['r'] ] ) ? $this->hyperlinks[ (string) $cell['r'] ] : ''; } public function sheets() { return $this->sheets; } public function sheetsCount() { return count( $this->sheets ); } public function sheetName( $worksheetIndex ) { if ( isset($this->sheetNames[ $worksheetIndex ])) { return $this->sheetNames[ $worksheetIndex ]; } return false; } public function sheetNames() { return $this->sheetNames; } // thx Gonzo public function getStyles() { return $this->styles; } public function getPackage() { return $this->package; } public function setDateTimeFormat( $value ) { $this->datetimeFormat = is_string( $value) ? $value : false; } private function _strlen( $str ) { return (ini_get('mbstring.func_overload') & 2) ? mb_strlen($str , '8bit') : strlen($str); } private function _strpos( $haystack, $needle, $offset = 0 ) { return (ini_get('mbstring.func_overload') & 2) ? mb_strpos( $haystack, $needle, $offset , '8bit') : strpos($haystack, $needle, $offset); } private function _strrpos( $haystack, $needle, $offset = 0 ) { return (ini_get('mbstring.func_overload') & 2) ? mb_strrpos( $haystack, $needle, $offset, '8bit') : strrpos($haystack, $needle, $offset); } private function _strtoupper( $str ) { return (ini_get('mbstring.func_overload') & 2) ? mb_strtoupper($str , '8bit') : strtoupper($str); } private function _substr( $str, $start, $length = null ) { return (ini_get('mbstring.func_overload') & 2) ? mb_substr( $str, $start, ($length === null) ? mb_strlen($str,'8bit') : $length, '8bit') : substr($str, $start, ($length === null) ? strlen($str) : $length ); } }
💾 Kaydet
İptal
📝 Yeniden Adlandır
İptal
Kaydet
🔐 Dosya İzinleri (chmod)
İzin Değeri:
Hızlı Seçim:
777
755
644
600
777
= Herkes okur/yazar/çalıştırır
755
= Sahip tam, diğerleri okur/çalıştırır
644
= Sahip okur/yazar, diğerleri okur
600
= Sadece sahip okur/yazar
İptal
Uygula