!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)

/var/www/html/main_file/vendor/phpoffice/phpspreadsheet/src/PhpSpreadsheet/   drwxr-xr-x
Free 13.26 GB of 57.97 GB (22.88%)
Home    Back    Forward    UPDIR    Refresh    Search    Buffer    Encoder    Tools    Proc.    FTP brute    Sec.    SQL    PHP-code    Update    Self remove    Logout    


Viewing file:     ReferenceHelper.php (52.79 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\Style\Conditional;
use 
PhpOffice\PhpSpreadsheet\Worksheet\AutoFilter;
use 
PhpOffice\PhpSpreadsheet\Worksheet\Table;
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;

    
/**
     * @var CellReferenceHelper
     */
    
private $cellReferenceHelper;

    
/**
     * Get an instance of this class.
     *
     * @return ReferenceHelper
     */
    
public static function getInstance()
    {
        if (
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);
    }

    
/** @var int */
    
private static $scrutinizer0 0;

    
/**
     * 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 $bc '';
        
$ar self::$scrutinizer0;
        
$br 0;
        
sscanf($a'%[A-Z]%d'$ac$ar);
        
sscanf($b'%[A-Z]%d'$bc$br);

        
$ac = (string) $ac;
        
$bc = (string) $bc;
        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 $bc '';
        
$ar self::$scrutinizer0;
        
$br 0;
        
sscanf($a'%[A-Z]%d'$ac$ar);
        
sscanf($b'%[A-Z]%d'$bc$br);

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

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

    
/**
     * Update page breaks when inserting/deleting rows/columns.
     *
     * @param Worksheet $worksheet The worksheet that we're editing
     * @param int $numberOfColumns Number of columns to insert/delete (negative values indicate deletion)
     * @param int $numberOfRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustPageBreaks(Worksheet $worksheet$numberOfColumns$numberOfRows): void
    
{
        
$aBreaks $worksheet->getBreaks();
        (
$numberOfColumns || $numberOfRows 0)
            ? 
uksort($aBreaks, [self::class, 'cellReverseSort'])
            : 
uksort($aBreaks, [self::class, 'cellSort']);

        foreach (
$aBreaks as $cellAddress => $value) {
            if (
$this->cellReferenceHelper->cellAddressInDeleteRange($cellAddress) === true) {
                
//    If we're deleting, then clear any defined breaks that are within the range
                //        of rows/columns that we're deleting
                
$worksheet->setBreak($cellAddressWorksheet::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($cellAddress);
                if (
$cellAddress !== $newReference) {
                    
$worksheet->setBreak($newReference$value)
                        ->
setBreak($cellAddressWorksheet::BREAK_NONE);
                }
            }
        }
    }

    
/**
     * Update cell comments when inserting/deleting rows/columns.
     *
     * @param Worksheet $worksheet The worksheet that we're editing
     */
    
protected function adjustComments($worksheet): void
    
{
        
$aComments $worksheet->getComments();
        
$aNewComments = []; // the new array of all comments

        
foreach ($aComments as $cellAddress => &$value) {
            
// Any comments inside a deleted range will be ignored
            
if ($this->cellReferenceHelper->cellAddressInDeleteRange($cellAddress) === false) {
                
// Otherwise build a new array of comments indexed by the adjusted cell reference
                
$newReference $this->updateCellReference($cellAddress);
                
$aNewComments[$newReference] = $value;
            }
        }
        
//    Replace the comments array with the new set of comments
        
$worksheet->setComments($aNewComments);
    }

    
/**
     * Update hyperlinks when inserting/deleting rows/columns.
     *
     * @param Worksheet $worksheet The worksheet that we're editing
     * @param int $numberOfColumns Number of columns to insert/delete (negative values indicate deletion)
     * @param int $numberOfRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustHyperlinks($worksheet$numberOfColumns$numberOfRows): void
    
{
        
$aHyperlinkCollection $worksheet->getHyperlinkCollection();
        (
$numberOfColumns || $numberOfRows 0)
            ? 
uksort($aHyperlinkCollection, [self::class, 'cellReverseSort'])
            : 
uksort($aHyperlinkCollection, [self::class, 'cellSort']);

        foreach (
$aHyperlinkCollection as $cellAddress => $value) {
            
$newReference $this->updateCellReference($cellAddress);
            if (
$this->cellReferenceHelper->cellAddressInDeleteRange($cellAddress) === true) {
                
$worksheet->setHyperlink($cellAddressnull);
            } elseif (
$cellAddress !== $newReference) {
                
$worksheet->setHyperlink($newReference$value);
                
$worksheet->setHyperlink($cellAddressnull);
            }
        }
    }

    
/**
     * Update conditional formatting styles when inserting/deleting rows/columns.
     *
     * @param Worksheet $worksheet The worksheet that we're editing
     * @param int $numberOfColumns Number of columns to insert/delete (negative values indicate deletion)
     * @param int $numberOfRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustConditionalFormatting($worksheet$numberOfColumns$numberOfRows): void
    
{
        
$aStyles $worksheet->getConditionalStylesCollection();
        (
$numberOfColumns || $numberOfRows 0)
            ? 
uksort($aStyles, [self::class, 'cellReverseSort'])
            : 
uksort($aStyles, [self::class, 'cellSort']);

        foreach (
$aStyles as $cellAddress => $cfRules) {
            
$worksheet->removeConditionalStyles($cellAddress);
            
$newReference $this->updateCellReference($cellAddress);

            foreach (
$cfRules as &$cfRule) {
                
/** @var Conditional $cfRule */
                
$conditions $cfRule->getConditions();
                foreach (
$conditions as &$condition) {
                    if (
is_string($condition)) {
                        
$condition $this->updateFormulaReferences(
                            
$condition,
                            
$this->cellReferenceHelper->beforeCellAddress(),
                            
$numberOfColumns,
                            
$numberOfRows,
                            
$worksheet->getTitle(),
                            
true
                        
);
                    }
                }
                
$cfRule->setConditions($conditions);
            }
            
