!C99Shell v. 2.5 [PHP 8 Update] [24.05.2025]!

Software: Apache/2.4.41 (Ubuntu). PHP/8.0.30 

uname -a: Linux apirnd 5.4.0-204-generic #224-Ubuntu SMP Thu Dec 5 13:38:28 UTC 2024 x86_64 

uid=33(www-data) gid=33(www-data) groups=33(www-data) 

Safe-mode: OFF (not secure)

/uploads/script/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/   drwxr-xr-x
Free 13.34 GB of 57.97 GB (23.02%)
Home    Back    Forward    UPDIR    Refresh    Search    Buffer    Encoder    Tools    Proc.    FTP brute    Sec.    SQL    PHP-code    Update    Self remove    Logout    


Viewing file:     ReferenceHelper.php (48.42 KB)      -rw-r--r--
Select action/file-type:
(+) | (+) | (+) | Code (+) | Session (+) | (+) | SDB (+) | (+) | (+) | (+) | (+) | (+) |
<?php

namespace PhpOffice\PhpSpreadsheet;

use 
PhpOffice\PhpSpreadsheet\Calculation\Calculation;
use 
PhpOffice\PhpSpreadsheet\Cell\Coordinate;
use 
PhpOffice\PhpSpreadsheet\Cell\DataType;
use 
PhpOffice\PhpSpreadsheet\Worksheet\Worksheet;

