Skip to main content
Drupal API
User account menu
  • Log in

Breadcrumb

  1. Drupal Core 11.1.x
  2. ControlStructureSpacingSniff.php

function ControlStructureSpacingSniff::process

Same name in this branch
  1. 11.1.x vendor/squizlabs/php_codesniffer/src/Standards/PSR12/Sniffs/ControlStructures/ControlStructureSpacingSniff.php \PHP_CodeSniffer\Standards\PSR12\Sniffs\ControlStructures\ControlStructureSpacingSniff::process()
  2. 11.1.x vendor/squizlabs/php_codesniffer/src/Standards/PSR2/Sniffs/ControlStructures/ControlStructureSpacingSniff.php \PHP_CodeSniffer\Standards\PSR2\Sniffs\ControlStructures\ControlStructureSpacingSniff::process()

Processes this test, when one of its tokens is encountered.

Parameters

\PHP_CodeSniffer\Files\File $phpcsFile The file being scanned.:

int $stackPtr The position of the current token: in the stack passed in $tokens.

Return value

void

Overrides Sniff::process

File

vendor/squizlabs/php_codesniffer/src/Standards/Squiz/Sniffs/WhiteSpace/ControlStructureSpacingSniff.php, line 64

Class

ControlStructureSpacingSniff

Namespace

PHP_CodeSniffer\Standards\Squiz\Sniffs\WhiteSpace

Code