$worksheet->setConditionalStyles($newReference$cfRules);
        }
    }

    
/**
     * Update data validations when inserting/deleting rows/columns.
     *
     * @param Worksheet $worksheet The worksheet that we're editing
     * @param int $numberOfColumns Number of columns to insert/delete (negative values indicate deletion)
     * @param int $numberOfRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustDataValidations(Worksheet $worksheet$numberOfColumns$numberOfRows): void
    
{
        
$aDataValidationCollection $worksheet->getDataValidationCollection();
        (
$numberOfColumns || $numberOfRows 0)
            ? 
uksort($aDataValidationCollection, [self::class, 'cellReverseSort'])
            : 
uksort($aDataValidationCollection, [self::class, 'cellSort']);

        foreach (
$aDataValidationCollection as $cellAddress => $dataValidation) {
            
$newReference $this->updateCellReference($cellAddress);
            if (
$cellAddress !== $newReference) {
                
$dataValidation->setSqref($newReference);
                
$worksheet->setDataValidation($newReference$dataValidation);
                
$worksheet->setDataValidation($cellAddressnull);
            }
        }
    }

    
/**
     * Update merged cells when inserting/deleting rows/columns.
     *
     * @param Worksheet $worksheet The worksheet that we're editing
     */
    
protected function adjustMergeCells(Worksheet $worksheet): void
    
{
        
$aMergeCells $worksheet->getMergeCells();
        
$aNewMergeCells = []; // the new array of all merge cells
        
foreach ($aMergeCells as $cellAddress => &$value) {
            
$newReference $this->updateCellReference($cellAddress);
            
$aNewMergeCells[$newReference] = $newReference;
        }
        
$worksheet->setMergeCells($aNewMergeCells); // replace the merge cells array
    
}

    
/**
     * Update protected cells when inserting/deleting rows/columns.
     *
     * @param Worksheet $worksheet The worksheet that we're editing
     * @param int $numberOfColumns Number of columns to insert/delete (negative values indicate deletion)
     * @param int $numberOfRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustProtectedCells(Worksheet $worksheet$numberOfColumns$numberOfRows): void
    
{
        
$aProtectedCells $worksheet->getProtectedCells();
        (
$numberOfColumns || $numberOfRows 0)
            ? 
uksort($aProtectedCells, [self::class, 'cellReverseSort'])
            : 
uksort($aProtectedCells, [self::class, 'cellSort']);
        foreach (
$aProtectedCells as $cellAddress => $value) {
            
$newReference $this->updateCellReference($cellAddress);
            if (
$cellAddress !== $newReference) {
                
$worksheet->protectCells($newReference$valuetrue);
                
$worksheet->unprotectCells($cellAddress);
            }
        }
    }

    
/**
     * Update column dimensions when inserting/deleting rows/columns.
     *
     * @param Worksheet $worksheet The worksheet that we're editing
     */
    
protected function adjustColumnDimensions(Worksheet $worksheet): void
    