class 
ReferenceHelper
{
    
/**    Constants                */
    /**    Regular Expressions      */
    
const REFHELPER_REGEXP_CELLREF '((\w*|\'[^!]*\')!)?(?<![:a-z\$])(\$?[a-z]{1,3}\$?\d+)(?=[^:!\d\'])';
    const 
REFHELPER_REGEXP_CELLRANGE '((\w*|\'[^!]*\')!)?(\$?[a-z]{1,3}\$?\d+):(\$?[a-z]{1,3}\$?\d+)';
    const 
REFHELPER_REGEXP_ROWRANGE '((\w*|\'[^!]*\')!)?(\$?\d+):(\$?\d+)';
    const 
REFHELPER_REGEXP_COLRANGE '((\w*|\'[^!]*\')!)?(\$?[a-z]{1,3}):(\$?[a-z]{1,3})';

    
/**
     * Instance of this class.
     *
     * @var ReferenceHelper
     */
    
private static $instance;

    
/**
     * Get an instance of this class.
     *
     * @return ReferenceHelper
     */
    
public static function getInstance()
    {
        if (!isset(
self::$instance) || (self::$instance === null)) {
            
self::$instance = new self();
        }

        return 
self::$instance;
    }

    
/**
     * Create a new ReferenceHelper.
     */
    
protected function __construct()
    {
    }

    
/**
     * Compare two column addresses
     * Intended for use as a Callback function for sorting column addresses by column.
     *
     * @param string $a First column to test (e.g. 'AA')
     * @param string $b Second column to test (e.g. 'Z')
     *
     * @return int
     */
    
public static function columnSort($a$b)
    {
        return 
strcasecmp(strlen($a) . $astrlen($b) . $b);
    }

    
/**
     * Compare two column addresses
     * Intended for use as a Callback function for reverse sorting column addresses by column.
     *
     * @param string $a First column to test (e.g. 'AA')
     * @param string $b Second column to test (e.g. 'Z')
     *
     * @return int
     */
    
public static function columnReverseSort($a$b)
    {
        return -
strcasecmp(strlen($a) . $astrlen($b) . $b);
    }

    
/**
     * Compare two cell addresses
     * Intended for use as a Callback function for sorting cell addresses by column and row.
     *
     * @param string $a First cell to test (e.g. 'AA1')
     * @param string $b Second cell to test (e.g. 'Z1')
     *
     * @return int
     */
    
public static function cellSort($a$b)
    {
        [
$ac$ar] = sscanf($a'%[A-Z]%d');
        [
$bc$br] = sscanf($b'%[A-Z]%d');

        if (
$ar === $br) {
            return 
strcasecmp(strlen($ac) . $acstrlen($bc) . $bc);
        }

        return (
$ar $br) ? -1;
    }

    
/**
     * Compare two cell addresses
     * Intended for use as a Callback function for sorting cell addresses by column and row.
     *
     * @param string $a First cell to test (e.g. 'AA1')
     * @param string $b Second cell to test (e.g. 'Z1')
     *
     * @return int
     */
    
public static function cellReverseSort($a$b)
    {
        [
$ac$ar] = sscanf($a'%[A-Z]%d');
        [
$bc$br] = sscanf($b'%[A-Z]%d');

        if (
$ar === $br) {
            return -
strcasecmp(strlen($ac) . $acstrlen($bc) . $bc);
        }

        return (
$ar $br) ? : -1;
    }

    
/**
     * Test whether a cell address falls within a defined range of cells.
     *
     * @param string $cellAddress Address of the cell we're testing
     * @param int $beforeRow Number of the row we're inserting/deleting before
     * @param int $pNumRows Number of rows to insert/delete (negative values indicate deletion)
     * @param int $beforeColumnIndex Index number of the column we're inserting/deleting before
     * @param int $pNumCols Number of columns to insert/delete (negative values indicate deletion)
     *
     * @return bool
     */
    
private static function cellAddressInDeleteRange($cellAddress$beforeRow$pNumRows$beforeColumnIndex$pNumCols)
    {
        [
$cellColumn$cellRow] = Coordinate::coordinateFromString($cellAddress);
        
$cellColumnIndex Coordinate::columnIndexFromString($cellColumn);
        
//    Is cell within the range of rows/columns if we're deleting
        
if (
            
$pNumRows &&
            (
$cellRow >= ($beforeRow $pNumRows)) &&
            (
$cellRow $beforeRow)
        ) {
            return 
true;
        } elseif (
            
$pNumCols &&
            (
$cellColumnIndex >= ($beforeColumnIndex $pNumCols)) &&
            (
$cellColumnIndex $beforeColumnIndex)
        ) {
            return 
true;
        }

        return 
false;
    }

    
/**
     * Update page breaks when inserting/deleting rows/columns.
     *
     * @param Worksheet $pSheet The worksheet that we're editing
     * @param string $pBefore Insert/Delete before this cell address (e.g. 'A1')
     * @param int $beforeColumnIndex Index number of the column we're inserting/deleting before
     * @param int $pNumCols Number of columns to insert/delete (negative values indicate deletion)
     * @param int $beforeRow Number of the row we're inserting/deleting before
     * @param int $pNumRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustPageBreaks(Worksheet $pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows): void
    
{
        
$aBreaks $pSheet->getBreaks();
        (
$pNumCols || $pNumRows 0) ?
            
uksort($aBreaks, ['self''cellReverseSort']) : uksort($aBreaks, ['self''cellSort']);

        foreach (
$aBreaks as $key => $value) {
            if (
self::cellAddressInDeleteRange($key$beforeRow$pNumRows$beforeColumnIndex$pNumCols)) {
                
//    If we're deleting, then clear any defined breaks that are within the range
                //        of rows/columns that we're deleting
                
$pSheet->setBreak($keyWorksheet::BREAK_NONE);
            } else {
                
//    Otherwise update any affected breaks by inserting a new break at the appropriate point
                //        and removing the old affected break
                
$newReference $this->updateCellReference($key$pBefore$pNumCols$pNumRows);
                if (
$key != $newReference) {
                    
$pSheet->setBreak($newReference$value)
                        ->
setBreak($keyWorksheet::BREAK_NONE);
                }
            }
        }
    }

    
/**
     * Update cell comments when inserting/deleting rows/columns.
     *
     * @param Worksheet $pSheet The worksheet that we're editing
     * @param string $pBefore Insert/Delete before this cell address (e.g. 'A1')
     * @param int $beforeColumnIndex Index number of the column we're inserting/deleting before
     * @param int $pNumCols Number of columns to insert/delete (negative values indicate deletion)
     * @param int $beforeRow Number of the row we're inserting/deleting before
     * @param int $pNumRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustComments($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows): void
    
{
        
$aComments $pSheet->getComments();
        
$aNewComments = []; // the new array of all comments

        
foreach ($aComments as $key => &$value) {
            
// Any comments inside a deleted range will be ignored
            
if (!self::cellAddressInDeleteRange($key$beforeRow$pNumRows$beforeColumnIndex$pNumCols)) {
                
// Otherwise build a new array of comments indexed by the adjusted cell reference
                
$newReference $this->updateCellReference($key$pBefore$pNumCols$pNumRows);
                
$aNewComments[$newReference] = $value;
            }
        }
        
//    Replace the comments array with the new set of comments
        
$pSheet->setComments($aNewComments);
    }

    
/**
     * Update hyperlinks when inserting/deleting rows/columns.
     *
     * @param Worksheet $pSheet The worksheet that we're editing
     * @param string $pBefore Insert/Delete before this cell address (e.g. 'A1')
     * @param int $beforeColumnIndex Index number of the column we're inserting/deleting before
     * @param int $pNumCols Number of columns to insert/delete (negative values indicate deletion)
     * @param int $beforeRow Number of the row we're inserting/deleting before
     * @param int $pNumRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustHyperlinks($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows): void
    
{
        
$aHyperlinkCollection $pSheet->getHyperlinkCollection();
        (
$pNumCols || $pNumRows 0) ?
            
uksort($aHyperlinkCollection, ['self''cellReverseSort']) : uksort($aHyperlinkCollection, ['self''cellSort']);

        foreach (
$aHyperlinkCollection as $key => $value) {
            
$newReference $this->updateCellReference($key$pBefore$pNumCols$pNumRows);
            if (
$key != $newReference) {
                
$pSheet->setHyperlink($newReference$value);
                
$pSheet->setHyperlink($keynull);
            }
        }
    }

    
/**
     * Update data validations when inserting/deleting rows/columns.
     *
     * @param Worksheet $pSheet The worksheet that we're editing
     * @param string $pBefore Insert/Delete before this cell address (e.g. 'A1')
     * @param int $beforeColumnIndex Index number of the column we're inserting/deleting before
     * @param int $pNumCols Number of columns to insert/delete (negative values indicate deletion)
     * @param int $beforeRow Number of the row we're inserting/deleting before
     * @param int $pNumRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustDataValidations($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows): void
    
{
        
$aDataValidationCollection $pSheet->getDataValidationCollection();
        (
$pNumCols || $pNumRows 0) ?
            
uksort($aDataValidationCollection, ['self''cellReverseSort']) : uksort($aDataValidationCollection, ['self''cellSort']);

        foreach (
$aDataValidationCollection as $key => $value) {
            
$newReference $this->updateCellReference($key$pBefore$pNumCols$pNumRows);
            if (
$key != $newReference) {
                
$pSheet->setDataValidation($newReference$value);
                
$pSheet->setDataValidation($keynull);
            }
        }
    }

    
/**
     * Update merged cells when inserting/deleting rows/columns.
     *
     * @param Worksheet $pSheet The worksheet that we're editing
     * @param string $pBefore Insert/Delete before this cell address (e.g. 'A1')
     * @param int $beforeColumnIndex Index number of the column we're inserting/deleting before
     * @param int $pNumCols Number of columns to insert/delete (negative values indicate deletion)
     * @param int $beforeRow Number of the row we're inserting/deleting before
     * @param int $pNumRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustMergeCells($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows): void
    
{
        
$aMergeCells $pSheet->getMergeCells();
        
$aNewMergeCells = []; // the new array of all merge cells
        
foreach ($aMergeCells as $key => &$value) {
            
$newReference $this->updateCellReference($key$pBefore$pNumCols$pNumRows);
            
$aNewMergeCells[$newReference] = $newReference;
        }
        
$pSheet->setMergeCells($aNewMergeCells); // replace the merge cells array
    
}

    
/**
     * Update protected cells when inserting/deleting rows/columns.
     *
     * @param Worksheet $pSheet The worksheet that we're editing
     * @param string $pBefore Insert/Delete before this cell address (e.g. 'A1')
     * @param int $beforeColumnIndex Index number of the column we're inserting/deleting before
     * @param int $pNumCols Number of columns to insert/delete (negative values indicate deletion)
     * @param int $beforeRow Number of the row we're inserting/deleting before
     * @param int $pNumRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustProtectedCells($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows): void
    
{
        
$aProtectedCells $pSheet->getProtectedCells();
        (
$pNumCols || $pNumRows 0) ?
            
uksort($aProtectedCells, ['self''cellReverseSort']) : uksort($aProtectedCells, ['self''cellSort']);
        foreach (
$aProtectedCells as $key => $value) {
            
$newReference $this->updateCellReference($key$pBefore$pNumCols$pNumRows);
            if (
$key != $newReference) {
                
$pSheet->protectCells($newReference$valuetrue);
                
$pSheet->unprotectCells($key);
            }
        }
    }

    
/**
     * Update column dimensions when inserting/deleting rows/columns.
     *
     * @param Worksheet $pSheet The worksheet that we're editing
     * @param string $pBefore Insert/Delete before this cell address (e.g. 'A1')
     * @param int $beforeColumnIndex Index number of the column we're inserting/deleting before
     * @param int $pNumCols Number of columns to insert/delete (negative values indicate deletion)
     * @param int $beforeRow Number of the row we're inserting/deleting before
     * @param int $pNumRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustColumnDimensions($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows): void
    
{
        
$aColumnDimensions array_reverse($pSheet->getColumnDimensions(), true);
        if (!empty(
$aColumnDimensions)) {
            foreach (
$aColumnDimensions as $objColumnDimension) {
                
$newReference $this->updateCellReference($objColumnDimension->getColumnIndex() . '1'$pBefore$pNumCols$pNumRows);
                [
$newReference] = Coordinate::coordinateFromString($newReference);
                if (
$objColumnDimension->getColumnIndex() != $newReference) {
                    
$objColumnDimension->setColumnIndex($newReference);
                }
            }
            
$pSheet->refreshColumnDimensions();
        }
    }

    
/**
     * Update row dimensions when inserting/deleting rows/columns.
     *
     * @param Worksheet $pSheet The worksheet that we're editing
     * @param string $pBefore Insert/Delete before this cell address (e.g. 'A1')
     * @param int $beforeColumnIndex Index number of the column we're inserting/deleting before
     * @param int $pNumCols Number of columns to insert/delete (negative values indicate deletion)
     * @param int $beforeRow Number of the row we're inserting/deleting before
     * @param int $pNumRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustRowDimensions($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows): void
    
{
        
$aRowDimensions array_reverse($pSheet->getRowDimensions(), true);
        if (!empty(
$aRowDimensions)) {
            foreach (
$aRowDimensions as $objRowDimension) {
                
$newReference $this->updateCellReference('A' $objRowDimension->getRowIndex(), $pBefore$pNumCols$pNumRows);
                [, 
$newReference] = Coordinate::coordinateFromString($newReference);
                if (
$objRowDimension->getRowIndex() != $newReference) {
                    
$objRowDimension->setRowIndex($newReference);
                }
            }
            
$pSheet->refreshRowDimensions();

            
$copyDimension $pSheet->getRowDimension($beforeRow 1);
            for (
$i $beforeRow$i <= $beforeRow $pNumRows; ++$i) {
                
$newDimension $pSheet->getRowDimension($i);
                
$newDimension->setRowHeight($copyDimension->getRowHeight());
                
$newDimension->setVisible($copyDimension->getVisible());
                
$newDimension->setOutlineLevel($copyDimension->getOutlineLevel());
                
$newDimension->setCollapsed($copyDimension->getCollapsed());
            }
        }
    }

    
/**
     * Insert a new column or row, updating all possible related data.
     *
     * @param string $pBefore Insert before this cell address (e.g. 'A1')
     * @param int $pNumCols Number of columns to insert/delete (negative values indicate deletion)
     * @param int $pNumRows Number of rows to insert/delete (negative values indicate deletion)
     * @param Worksheet $pSheet The worksheet that we're editing
     */
    
public function insertNewBefore($pBefore$pNumCols$pNumRowsWorksheet $pSheet): void
    
{
        
$remove = ($pNumCols || $pNumRows 0);
        
$allCoordinates $pSheet->getCoordinates();

        
// Get coordinate of $pBefore
        
[$beforeColumn$beforeRow] = Coordinate::coordinateFromString($pBefore);
        
$beforeColumnIndex Coordinate::columnIndexFromString($beforeColumn);

        
// Clear cells if we are removing columns or rows
        
$highestColumn $pSheet->getHighestColumn();
        
$highestRow $pSheet->getHighestRow();

        
// 1. Clear column strips if we are removing columns
        
if ($pNumCols && $beforeColumnIndex $pNumCols 0) {
            for (
$i 1$i <= $highestRow 1; ++$i) {
                for (
$j $beforeColumnIndex $pNumCols$j <= $beforeColumnIndex 2; ++$j) {
                    
$coordinate Coordinate::stringFromColumnIndex($j 1) . $i;
                    
$pSheet->removeConditionalStyles($coordinate);
                    if (
$pSheet->cellExists($coordinate)) {
                        
$pSheet->getCell($coordinate)->setValueExplicit(''DataType::TYPE_NULL);
                        
$pSheet->getCell($coordinate)->setXfIndex(0);
                    }
                }
            }
        }

        
// 2. Clear row strips if we are removing rows
        
if ($pNumRows && $beforeRow $pNumRows 0) {
            for (
$i $beforeColumnIndex 1$i <= Coordinate::columnIndexFromString($highestColumn) - 1; ++$i) {
                for (
$j $beforeRow $pNumRows$j <= $beforeRow 1; ++$j) {
                    
$coordinate Coordinate::stringFromColumnIndex($i 1) . $j;
                    
$pSheet->removeConditionalStyles($coordinate);
                    if (
$pSheet->cellExists($coordinate)) {
                        
$pSheet->getCell($coordinate)->setValueExplicit(''DataType::TYPE_NULL);
                        
$pSheet->getCell($coordinate)->setXfIndex(0);
                    }
                }
            }
        }

        
// Loop through cells, bottom-up, and change cell coordinate
        
if ($remove) {
            
// It's faster to reverse and pop than to use unshift, especially with large cell collections
            
$allCoordinates array_reverse($allCoordinates);
        }
        while (
$coordinate array_pop($allCoordinates)) {
            
$cell $pSheet->getCell($coordinate);
            
$cellIndex Coordinate::columnIndexFromString($cell->getColumn());

            if (
$cellIndex $pNumCols 0) {
                continue;
            }

            
// New coordinate
            
$newCoordinate Coordinate::stringFromColumnIndex($cellIndex $pNumCols) . ($cell->getRow() + $pNumRows);

            
// Should the cell be updated? Move value and cellXf index from one cell to another.
            
if (($cellIndex >= $beforeColumnIndex) && ($cell->getRow() >= $beforeRow)) {
                
// Update cell styles
                
$pSheet->getCell($newCoordinate)->setXfIndex($cell->getXfIndex());

                
// Insert this cell at its new location
                
if ($cell->getDataType() == DataType::TYPE_FORMULA) {
                    
// Formula should be adjusted
                    
$pSheet->getCell($newCoordinate)
                        ->
setValue($this->updateFormulaReferences($cell->getValue(), $pBefore$pNumCols$pNumRows$pSheet->getTitle()));
                } else {
                    
// Formula should not be adjusted
                    
$pSheet->getCell($newCoordinate)->setValue($cell->getValue());
                }

                
// Clear the original cell
                
$pSheet->getCellCollection()->delete($coordinate);
            } else {
                
/*    We don't need to update styles for rows/columns before our insertion position,
                        but we do still need to adjust any formulae    in those cells                    */
                
if ($cell->getDataType() == DataType::TYPE_FORMULA) {
                    
// Formula should be adjusted
                    
$cell->setValue($this->updateFormulaReferences($cell->getValue(), $pBefore$pNumCols$pNumRows$pSheet->getTitle()));
                }
            }
        }

        
// Duplicate styles for the newly inserted cells
        
$highestColumn $pSheet->getHighestColumn();
        
$highestRow $pSheet->getHighestRow();

        if (
$pNumCols && $beforeColumnIndex 0) {
            for (
$i $beforeRow$i <= $highestRow 1; ++$i) {
                
// Style
                
$coordinate Coordinate::stringFromColumnIndex($beforeColumnIndex 1) . $i;
                if (
$pSheet->cellExists($coordinate)) {
                    
$xfIndex $pSheet->getCell($coordinate)->getXfIndex();
                    
$conditionalStyles $pSheet->conditionalStylesExists($coordinate) ?
                        
$pSheet->getConditionalStyles($coordinate) : false;
                    for (
$j $beforeColumnIndex$j <= $beforeColumnIndex $pNumCols; ++$j) {
                        
$pSheet->getCellByColumnAndRow($j$i)->setXfIndex($xfIndex);
                        if (
$conditionalStyles) {
                            
$cloned = [];
                            foreach (
$conditionalStyles as $conditionalStyle) {
                                
$cloned[] = clone $conditionalStyle;
                            }
                            
$pSheet->setConditionalStyles(Coordinate::stringFromColumnIndex($j) . $i$cloned);
                        }
                    }
                }
            }
        }

        if (
$pNumRows && $beforeRow 0) {
            for (
$i $beforeColumnIndex$i <= Coordinate::columnIndexFromString($highestColumn); ++$i) {
                
// Style
                
$coordinate Coordinate::stringFromColumnIndex($i) . ($beforeRow 1);
                if (
$pSheet->cellExists($coordinate)) {
                    
$xfIndex $pSheet->getCell($coordinate)->getXfIndex();
                    
$conditionalStyles $pSheet->conditionalStylesExists($coordinate) ?
                        
$pSheet->getConditionalStyles($coordinate) : false;
                    for (
$j $beforeRow$j <= $beforeRow $pNumRows; ++$j) {
                        
$pSheet->getCell(Coordinate::stringFromColumnIndex($i) . $j)->setXfIndex($xfIndex);
                        if (
$conditionalStyles) {
                            
$cloned = [];
                            foreach (
$conditionalStyles as $conditionalStyle) {
                                
$cloned[] = clone $conditionalStyle;
                            }
                            
$pSheet->setConditionalStyles(Coordinate::stringFromColumnIndex($i) . $j$cloned);
                        }
                    }
                }
            }
        }

        
// Update worksheet: column dimensions
        
$this->adjustColumnDimensions($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows);

        
// Update worksheet: row dimensions
        
$this->adjustRowDimensions($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows);

        
//    Update worksheet: page breaks
        
$this->adjustPageBreaks($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows);

        
//    Update worksheet: comments
        
$this->adjustComments($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows);

        
// Update worksheet: hyperlinks
        
$this->adjustHyperlinks($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows);

        
// Update worksheet: data validations
        
$this->adjustDataValidations($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows);

        
// Update worksheet: merge cells
        
$this->adjustMergeCells($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows);

        
// Update worksheet: protected cells
        
$this->adjustProtectedCells($pSheet$pBefore$beforeColumnIndex$pNumCols$beforeRow$pNumRows);

        
// Update worksheet: autofilter
        
$autoFilter $pSheet->getAutoFilter();
        
$autoFilterRange $autoFilter->getRange();
        if (!empty(
$autoFilterRange)) {
            if (
$pNumCols != 0) {
                
$autoFilterColumns $autoFilter->getColumns();
                if (
count($autoFilterColumns) > 0) {
                    
$column '';
                    
$row 0;
                    
sscanf($pBefore'%[A-Z]%d'$column$row);
                    
$columnIndex Coordinate::columnIndexFromString($column);
                    [
$rangeStart$rangeEnd] = Coordinate::rangeBoundaries($autoFilterRange);
                    if (
$columnIndex <= $rangeEnd[0]) {
                        if (
$pNumCols 0) {
                            
//    If we're actually deleting any columns that fall within the autofilter range,
                            //        then we delete any rules for those columns
                            
$deleteColumn $columnIndex $pNumCols 1;
                            
$deleteCount abs($pNumCols);
                            for (
$i 1$i <= $deleteCount; ++$i) {
                                if (isset(
$autoFilterColumns[Coordinate::stringFromColumnIndex($deleteColumn 1)])) {
                                    
$autoFilter->clearColumn(Coordinate::stringFromColumnIndex($deleteColumn 1));
                                }
                                ++
$deleteColumn;
                            }
                        }
                        
$startCol = ($columnIndex $rangeStart[0]) ? $columnIndex $rangeStart[0];

                        
//    Shuffle columns in autofilter range
                        
if ($pNumCols 0) {
                            
$startColRef $startCol;
                            
$endColRef $rangeEnd[0];
                            
$toColRef $rangeEnd[0] + $pNumCols;

                            do {
                                
$autoFilter->shiftColumn(Coordinate::stringFromColumnIndex($endColRef), Coordinate::stringFromColumnIndex($toColRef));
                                --
$endColRef;
                                --
$toColRef;
                            } while (
$startColRef <= $endColRef);
                        } else {
                            
//    For delete, we shuffle from beginning to end to avoid overwriting
                            
$startColID Coordinate::stringFromColumnIndex($startCol);
                            
$toColID Coordinate::stringFromColumnIndex($startCol $pNumCols);
                            
$endColID Coordinate::stringFromColumnIndex($rangeEnd[0] + 1);
                            do {
                                
$autoFilter->shiftColumn($startColID$toColID);
                                ++
$startColID;
                                ++
$toColID;
                            } while (
$startColID != $endColID);
                        }
                    }
                }
            }
            
$pSheet->setAutoFilter($this->updateCellReference($autoFilterRange$pBefore$pNumCols$pNumRows));
        }

        
// Update worksheet: freeze pane
        
if ($pSheet->getFreezePane()) {
            
$splitCell $pSheet->getFreezePane();
            
$topLeftCell $pSheet->getTopLeftCell();

            
$splitCell $this->updateCellReference($splitCell$pBefore$pNumCols$pNumRows);
            
$topLeftCell $this->updateCellReference($topLeftCell$pBefore$pNumCols$pNumRows);

            
$pSheet->freezePane($splitCell$topLeftCell);
        }

        
// Page setup
        
if ($pSheet->getPageSetup()->isPrintAreaSet()) {
            
$pSheet->getPageSetup()->setPrintArea($this->updateCellReference($pSheet->getPageSetup()->getPrintArea(), $pBefore$pNumCols$pNumRows));
        }

        
// Update worksheet: drawings
        
$aDrawings $pSheet->getDrawingCollection();
        foreach (
$aDrawings as $objDrawing) {
            
$newReference $this->updateCellReference($objDrawing->getCoordinates(), $pBefore$pNumCols$pNumRows);
            if (
$objDrawing->getCoordinates() != $newReference) {
                
$objDrawing->setCoordinates($newReference);
            }
        }

        
// Update workbook: define names
        
if (count($pSheet->getParent()->getDefinedNames()) > 0) {
            foreach (
$pSheet->getParent()->getDefinedNames() as $definedName) {
                if (
$definedName->getWorksheet()->getHashCode() === $pSheet->getHashCode()) {
                    
$definedName->setValue($this->updateCellReference($definedName->getValue(), $pBefore$pNumCols$pNumRows));
                }
            }
        }

        
// Garbage collect
        
$pSheet->garbageCollect();
    }

    
/**
     * Update references within formulas.
     *
     * @param string $pFormula Formula to update
     * @param string $pBefore Insert before this one
     * @param int $pNumCols Number of columns to insert
     * @param int $pNumRows Number of rows to insert
     * @param string $sheetName Worksheet name/title
     *
     * @return string Updated formula
     */
    
public function updateFormulaReferences($pFormula ''$pBefore 'A1'$pNumCols 0$pNumRows 0$sheetName '')
    {
        
//    Update cell references in the formula
        
$formulaBlocks explode('"'$pFormula);
        
$i false;
        foreach (
$formulaBlocks as &$formulaBlock) {
            
//    Ignore blocks that were enclosed in quotes (alternating entries in the $formulaBlocks array after the explode)
            
if ($i = !$i) {
                
$adjustCount 0;
                
$newCellTokens $cellTokens = [];
                
//    Search for row ranges (e.g. 'Sheet1'!3:5 or 3:5) with or without $ absolutes (e.g. $3:5)
                
$matchCount preg_match_all('/' self::REFHELPER_REGEXP_ROWRANGE '/i'' ' $formulaBlock ' '$matchesPREG_SET_ORDER);
                if (
$matchCount 0) {
                    foreach (
$matches as $match) {
                        
$fromString = ($match[2] > '') ? $match[2] . '!' '';
                        
$fromString .= $match[3] . ':' $match[4];
                        
$modified3 substr($this->updateCellReference('$A' $match[3], $pBefore$pNumCols$pNumRows), 2);
                        
$modified4 substr($this->updateCellReference('$A' $match[4], $pBefore$pNumCols$pNumRows), 2);

                        if (
$match[3] . ':' $match[4] !== $modified3 ':' $modified4) {
                            if ((
$match[2] == '') || (trim($match[2], "'") == $sheetName)) {
                                
$toString = ($match[2] > '') ? $match[2] . '!' '';
                                
$toString .= $modified3 ':' $modified4;
                                
//    Max worksheet size is 1,048,576 rows by 16,384 columns in Excel 2007, so our adjustments need to be at least one digit more
                                
$column 100000;
                                
$row 10000000 trim($match[3], '$');
                                
$cellIndex $column $row;

                                
$newCellTokens[$cellIndex] = preg_quote($toString'/');
                                
$cellTokens[$cellIndex] = '/(?<!\d\$\!)' preg_quote($fromString'/') . '(?!\d)/i';
                                ++
$adjustCount;
                            }
                        }
                    }
                }
                
//    Search for column ranges (e.g. 'Sheet1'!C:E or C:E) with or without $ absolutes (e.g. $C:E)
                
$matchCount preg_match_all('/' self::REFHELPER_REGEXP_COLRANGE '/i'' ' $formulaBlock ' '$matchesPREG_SET_ORDER);
                if (
$matchCount 0) {
                    foreach (
$matches as $match) {
                        
$fromString = ($match[2] > '') ? $match[2] . '!' '';
                        
$fromString .= $match[3] . ':' $match[4];
                        
$modified3 substr($this->updateCellReference($match[3] . '$1'$pBefore$pNumCols$pNumRows), 0, -2);
                        
$modified4 substr($this->updateCellReference($match[4] . '$1'$pBefore$pNumCols$pNumRows), 0, -2);

                        if (
$match[3] . ':' $match[4] !== $modified3 ':' $modified4) {
                            if ((
$match[2] == '') || (trim($match[2], "'") == $sheetName)) {
                                
$toString = ($match[2] > '') ? $match[2] . '!' '';
                                
$toString .= $modified3 ':' $modified4;
                                
//    Max worksheet size is 1,048,576 rows by 16,384 columns in Excel 2007, so our adjustments need to be at least one digit more
                                
$column Coordinate::columnIndexFromString(trim($match[3], '$')) + 100000;
                                
$row 10000000;
                                
$cellIndex $column $row;

                                
$newCellTokens[$cellIndex] = preg_quote($toString'/');
                                
$cellTokens[$cellIndex] = '/(?<![A-Z\$\!])' preg_quote($fromString'/') . '(?![A-Z])/i';
                                ++
$adjustCount;
                            }
                        }
                    }
                }
                
//    Search for cell ranges (e.g. 'Sheet1'!A3:C5 or A3:C5) with or without $ absolutes (e.g. $A1:C$5)
                
$matchCount preg_match_all('/' self::REFHELPER_REGEXP_CELLRANGE '/i'' ' $formulaBlock ' '$matchesPREG_SET_ORDER);
                if (
$matchCount 0) {
                    foreach (
$matches as $match) {
                        
$fromString = ($match[2] > '') ? $match[2] . '!' '';
                        
$fromString .= $match[3] . ':' $match[4];
                        
$modified3 $this->updateCellReference($match[3], $pBefore$pNumCols$pNumRows);
                        
$modified4 $this->updateCellReference($match[4], $pBefore$pNumCols$pNumRows);

                        if (
$match[3] . $match[4] !== $modified3 $modified4) {
                            if ((
$match[2] == '') || (trim($match[2], "'") == $sheetName)) {
                                
$toString = ($match[2] > '') ? $match[2] . '!' '';
                                
$toString .= $modified3 ':' $modified4;
                                [
$column$row] = Coordinate::coordinateFromString($match[3]);
                                
//    Max worksheet size is 1,048,576 rows by 16,384 columns in Excel 2007, so our adjustments need to be at least one digit more
                                
$column Coordinate::columnIndexFromString(trim($column'$')) + 100000;
                                
$row trim($row'$') + 10000000;
                                
$cellIndex $column $row;

                                
$newCellTokens[$cellIndex] = preg_quote($toString'/');
                                
$cellTokens[$cellIndex] = '/(?<![A-Z]\$\!)' preg_quote($fromString'/') . '(?!\d)/i';
                                ++
$adjustCount;
                            }
                        }
                    }
                }
                
//    Search for cell references (e.g. 'Sheet1'!A3 or C5) with or without $ absolutes (e.g. $A1 or C$5)
                
$matchCount preg_match_all('/' self::REFHELPER_REGEXP_CELLREF '/i'' ' $formulaBlock ' '$matchesPREG_SET_ORDER);

                if (
$matchCount 0) {
                    foreach (
$matches as $match) {
                        
$fromString = ($match[2] > '') ? $match[2] . '!' '';
                        
$fromString .= $match[3];

                        
$modified3 $this->updateCellReference($match[3], $pBefore$pNumCols$pNumRows);
                        if (
$match[3] !== $modified3) {
                            if ((
$match[2] == '') || (trim($match[2], "'") == $sheetName)) {
                                
$toString = ($match[2] > '') ? $match[2] . '!' '';
                                
$toString .= $modified3;
                                [
$column$row] = Coordinate::coordinateFromString($match[3]);
                                
//    Max worksheet size is 1,048,576 rows by 16,384 columns in Excel 2007, so our adjustments need to be at least one digit more
                                
$column Coordinate::columnIndexFromString(trim($column'$')) + 100000;
                                
$row trim($row'$') + 10000000;
                                
$cellIndex $row $column;

                                
$newCellTokens[$cellIndex] = preg_quote($toString'/');
                                
$cellTokens[$cellIndex] = '/(?<![A-Z\$\!])' preg_quote($fromString'/') . '(?!\d)/i';
                                ++
$adjustCount;
                            }
                        }
                    }
                }
                if (
$adjustCount 0) {
                    if (
$pNumCols || $pNumRows 0) {
                        
krsort($cellTokens);
                        
krsort($newCellTokens);
                    } else {
                        
ksort($cellTokens);
                        
ksort($newCellTokens);
                    }   
//  Update cell references in the formula
                    
$formulaBlock str_replace('\\'''preg_replace($cellTokens$newCellTokens$formulaBlock));
                }
            }
        }
        unset(
$formulaBlock);

        
//    Then rebuild the formula string
        
return implode('"'$formulaBlocks);
    }

    
/**
     * Update all cell references within a formula, irrespective of worksheet.
     */
    
public function updateFormulaReferencesAnyWorksheet(string $formula ''int $insertColumns 0int $insertRows 0): string
    
{
        
$formula $this->updateCellReferencesAllWorksheets($formula$insertColumns$insertRows);

        if (
$insertColumns !== 0) {
            
$formula $this->updateColumnRangesAllWorksheets($formula$insertColumns);
        }

        if (
$insertRows !== 0) {
            
$formula $this->updateRowRangesAllWorksheets($formula$insertRows);
        }

        return 
$formula;
    }

    private function 
updateCellReferencesAllWorksheets(string $formulaint $insertColumnsint $insertRows): string
    
{
        
$splitCount preg_match_all(
            
'/' Calculation::CALCULATION_REGEXP_CELLREF_RELATIVE '/mui',
            
$formula,
            
$splitRanges,
            
PREG_OFFSET_CAPTURE
        
);

        
$columnLengths array_map('strlen'array_column($splitRanges[6], 0));
        
$rowLengths array_map('strlen'array_column($splitRanges[7], 0));
        
$columnOffsets array_column($splitRanges[6], 1);
        
$rowOffsets array_column($splitRanges[7], 1);

        
$columns $splitRanges[6];
        
$rows $splitRanges[7];

        while (
$splitCount 0) {
            --
$splitCount;
            
$columnLength $columnLengths[$splitCount];
            
$rowLength $rowLengths[$splitCount];
            
$columnOffset $columnOffsets[$splitCount];
            
$rowOffset $rowOffsets[$splitCount];
            
$column $columns[$splitCount][0];
            
$row $rows[$splitCount][0];

            if (!empty(
$column) && $column[0] !== '$') {
                
$column Coordinate::stringFromColumnIndex(Coordinate::columnIndexFromString($column) + $insertColumns);
                
$formula substr($formula0$columnOffset) . $column substr($formula$columnOffset $columnLength);
            }
            if (!empty(
$row) && $row[0] !== '$') {
                
$row += $insertRows;
                
$formula substr($formula0$rowOffset) . $row substr($formula$rowOffset $rowLength);
            }
        }

        return 
$formula;
    }

    private function 
updateColumnRangesAllWorksheets(string $formulaint $insertColumns): string
    
{
        
$splitCount preg_match_all(
            
'/' Calculation::CALCULATION_REGEXP_COLUMNRANGE_RELATIVE '/mui',
            
$formula,
            
$splitRanges,
            
PREG_OFFSET_CAPTURE
        
);

        
$fromColumnLengths array_map('strlen'array_column($splitRanges[1], 0));
        
$fromColumnOffsets array_column($splitRanges[1], 1);
        
$toColumnLengths array_map('strlen'array_column($splitRanges[2], 0));
        
$toColumnOffsets array_column($splitRanges[2], 1);

        
$fromColumns $splitRanges[1];
        
$toColumns $splitRanges[2];

        while (
$splitCount 0) {
            --
$splitCount;
            
$fromColumnLength $fromColumnLengths[$splitCount];
            
$toColumnLength $toColumnLengths[$splitCount];
            
$fromColumnOffset $fromColumnOffsets[$splitCount];
            
$toColumnOffset $toColumnOffsets[$splitCount];
            
$fromColumn $fromColumns[$splitCount][0];
            
$toColumn $toColumns[$splitCount][0];

            if (!empty(
$fromColumn) && $fromColumn[0] !== '$') {
                
$fromColumn Coordinate::stringFromColumnIndex(Coordinate::columnIndexFromString($fromColumn) + $insertColumns);
                
$formula substr($formula0$fromColumnOffset) . $fromColumn substr($formula$fromColumnOffset $fromColumnLength);
            }
            if (!empty(
$toColumn) && $toColumn[0] !== '$') {
                
$toColumn Coordinate::stringFromColumnIndex(Coordinate::columnIndexFromString($toColumn) + $insertColumns);
                
$formula substr($formula0$toColumnOffset) . $toColumn substr($formula$toColumnOffset $toColumnLength);
            }
        }

        return 
$formula;
    }

    private function 
updateRowRangesAllWorksheets(string $formulaint $insertRows): string
    
{
        
$splitCount preg_match_all(
            
'/' Calculation::CALCULATION_REGEXP_ROWRANGE_RELATIVE '/mui',
            
$formula,
            
$splitRanges,
            
PREG_OFFSET_CAPTURE
        
);

        
$fromRowLengths array_map('strlen'array_column($splitRanges[1], 0));
        
$fromRowOffsets array_column($splitRanges[1], 1);
        
$toRowLengths array_map('strlen'array_column($splitRanges[2], 0));
        
$toRowOffsets array_column($splitRanges[2], 1);

        
$fromRows $splitRanges[1];
        
$toRows $splitRanges[2];

        while (
$splitCount 0) {
            --
$splitCount;
            
$fromRowLength $fromRowLengths[$splitCount];
            
$toRowLength $toRowLengths[$splitCount];
            
$fromRowOffset $fromRowOffsets[$splitCount];
            
$toRowOffset $toRowOffsets[$splitCount];
            
$fromRow $fromRows[$splitCount][0];
            
$toRow $toRows[$splitCount][0];

            if (!empty(
$fromRow) && $fromRow[0] !== '$') {
                
$fromRow += $insertRows;
                
$formula substr($formula0$fromRowOffset) . $fromRow substr($formula$fromRowOffset $fromRowLength);
            }
            if (!empty(
$toRow) && $toRow[0] !== '$') {
                
$toRow += $insertRows;
                
$formula substr($formula0$toRowOffset) . $toRow substr($formula$toRowOffset $toRowLength);
            }
        }

        return 
$formula;
    }

    
/**
     * Update cell reference.
     *
     * @param string $pCellRange Cell range
     * @param string $pBefore Insert before this one
     * @param int $pNumCols Number of columns to increment
     * @param int $pNumRows Number of rows to increment
     *
     * @return string Updated cell range
     */
    
public function updateCellReference($pCellRange 'A1'$pBefore 'A1'$pNumCols 0$pNumRows 0)
    {
        
// Is it in another worksheet? Will not have to update anything.
        
if (strpos($pCellRange'!') !== false) {
            return 
$pCellRange;
        
// Is it a range or a single cell?
        
} elseif (!Coordinate::coordinateIsRange($pCellRange)) {
            
// Single cell
            
return $this->updateSingleCellReference($pCellRange$pBefore$pNumCols$pNumRows);
        } elseif (
Coordinate::coordinateIsRange($pCellRange)) {
            
// Range
            
return $this->updateCellRange($pCellRange$pBefore$pNumCols$pNumRows);
        }

        
// Return original
        
return $pCellRange;
    }

    
/**
     * Update named formulas (i.e. containing worksheet references / named ranges).
     *
     * @param Spreadsheet $spreadsheet Object to update
     * @param string $oldName Old name (name to replace)
     * @param string $newName New name
     */
    
public function updateNamedFormulas(Spreadsheet $spreadsheet$oldName ''$newName ''): void
    
{
        if (
$oldName == '') {
            return;
        }

        foreach (
$spreadsheet->getWorksheetIterator() as $sheet) {
            foreach (
$sheet->getCoordinates(false) as $coordinate) {
                
$cell $sheet->getCell($coordinate);
                if ((
$cell !== null) && ($cell->getDataType() == DataType::TYPE_FORMULA)) {
                    
$formula $cell->getValue();
                    if (
strpos($formula$oldName) !== false) {
                        
$formula str_replace("'" $oldName "'!""'" $newName "'!"$formula);
                        
$formula str_replace($oldName '!'$newName '!'$formula);
                        
$cell->setValueExplicit($formulaDataType::TYPE_FORMULA);
                    }
                }
            }
        }
    }

    
/**
     * Update cell range.
     *
     * @param string $pCellRange Cell range    (e.g. 'B2:D4', 'B:C' or '2:3')
     * @param string $pBefore Insert before this one
     * @param int $pNumCols Number of columns to increment
     * @param int $pNumRows Number of rows to increment
     *
     * @return string Updated cell range
     */
    
private function updateCellRange($pCellRange 'A1:A1'$pBefore 'A1'$pNumCols 0$pNumRows 0)
    {
        if (!
Coordinate::coordinateIsRange($pCellRange)) {
            throw new 
Exception('Only cell ranges may be passed to this method.');
        }

        
// Update range
        
$range Coordinate::splitRange($pCellRange);
        
$ic count($range);
        for (
$i 0$i $ic; ++$i) {
            
$jc count($range[$i]);
            for (
$j 0$j $jc; ++$j) {
                if (
ctype_alpha($range[$i][$j])) {
                    
$r Coordinate::coordinateFromString($this->updateSingleCellReference($range[$i][$j] . '1'$pBefore$pNumCols$pNumRows));
                    
$range[$i][$j] = $r[0];
                } elseif (
ctype_digit($range[$i][$j])) {
                    
$r Coordinate::coordinateFromString($this->updateSingleCellReference('A' $range[$i][$j], $pBefore$pNumCols$pNumRows));
                    
$range[$i][$j] = $r[1];
                } else {
                    
$range[$i][$j] = $this->updateSingleCellReference($range[$i][$j], $pBefore$pNumCols$pNumRows);
                }
            }
        }

        
// Recreate range string
        
return Coordinate::buildRange($range);
    }

    
/**
     * Update single cell reference.
     *
     * @param string $pCellReference Single cell reference
     * @param string $pBefore Insert before this one
     * @param int $pNumCols Number of columns to increment
     * @param int $pNumRows Number of rows to increment
     *
     * @return string Updated cell reference
     */
    
private function updateSingleCellReference($pCellReference 'A1'$pBefore 'A1'$pNumCols 0$pNumRows 0)
    {
        if (
Coordinate::coordinateIsRange($pCellReference)) {
            throw new 
Exception('Only single cell references may be passed to this method.');
        }

        
// Get coordinate of $pBefore
        
[$beforeColumn$beforeRow] = Coordinate::coordinateFromString($pBefore);

        
// Get coordinate of $pCellReference
        
[$newColumn$newRow] = Coordinate::coordinateFromString($pCellReference);

        
// Verify which parts should be updated
        
$updateColumn = (($newColumn[0] != '$') && ($beforeColumn[0] != '$') && (Coordinate::columnIndexFromString($newColumn) >= Coordinate::columnIndexFromString($beforeColumn)));
        
$updateRow = (($newRow[0] != '$') && ($beforeRow[0] != '$') && $newRow >= $beforeRow);

        
// Create new column reference
        
if ($updateColumn) {
            
$newColumn Coordinate::stringFromColumnIndex(Coordinate::columnIndexFromString($newColumn) + $pNumCols);
        }

        
// Create new row reference
        
if ($updateRow) {
            
$newRow $newRow $pNumRows;
        }

        
// Return new reference
        
return $newColumn $newRow;
    }

    
/**
     * __clone implementation. Cloning should not be allowed in a Singleton!
     */
    
final public function __clone()
    {
        throw new 
Exception('Cloning a Singleton is not allowed!');
    }
}

:: Command execute ::

Enter:
 
Select:
 

:: Search ::
  - regexp 

:: Upload ::
 
[ Read-Only ]

:: Make Dir ::
 
[ Read-Only ]
:: Make File ::
 
[ Read-Only ]

:: Go Dir ::
 
:: Go File ::
 

--[ c99shell v. 2.5 [PHP 8 Update] [24.05.2025] | Generation time: 0.0119 ]--