public function process(File $phpcsFile, $stackPtr) {
    $tokens = $phpcsFile->getTokens();
    if (isset($tokens[$stackPtr]['parenthesis_opener']) === true && isset($tokens[$stackPtr]['parenthesis_closer']) === true) {
        $parenOpener = $tokens[$stackPtr]['parenthesis_opener'];
        $parenCloser = $tokens[$stackPtr]['parenthesis_closer'];
        if ($tokens[$parenOpener + 1]['code'] === T_WHITESPACE) {
            $gap = $tokens[$parenOpener + 1]['length'];
            if ($gap === 0) {
                $phpcsFile->recordMetric($stackPtr, 'Spaces after control structure open parenthesis', 'newline');
                $gap = 'newline';
            }
            else {
                $phpcsFile->recordMetric($stackPtr, 'Spaces after control structure open parenthesis', $gap);
            }
            $error = 'Expected 0 spaces after opening bracket; %s found';
            $data = [
                $gap,
            ];
            $fix = $phpcsFile->addFixableError($error, $parenOpener + 1, 'SpacingAfterOpenBrace', $data);
            if ($fix === true) {
                $phpcsFile->fixer
                    ->replaceToken($parenOpener + 1, '');
            }
        }
        else {
            $phpcsFile->recordMetric($stackPtr, 'Spaces after control structure open parenthesis', 0);
        }
        if ($tokens[$parenOpener]['line'] === $tokens[$parenCloser]['line'] && $tokens[$parenCloser - 1]['code'] === T_WHITESPACE) {
            $gap = $tokens[$parenCloser - 1]['length'];
            $error = 'Expected 0 spaces before closing bracket; %s found';
            $data = [
                $gap,
            ];
            $fix = $phpcsFile->addFixableError($error, $parenCloser - 1, 'SpaceBeforeCloseBrace', $data);
            if ($fix === true) {
                $phpcsFile->fixer
                    ->replaceToken($parenCloser - 1, '');
            }
            if ($gap === 0) {
                $phpcsFile->recordMetric($stackPtr, 'Spaces before control structure close parenthesis', 'newline');
            }
            else {
                $phpcsFile->recordMetric($stackPtr, 'Spaces before control structure close parenthesis', $gap);
            }
        }
        else {
            $phpcsFile->recordMetric($stackPtr, 'Spaces before control structure close parenthesis', 0);
        }
    }
    
    //end if
    if (isset($tokens[$stackPtr]['scope_closer']) === false) {
        return;
    }
    $scopeOpener = $tokens[$stackPtr]['scope_opener'];
    $scopeCloser = $tokens[$stackPtr]['scope_closer'];
    for ($firstContent = $scopeOpener + 1; $firstContent < $phpcsFile->numTokens; $firstContent++) {
        $code = $tokens[$firstContent]['code'];
        if ($code === T_WHITESPACE || $code === T_INLINE_HTML && trim($tokens[$firstContent]['content']) === '') {
            continue;
        }
        // Skip all empty tokens on the same line as the opener.
        if ($tokens[$firstContent]['line'] === $tokens[$scopeOpener]['line'] && (isset(Tokens::$emptyTokens[$code]) === true || $code === T_CLOSE_TAG)) {
            continue;
        }
        break;
    }
    // We ignore spacing for some structures that tend to have their own rules.
    $ignore = [
        T_FUNCTION => true,
        T_CLASS => true,
        T_INTERFACE => true,
        T_TRAIT => true,
        T_ENUM => true,
        T_DOC_COMMENT_OPEN_TAG => true,
    ];
    if (isset($ignore[$tokens[$firstContent]['code']]) === false && $tokens[$firstContent]['line'] >= $tokens[$scopeOpener]['line'] + 2) {
        $gap = $tokens[$firstContent]['line'] - $tokens[$scopeOpener]['line'] - 1;
        $phpcsFile->recordMetric($stackPtr, 'Blank lines at start of control structure', $gap);
        $error = 'Blank line found at start of control structure';
        $fix = $phpcsFile->addFixableError($error, $scopeOpener, 'SpacingAfterOpen');
        if ($fix === true) {
            $phpcsFile->fixer
                ->beginChangeset();
            $i = $scopeOpener + 1;
            while ($tokens[$i]['line'] !== $tokens[$firstContent]['line']) {
                // Start removing content from the line after the opener.
                if ($tokens[$i]['line'] !== $tokens[$scopeOpener]['line']) {
                    $phpcsFile->fixer
                        ->replaceToken($i, '');
                }
                $i++;
            }
            $phpcsFile->fixer
                ->endChangeset();
        }
    }
    else {
        $phpcsFile->recordMetric($stackPtr, 'Blank lines at start of control structure', 0);
    }
    
    //end if
    if ($firstContent !== $scopeCloser) {
        $lastContent = $phpcsFile->findPrevious(T_WHITESPACE, $scopeCloser - 1, null, true);
        $lastNonEmptyContent = $phpcsFile->findPrevious(Tokens::$emptyTokens, $scopeCloser - 1, null, true);
        $checkToken = $lastContent;
        if (isset($tokens[$lastNonEmptyContent]['scope_condition']) === true) {
            $checkToken = $tokens[$lastNonEmptyContent]['scope_condition'];
        }
        if (isset($ignore[$tokens[$checkToken]['code']]) === false && $tokens[$lastContent]['line'] <= $tokens[$scopeCloser]['line'] - 2) {
            $errorToken = $scopeCloser;
            for ($i = $scopeCloser - 1; $i > $lastContent; $i--) {
                if ($tokens[$i]['line'] < $tokens[$scopeCloser]['line']) {
                    $errorToken = $i;
                    break;
                }
            }
            $gap = $tokens[$scopeCloser]['line'] - $tokens[$lastContent]['line'] - 1;
            $phpcsFile->recordMetric($stackPtr, 'Blank lines at end of control structure', $gap);
            $error = 'Blank line found at end of control structure';
            $fix = $phpcsFile->addFixableError($error, $errorToken, 'SpacingBeforeClose');
            if ($fix === true) {
                $phpcsFile->fixer
                    ->beginChangeset();
                for ($i = $scopeCloser - 1; $i > $lastContent; $i--) {
                    if ($tokens[$i]['line'] === $tokens[$scopeCloser]['line']) {
                        continue;
                    }
                    if ($tokens[$i]['line'] === $tokens[$lastContent]['line']) {
                        break;
                    }
                    $phpcsFile->fixer
                        ->replaceToken($i, '');
                }
                $phpcsFile->fixer
                    ->endChangeset();
            }
        }
        else {
            $phpcsFile->recordMetric($stackPtr, 'Blank lines at end of control structure', 0);
        }
        
        //end if
    }
    
    //end if
    if ($tokens[$stackPtr]['code'] === T_MATCH) {
        // Move the scope closer to the semicolon/comma.
        $next = $phpcsFile->findNext(Tokens::$emptyTokens, $scopeCloser + 1, null, true);
        if ($next !== false && ($tokens[$next]['code'] === T_SEMICOLON || $tokens[$next]['code'] === T_COMMA)) {
            $scopeCloser = $next;
        }
    }
    $trailingContent = $phpcsFile->findNext(T_WHITESPACE, $scopeCloser + 1, null, true);
    if ($tokens[$trailingContent]['code'] === T_COMMENT || isset(Tokens::$phpcsCommentTokens[$tokens[$trailingContent]['code']]) === true) {
        // Special exception for code where the comment about
        // an ELSE or ELSEIF is written between the control structures.
        $nextCode = $phpcsFile->findNext(Tokens::$emptyTokens, $scopeCloser + 1, null, true);
        if ($tokens[$nextCode]['code'] === T_ELSE || $tokens[$nextCode]['code'] === T_ELSEIF || $tokens[$trailingContent]['line'] === $tokens[$scopeCloser]['line']) {
            $trailingContent = $nextCode;
        }
    }
    
    //end if
    if ($tokens[$trailingContent]['code'] === T_ELSE) {
        if ($tokens[$stackPtr]['code'] === T_IF) {
            // IF with ELSE.
            return;
        }
    }
    if ($tokens[$trailingContent]['code'] === T_WHILE && $tokens[$stackPtr]['code'] === T_DO) {
        // DO with WHILE.
        return;
    }
    if ($tokens[$trailingContent]['code'] === T_CLOSE_TAG) {
        // At the end of the script or embedded code.
        return;
    }
    if (isset($tokens[$trailingContent]['scope_condition']) === true && $tokens[$trailingContent]['scope_condition'] !== $trailingContent && isset($tokens[$trailingContent]['scope_opener']) === true && $tokens[$trailingContent]['scope_opener'] !== $trailingContent) {
        // Another control structure's closing brace.
        $owner = $tokens[$trailingContent]['scope_condition'];
        if ($tokens[$owner]['code'] === T_FUNCTION) {
            // The next content is the closing brace of a function
            // so normal function rules apply and we can ignore it.
            return;
        }
        if ($tokens[$owner]['code'] === T_CLOSURE && ($phpcsFile->hasCondition($stackPtr, [
            T_FUNCTION,
            T_CLOSURE,
        ]) === true || isset($tokens[$stackPtr]['nested_parenthesis']) === true)) {
            return;
        }
        if ($tokens[$trailingContent]['line'] !== $tokens[$scopeCloser]['line'] + 1) {
            $error = 'Blank line found after control structure';
            $fix = $phpcsFile->addFixableError($error, $scopeCloser, 'LineAfterClose');
            if ($fix === true) {
                $phpcsFile->fixer
                    ->beginChangeset();
                $i = $scopeCloser + 1;
                while ($tokens[$i]['line'] !== $tokens[$trailingContent]['line']) {
                    $phpcsFile->fixer
                        ->replaceToken($i, '');
                    $i++;
                }
                $phpcsFile->fixer
                    ->addNewline($scopeCloser);
                $phpcsFile->fixer
                    ->endChangeset();
            }
        }
    }
    else {
        if ($tokens[$trailingContent]['code'] !== T_ELSE && $tokens[$trailingContent]['code'] !== T_ELSEIF && $tokens[$trailingContent]['code'] !== T_CATCH && $tokens[$trailingContent]['code'] !== T_FINALLY && $tokens[$trailingContent]['line'] === $tokens[$scopeCloser]['line'] + 1) {
            $error = 'No blank line found after control structure';
            $fix = $phpcsFile->addFixableError($error, $scopeCloser, 'NoLineAfterClose');
            if ($fix === true) {
                $trailingContent = $phpcsFile->findNext(T_WHITESPACE, $scopeCloser + 1, null, true);
                if (($tokens[$trailingContent]['code'] === T_COMMENT || isset(Tokens::$phpcsCommentTokens[$tokens[$trailingContent]['code']]) === true) && $tokens[$trailingContent]['line'] === $tokens[$scopeCloser]['line']) {
                    $phpcsFile->fixer
                        ->addNewline($trailingContent);
                }
                else {
                    $phpcsFile->fixer
                        ->addNewline($scopeCloser);
                }
            }
        }
    }
    
    //end if
}

API Navigation

  • Drupal Core 11.1.x
  • Topics
  • Classes
  • Functions
  • Constants
  • Globals
  • Files
  • Namespaces
  • Deprecated
  • Services
RSS feed
Powered by Drupal