{
        
$aColumnDimensions array_reverse($worksheet->getColumnDimensions(), true);
        if (!empty(
$aColumnDimensions)) {
            foreach (
$aColumnDimensions as $objColumnDimension) {
                
$newReference $this->updateCellReference($objColumnDimension->getColumnIndex() . '1');
                [
$newReference] = Coordinate::coordinateFromString($newReference);
                if (
$objColumnDimension->getColumnIndex() !== $newReference) {
                    
$objColumnDimension->setColumnIndex($newReference);
                }
            }

            
$worksheet->refreshColumnDimensions();
        }
    }

    
/**
     * Update row dimensions when inserting/deleting rows/columns.
     *
     * @param Worksheet $worksheet The worksheet that we're editing
     * @param int $beforeRow Number of the row we're inserting/deleting before
     * @param int $numberOfRows Number of rows to insert/delete (negative values indicate deletion)
     */
    
protected function adjustRowDimensions(Worksheet $worksheet$beforeRow$numberOfRows): void
    
{
        
$aRowDimensions array_reverse($worksheet->getRowDimensions(), true);
        if (!empty(
$aRowDimensions)) {
            foreach (
$aRowDimensions as $objRowDimension) {
                
$newReference $this->updateCellReference('A' $objRowDimension->getRowIndex());
                [, 
$newReference] = Coordinate::coordinateFromString($newReference);
                
$newRoweference = (int) $newReference;
                if (
$objRowDimension->getRowIndex() !== $newRoweference) {
                    
$objRowDimension->setRowIndex($newRoweference);
                }
            }

            
$worksheet->refreshRowDimensions();

            
$copyDimension $worksheet->getRowDimension($beforeRow 1);
            for (
$i $beforeRow$i <= $beforeRow $numberOfRows; ++$i) {
                
$newDimension $worksheet->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 $beforeCellAddress Insert before this cell address (e.g. 'A1')
     * @param int $numberOfColumns Number of columns to insert/delete (negative values indicate deletion)
     * @param int $numberOfRows Number of rows to insert/delete (negative values indicate deletion)
     * @param Worksheet $worksheet The worksheet that we're editing
     */
    
public function insertNewBefore(
        
string $beforeCellAddress,
        
int $numberOfColumns,
        
int $numberOfRows,
        
Worksheet $worksheet
    
): void {
        
$remove = ($numberOfColumns || $numberOfRows 0);

        if (
            
$this->cellReferenceHelper === null ||
            
$this->cellReferenceHelper->refreshRequired($beforeCellAddress$numberOfColumns$numberOfRows)
        ) {
            
$this->cellReferenceHelper = new CellReferenceHelper($beforeCellAddress$numberOfColumns$numberOfRows);
        }

        
// Get coordinate of $beforeCellAddress
        
[$beforeColumn$beforeRow] = Coordinate::indexesFromString($beforeCellAddress);

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

        
// 1. Clear column strips if we are removing columns
        
if ($numberOfColumns && $beforeColumn $numberOfColumns 0) {
            
$this->clearColumnStrips($highestRow$beforeColumn$numberOfColumns$worksheet);
        }

        
// 2. Clear row strips if we are removing rows
        
if ($numberOfRows && $beforeRow $numberOfRows 0) {
            
$this->clearRowStrips($highestColumn$beforeColumn$beforeRow$numberOfRows$worksheet);
        }

        
// Find missing coordinates. This is important when inserting column before the last column
        
$cellCollection $worksheet->getCellCollection();
        
$missingCoordinates array_filter(
            
array_map(function ($row) use ($highestColumn) {
                return 
$highestColumn $row;
            }, 
range(1$highestRow)),
            function (
$coordinate) use ($cellCollection) {
                return 
$cellCollection->has($coordinate) === false;
            }
        );

        
// Create missing cells with null values
        
if (!empty($missingCoordinates)) {
            foreach (
$missingCoordinates as $coordinate) {
                
$worksheet->createNewCell($coordinate);
            }
        }

        
$allCoordinates $worksheet->getCoordinates();
        if (
$remove) {
            
// It's faster to reverse and pop than to use unshift, especially with large cell collections
            
$allCoordinates array_reverse($allCoordinates);
        }

        
// Loop through cells, bottom-up, and change cell coordinate
        
while ($coordinate array_pop($allCoordinates)) {
            
$cell $worksheet->getCell($coordinate);
            
$cellIndex Coordinate::columnIndexFromString($cell->getColumn());

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

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

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

                
// Insert this cell at its new location
                
if ($cell->getDataType() === DataType::TYPE_FORMULA) {
                    
// Formula should be adjusted
                    
$worksheet->getCell($newCoordinate)
                        ->
setValue($this->updateFormulaReferences($cell->getValue(), $beforeCellAddress$numberOfColumns$numberOfRows$worksheet->getTitle()));
                } else {
                    
// Formula should not be adjusted
                    
$worksheet->getCell($newCoordinate)->setValueExplicit($cell->getValue(), $cell->getDataType());
                }

                
// Clear the original cell
                
$worksheet->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(), $beforeCellAddress$numberOfColumns$numberOfRows$worksheet->getTitle()));
                }
            }
        }

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

        if (
$numberOfColumns && $beforeColumn 0) {
            
$this->duplicateStylesByColumn($worksheet$beforeColumn$beforeRow$highestRow$numberOfColumns);
        }

        if (
$numberOfRows && $beforeRow 0) {
            
$this->duplicateStylesByRow($worksheet$beforeColumn$beforeRow$highestColumn$numberOfRows);
        }

        
