Passed
Push — main ( 5de797...be0e90 )
by Colin
03:01
created

AttributesHelper::mergeAttributes()   B

Complexity

Conditions 7
Paths 14

Size

Total Lines 30
Code Lines 15

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 16
CRAP Score 7

Importance

Changes 0
Metric Value
eloc 15
c 0
b 0
f 0
dl 0
loc 30
ccs 16
cts 16
cp 1
rs 8.8333
cc 7
nc 14
nop 2
crap 7
1
<?php
2
3
/*
4
 * This file is part of the league/commonmark package.
5
 *
6
 * (c) Colin O'Dell <[email protected]>
7
 * (c) 2015 Martin Hasoň <[email protected]>
8
 *
9
 * For the full copyright and license information, please view the LICENSE
10
 * file that was distributed with this source code.
11
 */
12
13
declare(strict_types=1);
14
15
namespace League\CommonMark\Extension\Attributes\Util;
16
17
use League\CommonMark\Node\Node;
18
use League\CommonMark\Parser\Cursor;
19
use League\CommonMark\Util\RegexHelper;
20
21
/**
22
 * @internal
23
 */
24
final class AttributesHelper
25
{
26
    private const SINGLE_ATTRIBUTE = '\s*([.]-?[_a-z][^\s}]*|[#][^\s}]+|' . RegexHelper::PARTIAL_ATTRIBUTENAME . RegexHelper::PARTIAL_ATTRIBUTEVALUESPEC . '?)\s*';
27
    private const ATTRIBUTE_LIST   = '/^{:?(' . self::SINGLE_ATTRIBUTE . ')+}/i';
28
29
    /**
30
     * @return array<string, mixed>
31
     */
32 100
    public static function parseAttributes(Cursor $cursor): array
33
    {
34 100
        $state = $cursor->saveState();
35 100
        $cursor->advanceToNextNonSpaceOrNewline();
36
37
        // Quick check to see if we might have attributes
38 100
        if ($cursor->getCharacter() !== '{') {
39 16
            $cursor->restoreState($state);
40
41 16
            return [];
42
        }
43
44
        // Attempt to match the entire attribute list expression
45
        // While this is less performant than checking for '{' now and '}' later, it simplifies
46
        // matching individual attributes since they won't need to look ahead for the closing '}'
47
        // while dealing with the fact that attributes can technically contain curly braces.
48
        // So we'll just match the start and end braces up front.
49 98
        $attributeExpression = $cursor->match(self::ATTRIBUTE_LIST);
50 98
        if ($attributeExpression === null) {
51 8
            $cursor->restoreState($state);
52
53 8
            return [];
54
        }
55
56
        // Trim the leading '{' or '{:' and the trailing '}'
57 92
        $attributeExpression = \ltrim(\substr($attributeExpression, 1, -1), ':');
58 92
        $attributeCursor     = new Cursor($attributeExpression);
59
60
        /** @var array<string, mixed> $attributes */
61 92
        $attributes = [];
62 92
        while ($attribute = \trim((string) $attributeCursor->match('/^' . self::SINGLE_ATTRIBUTE . '/i'))) {
63 92
            if ($attribute[0] === '#') {
64 38
                $attributes['id'] = \substr($attribute, 1);
65
66 38
                continue;
67
            }
68
69 66
            if ($attribute[0] === '.') {
70 22
                $attributes['class'][] = \substr($attribute, 1);
71
72 22
                continue;
73
            }
74
75 54
            $parts = \explode('=', $attribute, 2);
76 54
            if (\count($parts) === 1) {
77 10
                $attributes[$attribute] = true;
78 10
                continue;
79
            }
80
81
            /** @psalm-suppress PossiblyUndefinedArrayOffset */
82 50
            [$name, $value] = $parts;
83
84 50
            $first = $value[0];
85 50
            $last  = \substr($value, -1);
86 50
            if (($first === '"' && $last === '"') || ($first === "'" && $last === "'") && \strlen($value) > 1) {
87 32
                $value = \substr($value, 1, -1);
88
            }
89
90 50
            if (\strtolower(\trim($name)) === 'class') {
91 2
                foreach (\array_filter(\explode(' ', \trim($value))) as $class) {
92 2
                    $attributes['class'][] = $class;
93
                }
94
            } else {
95 50
                $attributes[\trim($name)] = \trim($value);
96
            }
97
        }
98
99 92
        if (isset($attributes['class'])) {
100 22
            $attributes['class'] = \implode(' ', (array) $attributes['class']);
101
        }
102
103 92
        return $attributes;
104
    }
105
106
    /**
107
     * @param Node|array<string, mixed> $attributes1
108
     * @param Node|array<string, mixed> $attributes2
109
     *
110
     * @return array<string, mixed>
111
     */
112 50
    public static function mergeAttributes($attributes1, $attributes2): array
113
    {
114 50
        $attributes = [];
115 50
        foreach ([$attributes1, $attributes2] as $arg) {
116 50
            if ($arg instanceof Node) {
117 30
                $arg = $arg->data->get('attributes');
118
            }
119
120
            /** @var array<string, mixed> $arg */
121 50
            $arg = (array) $arg;
122 50
            if (isset($arg['class'])) {
123 36
                if (\is_string($arg['class'])) {
124 36
                    $arg['class'] = \array_filter(\explode(' ', \trim($arg['class'])));
125
                }
126
127 36
                foreach ($arg['class'] as $class) {
128 36
                    $attributes['class'][] = $class;
129
                }
130
131 36
                unset($arg['class']);
132
            }
133
134 50
            $attributes = \array_merge($attributes, $arg);
135
        }
136
137 50
        if (isset($attributes['class'])) {
138 36
            $attributes['class'] = \implode(' ', $attributes['class']);
139
        }
140
141 50
        return $attributes;
142
    }
143
}
144