barudo
barudo

Reputation: 665

Using preg_replace to modify square brace placeholders in a string is incorrectly matching multiple placeholders at once

I need to remove an attribute declaration expression inside of a square-braced shortcode. This is my code:

$pattern = "#\[(.*)( rep=['\"]{1}1['\"]{1})(.*)\]#i";
$replacement = '[$1$3]';

//Test cases:
$values = [
  'This is a [s1 val="1"] test',
  'This is a [s1 val="1"] test rep="1"',
  'This is a [s1 val="1" rep="1"] test',
  'This is a [s1 de="2" rep="1" be="2"] test',
  'This is a [rep="1" de="2" ] test',
  'This is a [s2 val="1" rep="1" de="2"] test ... This is a [s3 val="1" rep="1" de="2" ] test',
];

foreach($values as $value)
{
    echo preg_replace($pattern, $replacement, $value) . "\n";
}

The first five cases work as expected. Unfortunately, the sixth case which has two shortcodes with rep="1" inside doesn't work as expected. I want the rep="1" in both shortcodes removed. Something must be wrong with the regular expression.

NOTE: the rep="1" must be inside the shortcode (inside the [ and ]) for it to be replaced.

Upvotes: 0

Views: 89

Answers (1)

Narendrasingh Sisodia
Narendrasingh Sisodia

Reputation: 21437

You need to place the non-greedy quantifier within your regex

\[(.*?)(rep="1")(.*?)\]
   //^^          //^^

Demo

Upvotes: 1

Related Questions