// Update worksheet: column dimensions
        
$this->adjustColumnDimensions($worksheet);

        
// Update worksheet: row dimensions
        
$this->adjustRowDimensions($worksheet$beforeRow$numberOfRows);

        
//    Update worksheet: page breaks
        
$this->adjustPageBreaks($worksheet$numberOfColumns$numberOfRows);

        
//    Update worksheet: comments
        
$this->adjustComments($worksheet);

        
// Update worksheet: hyperlinks
        
$this->adjustHyperlinks($worksheet$numberOfColumns$numberOfRows);

        
// Update worksheet: conditional formatting styles
        
$this->adjustConditionalFormatting($worksheet$numberOfColumns$numberOfRows);

        
// Update worksheet: data validations
        
$this->adjustDataValidations($worksheet$numberOfColumns$numberOfRows);

        
// Update worksheet: merge cells
        
$this->adjustMergeCells($worksheet);

        
// Update worksheet: protected cells
        
$this->adjustProtectedCells($worksheet$numberOfColumns$numberOfRows);

        
// Update worksheet: autofilter
        
$this->adjustAutoFilter($worksheet$beforeCellAddress$numberOfColumns);

        
// Update worksheet: table
        
$this->adjustTable($worksheet$beforeCellAddress$numberOfColumns);

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

            
$splitCell $this->updateCellReference($splitCell);
            
$topLeftCell $this->updateCellReference($topLeftCell);

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

        
// Page setup
        
if ($worksheet->getPageSetup()->isPrintAreaSet()) {
            
$worksheet->getPageSetup()->setPrintArea(
                
$this->updateCellReference($worksheet->getPageSetup()->getPrintArea())
            );
        }

        
// Update worksheet: drawings
        
$aDrawings $worksheet->getDrawingCollection();
        foreach (
$aDrawings as $objDrawing) {
            
$newReference $this->updateCellReference($objDrawing->getCoordinates());
            if (
$objDrawing->getCoordinates() != $newReference) {
                
$objDrawing->setCoordinates($newReference);
            }
            if (
$objDrawing->getCoordinates2() !== '') {
                
$newReference $this->updateCellReference($objDrawing->getCoordinates2());
                if (
$objDrawing->getCoordinates2() != $newReference) {
                    
$objDrawing->setCoordinates2($newReference);
                }
            }
        }

        
// Update workbook: define names
        
if (count($worksheet->getParentOrThrow()->getDefinedNames()) > 0) {
            
$this->updateDefinedNames($worksheet$beforeCellAddress$numberOfColumns$numberOfRows);
        }

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

    
/**
     * Update references within formulas.
     *
     * @param string $formula Formula to update
     * @param string $beforeCellAddress Insert before this one
     * @param int $numberOfColumns Number of columns to insert
     * @param int $numberOfRows Number of rows to insert
     * @param string $worksheetName Worksheet name/title
     *
     * @return string Updated formula
     */
    
public function updateFormulaReferences(
        
$formula '',
        
$beforeCellAddress 'A1',
        
$numberOfColumns 0,
        
$numberOfRows 0,
        
$worksheetName '',
        
bool $includeAbsoluteReferences false
    
) {
        if (
            
$this->cellReferenceHelper === null ||
            
$this->cellReferenceHelper->refreshRequired($beforeCellAddress$numberOfColumns$numberOfRows)
        ) {
            
$this->cellReferenceHelper = new CellReferenceHelper($beforeCellAddress$numberOfColumns$numberOfRows);
        }

        
//    Update cell references in the formula
        
$formulaBlocks explode('"'$formula);
        
$i false;
        foreach (
$formulaBlocks as &$formulaBlock) {
            
//    Ignore blocks that were enclosed in quotes (alternating entries in the $formulaBlocks array after the explode)
            
$i $i === false;
            if (
$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 '/mui'' ' $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], $includeAbsoluteReferences), 2);
                        
$modified4 substr($this->updateCellReference('$A' $match[4], $includeAbsoluteReferences), 2);

                        if (
$match[3] . ':' $match[4] !== $modified3 ':' $modified4) {
                            if ((
$match[2] == '') || (trim($match[2], "'") == $worksheetName)) {
                                
$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 + (int) 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 '/mui'' ' $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'$includeAbsoluteReferences), 0, -2);
                        
$modified4 substr($this->updateCellReference($match[4] . '$1'$includeAbsoluteReferences), 0, -2);

                        if (
$match[3] . ':' $match[4] !== $modified3 ':' $modified4) {
                            if ((
$match[2] == '') || (trim($match[2], "'") == $worksheetName)) {
                                
$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 '/mui'' ' $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], $includeAbsoluteReferences);
                        
$modified4 $this->updateCellReference($match[4], $includeAbsoluteReferences);

                        if (
$match[3] . $match[4] !== $modified3 $modified4) {
                            if ((
$match[2] == '') || (trim($match[2], "'") == $worksheetName)) {
                                
$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 = (int) 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 '/mui'' ' $formulaBlock ' '$matchesPREG_SET_ORDER);

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

                        
$modified3 $this->updateCellReference($match[3], $includeAbsoluteReferences);
                        if (
$match[3] !== $modified3) {
                            if ((
$match[2] == '') || (trim($match[2], "'") == $worksheetName)) {
                                
$toString = ($match[2] > '') ? $match[2] . '!' '';
                                
$toString .= $modified3;
                                [
$column$row] = Coordinate::coordinateFromString($match[3]);
                                
$columnAdditionalIndex $column[0] === '$' 0;
                                
$rowAdditionalIndex $row[0] === '$' 0;
                                
//    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 = (int) trim($row'$') + 10000000;
                                
$cellIndex $row $rowAdditionalIndex $column $columnAdditionalIndex;

                                
$newCellTokens[$cellIndex] = preg_quote($toString'/');
                                
$cellTokens[$cellIndex] = '/(?<![A-Z\$\!])' preg_quote($fromString'/') . '(?!\d)/i';
                                ++
$adjustCount;
                            }
                        }
                    }
                }
                if (
$adjustCount 0) {
                    if (
$numberOfColumns || $numberOfRows 0) {
                        
krsort($cellTokens);
                        
krsort($newCellTokens);
                    } else {
                        
ksort($cellTokens);
                        
ksort($newCellTokens);
                    }   
//  Update cell references in the formula
                    
$formulaBlock str_replace('\\''', (string) 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 $numberOfColumns 0int $numberOfRows 0): string
    
{
        
$formula $this->updateCellReferencesAllWorksheets($formula$numberOfColumns$numberOfRows);

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

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

        return 
$formula;
    }

    private function 
updateCellReferencesAllWorksheets(string $formulaint $numberOfColumnsint $numberOfRows): 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) + $numberOfColumns);
                
$formula substr($formula0$columnOffset) . $column substr($formula$columnOffset $columnLength);
            }
            if (!empty(
$row) && $row[0] !== '$') {
                
$row = (int) $row $numberOfRows;
                
$formula substr($formula0$rowOffset) . $row substr($formula$rowOffset $rowLength);
            }
        }

        return 
$formula;
    }

    private function 
updateColumnRangesAllWorksheets(string $formulaint $numberOfColumns): 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) + $numberOfColumns);
                
$formula substr($formula0$fromColumnOffset) . $fromColumn substr($formula$fromColumnOffset $fromColumnLength);
            }
            if (!empty(
$toColumn) && $toColumn[0] !== '$') {
                
$toColumn Coordinate::stringFromColumnIndex(Coordinate::columnIndexFromString($toColumn) + $numberOfColumns);
                
$formula substr($formula0$toColumnOffset) . $toColumn substr($formula$toColumnOffset $toColumnLength);
            }
        }

        return 
$formula;
    }

    private function 
updateRowRangesAllWorksheets(string $formulaint $numberOfRows): 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 = (int) $fromRow $numberOfRows;
                
$formula substr($formula0$fromRowOffset) . $fromRow substr($formula$fromRowOffset $fromRowLength);
            }
            if (!empty(
$toRow) && $toRow[0] !== '$') {
                
$toRow = (int) $toRow $numberOfRows;
                
$formula substr($formula0$toRowOffset) . $toRow substr($formula$toRowOffset $toRowLength);
            }
        }

        return 
$formula;
    }

    
/**
     * Update cell reference.
     *
     * @param string $cellReference Cell address or range of addresses
     *
     * @return string Updated cell range
     */
    
private function updateCellReference($cellReference 'A1'bool $includeAbsoluteReferences false)
    {
        
// Is it in another worksheet? Will not have to update anything.
        
if (strpos($cellReference'!') !== false) {
            return 
$cellReference;
        }
        
// Is it a range or a single cell?
        
if (!Coordinate::coordinateIsRange($cellReference)) {
            
// Single cell
            
return $this->cellReferenceHelper->updateCellReference($cellReference$includeAbsoluteReferences);
        }

        
// Range
        
return $this->updateCellRange($cellReference$includeAbsoluteReferences);
    }

    
/**
     * Update named formulae (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 updateNamedFormulae(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->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);
                    }
                }
            }
        }
    }

    private function 
updateDefinedNames(Worksheet $worksheetstring $beforeCellAddressint $numberOfColumnsint $numberOfRows): void
    
{
        foreach (
$worksheet->getParentOrThrow()->getDefinedNames() as $definedName) {
            if (
$definedName->isFormula() === false) {
                
$this->updateNamedRange($definedName$worksheet$beforeCellAddress$numberOfColumns$numberOfRows);
            } else {
                
$this->updateNamedFormula($definedName$worksheet$beforeCellAddress$numberOfColumns$numberOfRows);
            }
        }
    }

    private function 
updateNamedRange(DefinedName $definedNameWorksheet $worksheetstring $beforeCellAddressint $numberOfColumnsint $numberOfRows): void
    
{
        
$cellAddress $definedName->getValue();
        
$asFormula = ($cellAddress[0] === '=');
        if (
$definedName->getWorksheet() !== null && $definedName->getWorksheet()->getHashCode() === $worksheet->getHashCode()) {
            if (
$asFormula === true) {
                
$formula $this->updateFormulaReferences($cellAddress$beforeCellAddress$numberOfColumns$numberOfRows$worksheet->getTitle());
                
$definedName->setValue($formula);
            } else {
                
$definedName->setValue($this->updateCellReference(ltrim($cellAddress'=')));
            }
        }
    }

    private function 
updateNamedFormula(DefinedName $definedNameWorksheet $worksheetstring $beforeCellAddressint $numberOfColumnsint $numberOfRows): void
    
{
        if (
$definedName->getWorksheet() !== null && $definedName->getWorksheet()->getHashCode() === $worksheet->getHashCode()) {
            
$formula $definedName->getValue();
            
$formula $this->updateFormulaReferences($formula$beforeCellAddress$numberOfColumns$numberOfRows$worksheet->getTitle());
            
$definedName->setValue($formula);
        }
    }

    
/**
     * Update cell range.
     *
     * @param string $cellRange Cell range    (e.g. 'B2:D4', 'B:C' or '2:3')
     *
     * @return string Updated cell range
     */
    
private function updateCellRange(string $cellRange 'A1:A1'bool $includeAbsoluteReferences false): string
    
{
        if (!
Coordinate::coordinateIsRange($cellRange)) {
            throw new 
Exception('Only cell ranges may be passed to this method.');
        }

        
// Update range
        
$range Coordinate::splitRange($cellRange);
        
$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])) {
                    
$range[$i][$j] = Coordinate::coordinateFromString(
                        
$this->cellReferenceHelper->updateCellReference($range[$i][$j] . '1'$includeAbsoluteReferences)
                    )[
0];
                } elseif (
ctype_digit($range[$i][$j])) {
                    
$range[$i][$j] = Coordinate::coordinateFromString(
                        
$this->cellReferenceHelper->updateCellReference('A' $range[$i][$j], $includeAbsoluteReferences)
                    )[
1];
                } else {
                    
$range[$i][$j] = $this->cellReferenceHelper->updateCellReference($range[$i][$j], $includeAbsoluteReferences);
                }
            }
        }

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

    private function 
clearColumnStrips(int $highestRowint $beforeColumnint $numberOfColumnsWorksheet $worksheet): void
    
{
        
$startColumnId Coordinate::stringFromColumnIndex($beforeColumn $numberOfColumns);
        
$endColumnId Coordinate::stringFromColumnIndex($beforeColumn);

        for (
$row 1$row <= $highestRow 1; ++$row) {
            for (
$column $startColumnId$column !== $endColumnId; ++$column) {
                
$coordinate $column $row;
                
$this->clearStripCell($worksheet$coordinate);
            }
        }
    }

    private function 
clearRowStrips(string $highestColumnint $beforeColumnint $beforeRowint $numberOfRowsWorksheet $worksheet): void
    
{
        
$startColumnId Coordinate::stringFromColumnIndex($beforeColumn);
        ++
$highestColumn;

        for (
$column $startColumnId$column !== $highestColumn; ++$column) {
            for (
$row $beforeRow $numberOfRows$row <= $beforeRow 1; ++$row) {
                
$coordinate $column $row;
                
$this->clearStripCell($worksheet$coordinate);
            }
        }
    }

    private function 
clearStripCell(Worksheet $worksheetstring $coordinate): void
    
{
        
$worksheet->removeConditionalStyles($coordinate);
        
$worksheet->setHyperlink($coordinate);
        
$worksheet->setDataValidation($coordinate);
        
$worksheet->removeComment($coordinate);

        if (
$worksheet->cellExists($coordinate)) {
            
$worksheet->getCell($coordinate)->setValueExplicit(nullDataType::TYPE_NULL);
            
$worksheet->getCell($coordinate)->setXfIndex(0);
        }
    }

    private function 
adjustAutoFilter(Worksheet $worksheetstring $beforeCellAddressint $numberOfColumns): void
    
{
        
$autoFilter $worksheet->getAutoFilter();
        
$autoFilterRange $autoFilter->getRange();
        if (!empty(
$autoFilterRange)) {
            if (
$numberOfColumns !== 0) {
                
$autoFilterColumns $autoFilter->getColumns();
                if (
count($autoFilterColumns) > 0) {
                    
$column '';
                    
$row 0;
                    
sscanf($beforeCellAddress'%[A-Z]%d'$column$row);
                    
$columnIndex Coordinate::columnIndexFromString((string) $column);
                    [
$rangeStart$rangeEnd] = Coordinate::rangeBoundaries($autoFilterRange);
                    if (
$columnIndex <= $rangeEnd[0]) {
                        if (
$numberOfColumns 0) {
                            
$this->adjustAutoFilterDeleteRules($columnIndex$numberOfColumns$autoFilterColumns$autoFilter);
                        }
                        
$startCol = ($columnIndex $rangeStart[0]) ? $columnIndex $rangeStart[0];

                        
//    Shuffle columns in autofilter range
                        
if ($numberOfColumns 0) {
                            
$this->adjustAutoFilterInsert($startCol$numberOfColumns$rangeEnd[0], $autoFilter);
                        } else {
                            
$this->adjustAutoFilterDelete($startCol$numberOfColumns$rangeEnd[0], $autoFilter);
                        }
                    }
                }
            }

            
$worksheet->setAutoFilter(
                
$this->updateCellReference($autoFilterRange)
            );
        }
    }

    private function 
adjustAutoFilterDeleteRules(int $columnIndexint $numberOfColumns, array $autoFilterColumnsAutoFilter $autoFilter): void
    
{
        
// If we're actually deleting any columns that fall within the autofilter range,
        //    then we delete any rules for those columns
        
$deleteColumn $columnIndex $numberOfColumns 1;
        
$deleteCount abs($numberOfColumns);

        for (
$i 1$i <= $deleteCount; ++$i) {
            
$columnName Coordinate::stringFromColumnIndex($deleteColumn 1);
            if (isset(
$autoFilterColumns[$columnName])) {
                
$autoFilter->clearColumn($columnName);
            }
            ++
$deleteColumn;
        }
    }

    private function 
adjustAutoFilterInsert(int $startColint $numberOfColumnsint $rangeEndAutoFilter $autoFilter): void
    
{
        
$startColRef $startCol;
        
$endColRef $rangeEnd;
        
$toColRef $rangeEnd $numberOfColumns;

        do {
            
$autoFilter->shiftColumn(Coordinate::stringFromColumnIndex($endColRef), Coordinate::stringFromColumnIndex($toColRef));
            --
$endColRef;
            --
$toColRef;
        } while (
$startColRef <= $endColRef);
    }

    private function 
adjustAutoFilterDelete(int $startColint $numberOfColumnsint $rangeEndAutoFilter $autoFilter): void
    
{
        
// For delete, we shuffle from beginning to end to avoid overwriting
        
$startColID Coordinate::stringFromColumnIndex($startCol);
        
$toColID Coordinate::stringFromColumnIndex($startCol $numberOfColumns);
        
$endColID Coordinate::stringFromColumnIndex($rangeEnd 1);

        do {
            
$autoFilter->shiftColumn($startColID$toColID);
            ++
$startColID;
            ++
$toColID;
        } while (
$startColID !== $endColID);
    }

    private function 
adjustTable(Worksheet $worksheetstring $beforeCellAddressint $numberOfColumns): void
    
{
        
$tableCollection $worksheet->getTableCollection();

        foreach (
$tableCollection as $table) {
            
$tableRange $table->getRange();
            if (!empty(
$tableRange)) {
                if (
$numberOfColumns !== 0) {
                    
$tableColumns $table->getColumns();
                    if (
count($tableColumns) > 0) {
                        
$column '';
                        
$row 0;
                        
sscanf($beforeCellAddress'%[A-Z]%d'$column$row);
                        
$columnIndex Coordinate::columnIndexFromString((string) $column);
                        [
$rangeStart$rangeEnd] = Coordinate::rangeBoundaries($tableRange);
                        if (
$columnIndex <= $rangeEnd[0]) {
                            if (
$numberOfColumns 0) {
                                
$this->adjustTableDeleteRules($columnIndex$numberOfColumns$tableColumns$table);
                            }
                            
$startCol = ($columnIndex $rangeStart[0]) ? $columnIndex $rangeStart[0];

                            
//    Shuffle columns in table range
                            
if ($numberOfColumns 0) {
                                
$this->adjustTableInsert($startCol$numberOfColumns$rangeEnd[0], $table);
                            } else {
                                
$this->adjustTableDelete($startCol$numberOfColumns$rangeEnd[0], $table);
                            }
                        }
                    }
                }

                
$table->setRange($this->updateCellReference($tableRange));
            }
        }
    }

    private function 
adjustTableDeleteRules(int $columnIndexint $numberOfColumns, array $tableColumnsTable $table): void
    
{
        
// If we're actually deleting any columns that fall within the table range,
        //    then we delete any rules for those columns
        
$deleteColumn $columnIndex $numberOfColumns 1;
        
$deleteCount abs($numberOfColumns);

        for (
$i 1$i <= $deleteCount; ++$i) {
            
$columnName Coordinate::stringFromColumnIndex($deleteColumn 1);
            if (isset(
$tableColumns[$columnName])) {
                
$table->clearColumn($columnName);
            }
            ++
$deleteColumn;
        }
    }

    private function 
adjustTableInsert(int $startColint $numberOfColumnsint $rangeEndTable $table): void
    
{
        
$startColRef $startCol;
        
$endColRef $rangeEnd;
        
$toColRef $rangeEnd $numberOfColumns;

        do {
            
$table->shiftColumn(Coordinate::stringFromColumnIndex($endColRef), Coordinate::stringFromColumnIndex($toColRef));
            --
$endColRef;
            --
$toColRef;
        } while (
$startColRef <= $endColRef);
    }

    private function 
adjustTableDelete(int $startColint $numberOfColumnsint $rangeEndTable $table): void
    
{
        
// For delete, we shuffle from beginning to end to avoid overwriting
        
$startColID Coordinate::stringFromColumnIndex($startCol);
        
$toColID Coordinate::stringFromColumnIndex($startCol $numberOfColumns);
        
$endColID Coordinate::stringFromColumnIndex($rangeEnd 1);

        do {
            
$table->shiftColumn($startColID$toColID);
            ++
$startColID;
            ++
$toColID;
        } while (
$startColID !== $endColID);
    }

    private function 
duplicateStylesByColumn(Worksheet $worksheetint $beforeColumnint $beforeRowint $highestRowint $numberOfColumns): void
    
{
        
$beforeColumnName Coordinate::stringFromColumnIndex($beforeColumn 1);
        for (
$i $beforeRow$i <= $highestRow 1; ++$i) {
            
// Style
            
$coordinate $beforeColumnName $i;
            if (
$worksheet->cellExists($coordinate)) {
                
$xfIndex $worksheet->getCell($coordinate)->getXfIndex();
                for (
$j $beforeColumn$j <= $beforeColumn $numberOfColumns; ++$j) {
                    
$worksheet->getCell([$j$i])->setXfIndex($xfIndex);
                }
            }
        }
    }

    private function 
duplicateStylesByRow(Worksheet $worksheetint $beforeColumnint $beforeRowstring $highestColumnint $numberOfRows): void
    
{
        
$highestColumnIndex Coordinate::columnIndexFromString($highestColumn);
        for (
$i $beforeColumn$i <= $highestColumnIndex; ++$i) {
            
// Style
            
$coordinate Coordinate::stringFromColumnIndex($i) . ($beforeRow 1);
            if (
$worksheet->cellExists($coordinate)) {
                
$xfIndex $worksheet->getCell($coordinate)->getXfIndex();
                for (
$j $beforeRow$j <= $beforeRow $numberOfRows; ++$j) {
                    
$worksheet->getCell(Coordinate::stringFromColumnIndex($i) . $j)->setXfIndex($xfIndex);
                }
            }
        }
    }

    
/**
     * __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.012 ]--