Complex classes like HtmlMin often do a lot of different things. To break such a class down, we need to identify a cohesive component within that class. A common approach to find such a component is to look for fields/methods that share the same prefixes, or suffixes. You can also have a look at the cohesion graph to spot any un-connected, or weakly-connected components.
Once you have determined the fields that belong together, you can apply the Extract Class refactoring. If the component makes sense as a sub-class, Extract Subclass is also a candidate, and is often faster.
While breaking up the class, it is a good idea to analyze how other classes use HtmlMin, and based on these observations, apply Extract Interface, too.
1 | <?php |
||
21 | class HtmlMin |
||
22 | { |
||
23 | /** |
||
24 | * @var string |
||
25 | */ |
||
26 | private static $regExSpace = "/[[:space:]]{2,}|[\r\n]+/u"; |
||
27 | |||
28 | /** |
||
29 | * @var array |
||
30 | */ |
||
31 | private static $optional_end_tags = [ |
||
32 | 'html', |
||
33 | 'head', |
||
34 | 'body', |
||
35 | ]; |
||
36 | |||
37 | private static $selfClosingTags = [ |
||
38 | 'area', |
||
39 | 'base', |
||
40 | 'basefont', |
||
41 | 'br', |
||
42 | 'col', |
||
43 | 'command', |
||
44 | 'embed', |
||
45 | 'frame', |
||
46 | 'hr', |
||
47 | 'img', |
||
48 | 'input', |
||
49 | 'isindex', |
||
50 | 'keygen', |
||
51 | 'link', |
||
52 | 'meta', |
||
53 | 'param', |
||
54 | 'source', |
||
55 | 'track', |
||
56 | 'wbr', |
||
57 | ]; |
||
58 | |||
59 | private static $trimWhitespaceFromTags = [ |
||
60 | 'article' => '', |
||
61 | 'br' => '', |
||
62 | 'div' => '', |
||
63 | 'footer' => '', |
||
64 | 'hr' => '', |
||
65 | 'nav' => '', |
||
66 | 'p' => '', |
||
67 | 'script' => '', |
||
68 | ]; |
||
69 | |||
70 | /** |
||
71 | * @var array |
||
72 | */ |
||
73 | private static $booleanAttributes = [ |
||
74 | 'allowfullscreen' => '', |
||
75 | 'async' => '', |
||
76 | 'autofocus' => '', |
||
77 | 'autoplay' => '', |
||
78 | 'checked' => '', |
||
79 | 'compact' => '', |
||
80 | 'controls' => '', |
||
81 | 'declare' => '', |
||
82 | 'default' => '', |
||
83 | 'defaultchecked' => '', |
||
84 | 'defaultmuted' => '', |
||
85 | 'defaultselected' => '', |
||
86 | 'defer' => '', |
||
87 | 'disabled' => '', |
||
88 | 'enabled' => '', |
||
89 | 'formnovalidate' => '', |
||
90 | 'hidden' => '', |
||
91 | 'indeterminate' => '', |
||
92 | 'inert' => '', |
||
93 | 'ismap' => '', |
||
94 | 'itemscope' => '', |
||
95 | 'loop' => '', |
||
96 | 'multiple' => '', |
||
97 | 'muted' => '', |
||
98 | 'nohref' => '', |
||
99 | 'noresize' => '', |
||
100 | 'noshade' => '', |
||
101 | 'novalidate' => '', |
||
102 | 'nowrap' => '', |
||
103 | 'open' => '', |
||
104 | 'pauseonexit' => '', |
||
105 | 'readonly' => '', |
||
106 | 'required' => '', |
||
107 | 'reversed' => '', |
||
108 | 'scoped' => '', |
||
109 | 'seamless' => '', |
||
110 | 'selected' => '', |
||
111 | 'sortable' => '', |
||
112 | 'truespeed' => '', |
||
113 | 'typemustmatch' => '', |
||
114 | 'visible' => '', |
||
115 | ]; |
||
116 | |||
117 | /** |
||
118 | * @var array |
||
119 | */ |
||
120 | private static $skipTagsForRemoveWhitespace = [ |
||
121 | 'code', |
||
122 | 'pre', |
||
123 | 'script', |
||
124 | 'style', |
||
125 | 'textarea', |
||
126 | ]; |
||
127 | |||
128 | /** |
||
129 | * @var array |
||
130 | */ |
||
131 | private $protectedChildNodes = []; |
||
132 | |||
133 | /** |
||
134 | * @var string |
||
135 | */ |
||
136 | private $protectedChildNodesHelper = 'html-min--voku--saved-content'; |
||
137 | |||
138 | /** |
||
139 | * @var bool |
||
140 | */ |
||
141 | private $doOptimizeViaHtmlDomParser = true; |
||
142 | |||
143 | /** |
||
144 | * @var bool |
||
145 | */ |
||
146 | private $doOptimizeAttributes = true; |
||
147 | |||
148 | /** |
||
149 | * @var bool |
||
150 | */ |
||
151 | private $doRemoveComments = true; |
||
152 | |||
153 | /** |
||
154 | * @var bool |
||
155 | */ |
||
156 | private $doRemoveWhitespaceAroundTags = false; |
||
157 | |||
158 | /** |
||
159 | * @var bool |
||
160 | */ |
||
161 | private $doRemoveOmittedQuotes = true; |
||
162 | |||
163 | /** |
||
164 | * @var bool |
||
165 | */ |
||
166 | private $doRemoveOmittedHtmlTags = true; |
||
167 | |||
168 | /** |
||
169 | * @var bool |
||
170 | */ |
||
171 | private $doRemoveHttpPrefixFromAttributes = false; |
||
172 | |||
173 | /** |
||
174 | * @var array |
||
175 | */ |
||
176 | private $domainsToRemoveHttpPrefixFromAttributes = [ |
||
177 | 'google.com', |
||
178 | 'google.de', |
||
179 | ]; |
||
180 | |||
181 | /** |
||
182 | * @var bool |
||
183 | */ |
||
184 | private $doSortCssClassNames = true; |
||
185 | |||
186 | /** |
||
187 | * @var bool |
||
188 | */ |
||
189 | private $doSortHtmlAttributes = true; |
||
190 | |||
191 | /** |
||
192 | * @var bool |
||
193 | */ |
||
194 | private $doRemoveDeprecatedScriptCharsetAttribute = true; |
||
195 | |||
196 | /** |
||
197 | * @var bool |
||
198 | */ |
||
199 | private $doRemoveDefaultAttributes = false; |
||
200 | |||
201 | /** |
||
202 | * @var bool |
||
203 | */ |
||
204 | private $doRemoveDeprecatedAnchorName = true; |
||
205 | |||
206 | /** |
||
207 | * @var bool |
||
208 | */ |
||
209 | private $doRemoveDeprecatedTypeFromStylesheetLink = true; |
||
210 | |||
211 | /** |
||
212 | * @var bool |
||
213 | */ |
||
214 | private $doRemoveDeprecatedTypeFromScriptTag = true; |
||
215 | |||
216 | /** |
||
217 | * @var bool |
||
218 | */ |
||
219 | private $doRemoveValueFromEmptyInput = true; |
||
220 | |||
221 | /** |
||
222 | * @var bool |
||
223 | */ |
||
224 | private $doRemoveEmptyAttributes = true; |
||
225 | |||
226 | /** |
||
227 | * @var bool |
||
228 | */ |
||
229 | private $doSumUpWhitespace = true; |
||
230 | |||
231 | /** |
||
232 | * @var bool |
||
233 | */ |
||
234 | private $doRemoveSpacesBetweenTags = false; |
||
235 | |||
236 | /** |
||
237 | * @var bool |
||
238 | */ |
||
239 | private $keepBrokenHtml = false; |
||
240 | |||
241 | /** |
||
242 | * @var bool |
||
243 | */ |
||
244 | private $withDocType = false; |
||
245 | |||
246 | /** |
||
247 | * @var \SplObjectStorage|HtmlMinDomObserverInterface[] |
||
248 | */ |
||
249 | private $domLoopObservers; |
||
250 | |||
251 | /** |
||
252 | * HtmlMin constructor. |
||
253 | */ |
||
254 | 42 | public function __construct() |
|
260 | |||
261 | /** |
||
262 | * @param HtmlMinDomObserverInterface $observer |
||
263 | * |
||
264 | * @return void |
||
265 | */ |
||
266 | 42 | public function attachObserverToTheDomLoop(HtmlMinDomObserverInterface $observer) |
|
270 | |||
271 | /** |
||
272 | * @param $domElement SimpleHtmlDom |
||
273 | * |
||
274 | * @return void |
||
275 | */ |
||
276 | 38 | private function notifyObserversAboutDomElementBeforeMinification(SimpleHtmlDom $domElement) |
|
282 | |||
283 | 38 | private function notifyObserversAboutDomElementAfterMinification(SimpleHtmlDom $domElement) |
|
289 | |||
290 | /** |
||
291 | * @param bool $doOptimizeAttributes |
||
292 | * |
||
293 | * @return $this |
||
294 | */ |
||
295 | 2 | public function doOptimizeAttributes(bool $doOptimizeAttributes = true): self |
|
301 | |||
302 | /** |
||
303 | * @param bool $doOptimizeViaHtmlDomParser |
||
304 | * |
||
305 | * @return $this |
||
306 | */ |
||
307 | 1 | public function doOptimizeViaHtmlDomParser(bool $doOptimizeViaHtmlDomParser = true): self |
|
313 | |||
314 | /** |
||
315 | * @param bool $doRemoveComments |
||
316 | * |
||
317 | * @return $this |
||
318 | */ |
||
319 | 3 | public function doRemoveComments(bool $doRemoveComments = true): self |
|
325 | |||
326 | /** |
||
327 | * @param bool $doRemoveDefaultAttributes |
||
328 | * |
||
329 | * @return $this |
||
330 | */ |
||
331 | 2 | public function doRemoveDefaultAttributes(bool $doRemoveDefaultAttributes = true): self |
|
337 | |||
338 | /** |
||
339 | * @param bool $doRemoveDeprecatedAnchorName |
||
340 | * |
||
341 | * @return $this |
||
342 | */ |
||
343 | 2 | public function doRemoveDeprecatedAnchorName(bool $doRemoveDeprecatedAnchorName = true): self |
|
349 | |||
350 | /** |
||
351 | * @param bool $doRemoveDeprecatedScriptCharsetAttribute |
||
352 | * |
||
353 | * @return $this |
||
354 | */ |
||
355 | 2 | public function doRemoveDeprecatedScriptCharsetAttribute(bool $doRemoveDeprecatedScriptCharsetAttribute = true): self |
|
361 | |||
362 | /** |
||
363 | * @param bool $doRemoveDeprecatedTypeFromScriptTag |
||
364 | * |
||
365 | * @return $this |
||
366 | */ |
||
367 | 2 | public function doRemoveDeprecatedTypeFromScriptTag(bool $doRemoveDeprecatedTypeFromScriptTag = true): self |
|
373 | |||
374 | /** |
||
375 | * @param bool $doRemoveDeprecatedTypeFromStylesheetLink |
||
376 | * |
||
377 | * @return $this |
||
378 | */ |
||
379 | 2 | public function doRemoveDeprecatedTypeFromStylesheetLink(bool $doRemoveDeprecatedTypeFromStylesheetLink = true): self |
|
385 | |||
386 | /** |
||
387 | * @param bool $doRemoveEmptyAttributes |
||
388 | * |
||
389 | * @return $this |
||
390 | */ |
||
391 | 2 | public function doRemoveEmptyAttributes(bool $doRemoveEmptyAttributes = true): self |
|
397 | |||
398 | /** |
||
399 | * @param bool $doRemoveHttpPrefixFromAttributes |
||
400 | * |
||
401 | * @return $this |
||
402 | */ |
||
403 | 4 | public function doRemoveHttpPrefixFromAttributes(bool $doRemoveHttpPrefixFromAttributes = true): self |
|
409 | |||
410 | /** |
||
411 | * @return bool |
||
412 | */ |
||
413 | 23 | public function isDoSortCssClassNames(): bool |
|
417 | |||
418 | /** |
||
419 | * @return bool |
||
420 | */ |
||
421 | 23 | public function isDoSortHtmlAttributes(): bool |
|
425 | |||
426 | /** |
||
427 | * @return bool |
||
428 | */ |
||
429 | 23 | public function isDoRemoveDeprecatedScriptCharsetAttribute(): bool |
|
433 | |||
434 | /** |
||
435 | * @return bool |
||
436 | */ |
||
437 | 23 | public function isDoRemoveDefaultAttributes(): bool |
|
441 | |||
442 | /** |
||
443 | * @return bool |
||
444 | */ |
||
445 | 23 | public function isDoRemoveDeprecatedAnchorName(): bool |
|
449 | |||
450 | /** |
||
451 | * @return bool |
||
452 | */ |
||
453 | 23 | public function isDoRemoveDeprecatedTypeFromStylesheetLink(): bool |
|
457 | |||
458 | /** |
||
459 | * @return bool |
||
460 | */ |
||
461 | 23 | public function isDoRemoveDeprecatedTypeFromScriptTag(): bool |
|
465 | |||
466 | /** |
||
467 | * @return bool |
||
468 | */ |
||
469 | 23 | public function isDoRemoveValueFromEmptyInput(): bool |
|
473 | |||
474 | /** |
||
475 | * @return bool |
||
476 | */ |
||
477 | 23 | public function isDoRemoveEmptyAttributes(): bool |
|
481 | |||
482 | /** |
||
483 | * @return bool |
||
484 | */ |
||
485 | public function isDoSumUpWhitespace(): bool |
||
489 | |||
490 | /** |
||
491 | * @return bool |
||
492 | */ |
||
493 | public function isDoRemoveSpacesBetweenTags(): bool |
||
497 | |||
498 | /** |
||
499 | * @return bool |
||
500 | */ |
||
501 | public function isDoOptimizeViaHtmlDomParser(): bool |
||
505 | |||
506 | /** |
||
507 | * @return bool |
||
508 | */ |
||
509 | public function isDoOptimizeAttributes(): bool |
||
513 | |||
514 | /** |
||
515 | * @return bool |
||
516 | */ |
||
517 | public function isDoRemoveComments(): bool |
||
521 | |||
522 | /** |
||
523 | * @return bool |
||
524 | */ |
||
525 | public function isDoRemoveWhitespaceAroundTags(): bool |
||
529 | |||
530 | /** |
||
531 | * @return bool |
||
532 | */ |
||
533 | public function isDoRemoveOmittedQuotes(): bool |
||
537 | |||
538 | /** |
||
539 | * @return bool |
||
540 | */ |
||
541 | public function isDoRemoveOmittedHtmlTags(): bool |
||
545 | |||
546 | /** |
||
547 | * @return bool |
||
548 | */ |
||
549 | 23 | public function isDoRemoveHttpPrefixFromAttributes(): bool |
|
553 | |||
554 | /** |
||
555 | * @return array |
||
556 | */ |
||
557 | public function getDomainsToRemoveHttpPrefixFromAttributes(): array |
||
561 | |||
562 | /** |
||
563 | * @param bool $doRemoveOmittedHtmlTags |
||
564 | * |
||
565 | * @return $this |
||
566 | */ |
||
567 | 1 | public function doRemoveOmittedHtmlTags(bool $doRemoveOmittedHtmlTags = true): self |
|
573 | |||
574 | /** |
||
575 | * @param bool $doRemoveOmittedQuotes |
||
576 | * |
||
577 | * @return $this |
||
578 | */ |
||
579 | 1 | public function doRemoveOmittedQuotes(bool $doRemoveOmittedQuotes = true): self |
|
585 | |||
586 | /** |
||
587 | * @param bool $doRemoveSpacesBetweenTags |
||
588 | * |
||
589 | * @return $this |
||
590 | */ |
||
591 | public function doRemoveSpacesBetweenTags(bool $doRemoveSpacesBetweenTags = true): self |
||
597 | |||
598 | /** |
||
599 | * @param bool $doRemoveValueFromEmptyInput |
||
600 | * |
||
601 | * @return $this |
||
602 | */ |
||
603 | 2 | public function doRemoveValueFromEmptyInput(bool $doRemoveValueFromEmptyInput = true): self |
|
609 | |||
610 | /** |
||
611 | * @param bool $doRemoveWhitespaceAroundTags |
||
612 | * |
||
613 | * @return $this |
||
614 | */ |
||
615 | 4 | public function doRemoveWhitespaceAroundTags(bool $doRemoveWhitespaceAroundTags = true): self |
|
621 | |||
622 | /** |
||
623 | * @param bool $doSortCssClassNames |
||
624 | * |
||
625 | * @return $this |
||
626 | */ |
||
627 | 2 | public function doSortCssClassNames(bool $doSortCssClassNames = true): self |
|
633 | |||
634 | /** |
||
635 | * @param bool $doSortHtmlAttributes |
||
636 | * |
||
637 | * @return $this |
||
638 | */ |
||
639 | 2 | public function doSortHtmlAttributes(bool $doSortHtmlAttributes = true): self |
|
645 | |||
646 | /** |
||
647 | * @param bool $doSumUpWhitespace |
||
648 | * |
||
649 | * @return $this |
||
650 | */ |
||
651 | 2 | public function doSumUpWhitespace(bool $doSumUpWhitespace = true): self |
|
657 | |||
658 | 38 | private function domNodeAttributesToString(\DOMNode $node): string |
|
706 | |||
707 | /** |
||
708 | * @param \DOMNode $node |
||
709 | * |
||
710 | * @return bool |
||
711 | */ |
||
712 | 37 | private function domNodeClosingTagOptional(\DOMNode $node): bool |
|
931 | |||
932 | 38 | protected function domNodeToString(\DOMNode $node): string |
|
1019 | |||
1020 | /** |
||
1021 | * @param \DOMNode $node |
||
1022 | * |
||
1023 | * @return \DOMNode|null |
||
1024 | */ |
||
1025 | 37 | protected function getNextSiblingOfTypeDOMElement(\DOMNode $node) |
|
1033 | |||
1034 | /** |
||
1035 | * Check if the current string is an conditional comment. |
||
1036 | * |
||
1037 | * INFO: since IE >= 10 conditional comment are not working anymore |
||
1038 | * |
||
1039 | * <!--[if expression]> HTML <![endif]--> |
||
1040 | * <![if expression]> HTML <![endif]> |
||
1041 | * |
||
1042 | * @param string $comment |
||
1043 | * |
||
1044 | * @return bool |
||
1045 | */ |
||
1046 | 4 | private function isConditionalComment($comment): bool |
|
1058 | |||
1059 | /** |
||
1060 | * @param string $html |
||
1061 | * @param bool $decodeUtf8Specials <p>Use this only in special cases, e.g. for PHP 5.3</p> |
||
1062 | * |
||
1063 | * @return string |
||
1064 | */ |
||
1065 | 42 | public function minify($html, $decodeUtf8Specials = false): string |
|
1066 | { |
||
1067 | 42 | $html = (string) $html; |
|
1068 | 42 | if (!isset($html[0])) { |
|
1069 | 1 | return ''; |
|
1070 | } |
||
1071 | |||
1072 | 42 | $html = \trim($html); |
|
1073 | 42 | if (!$html) { |
|
1074 | 3 | return ''; |
|
1075 | } |
||
1076 | |||
1077 | // init |
||
1078 | 39 | static $CACHE_SELF_CLOSING_TAGS = null; |
|
1079 | 39 | if ($CACHE_SELF_CLOSING_TAGS === null) { |
|
1080 | 1 | $CACHE_SELF_CLOSING_TAGS = \implode('|', self::$selfClosingTags); |
|
1081 | } |
||
1082 | |||
1083 | // reset |
||
1084 | 39 | $this->protectedChildNodes = []; |
|
1085 | |||
1086 | // save old content |
||
1087 | 39 | $origHtml = $html; |
|
1088 | 39 | $origHtmlLength = \strlen($html); |
|
1089 | |||
1090 | // ------------------------------------------------------------------------- |
||
1091 | // Minify the HTML via "HtmlDomParser" |
||
1092 | // ------------------------------------------------------------------------- |
||
1093 | |||
1094 | 39 | if ($this->doOptimizeViaHtmlDomParser) { |
|
1095 | 38 | $html = $this->minifyHtmlDom($html, $decodeUtf8Specials); |
|
1096 | } |
||
1097 | |||
1098 | // ------------------------------------------------------------------------- |
||
1099 | // Trim whitespace from html-string. [protected html is still protected] |
||
1100 | // ------------------------------------------------------------------------- |
||
1101 | |||
1102 | // Remove extra white-space(s) between HTML attribute(s) |
||
1103 | 39 | $html = (string) \preg_replace_callback( |
|
1104 | 39 | '#<([^/\s<>!]+)(?:\s+([^<>]*?)\s*|\s*)(/?)>#', |
|
1105 | 39 | function ($matches) { |
|
1106 | 39 | return '<' . $matches[1] . (string) \preg_replace('#([^\s=]+)(\=([\'"]?)(.*?)\3)?(\s+|$)#s', ' $1$2', $matches[2]) . $matches[3] . '>'; |
|
1107 | 39 | }, |
|
1108 | 39 | $html |
|
1109 | ); |
||
1110 | |||
1111 | 39 | if ($this->doRemoveSpacesBetweenTags) { |
|
1112 | // Remove spaces that are between > and < |
||
1113 | $html = (string) \preg_replace('/(>) (<)/', '>$2', $html); |
||
1114 | } |
||
1115 | |||
1116 | // ------------------------------------------------------------------------- |
||
1117 | // Restore protected HTML-code. |
||
1118 | // ------------------------------------------------------------------------- |
||
1119 | |||
1120 | 39 | $html = (string) \preg_replace_callback( |
|
1121 | 39 | '/<(?<element>' . $this->protectedChildNodesHelper . ')(?<attributes> [^>]*)?>(?<value>.*?)<\/' . $this->protectedChildNodesHelper . '>/', |
|
1122 | 39 | [$this, 'restoreProtectedHtml'], |
|
1123 | 39 | $html |
|
1124 | ); |
||
1125 | |||
1126 | // ------------------------------------------------------------------------- |
||
1127 | // Restore protected HTML-entities. |
||
1128 | // ------------------------------------------------------------------------- |
||
1129 | |||
1130 | 39 | if ($this->doOptimizeViaHtmlDomParser) { |
|
1131 | 38 | $html = HtmlDomParser::putReplacedBackToPreserveHtmlEntities($html); |
|
1132 | } |
||
1133 | |||
1134 | // ------------------------------------ |
||
1135 | // Final clean-up |
||
1136 | // ------------------------------------ |
||
1137 | |||
1138 | 39 | $html = \str_replace( |
|
1139 | [ |
||
1140 | 39 | 'html>' . "\n", |
|
1141 | "\n" . '<html', |
||
1142 | 'html/>' . "\n", |
||
1143 | "\n" . '</html', |
||
1144 | 'head>' . "\n", |
||
1145 | "\n" . '<head', |
||
1146 | 'head/>' . "\n", |
||
1147 | "\n" . '</head', |
||
1148 | ], |
||
1149 | [ |
||
1150 | 39 | 'html>', |
|
1151 | '<html', |
||
1152 | 'html/>', |
||
1153 | '</html', |
||
1154 | 'head>', |
||
1155 | '<head', |
||
1156 | 'head/>', |
||
1157 | '</head', |
||
1158 | ], |
||
1159 | 39 | $html |
|
1160 | ); |
||
1161 | |||
1162 | // self closing tags, don't need a trailing slash ... |
||
1163 | 39 | $replace = []; |
|
1164 | 39 | $replacement = []; |
|
1165 | 39 | foreach (self::$selfClosingTags as $selfClosingTag) { |
|
1166 | 39 | $replace[] = '<' . $selfClosingTag . '/>'; |
|
1167 | 39 | $replacement[] = '<' . $selfClosingTag . '>'; |
|
1168 | 39 | $replace[] = '<' . $selfClosingTag . ' />'; |
|
1169 | 39 | $replacement[] = '<' . $selfClosingTag . '>'; |
|
1170 | } |
||
1171 | 39 | $html = \str_replace( |
|
1172 | 39 | $replace, |
|
1173 | 39 | $replacement, |
|
1174 | 39 | $html |
|
1175 | ); |
||
1176 | |||
1177 | 39 | $html = (string) \preg_replace('#<\b(' . $CACHE_SELF_CLOSING_TAGS . ')([^>]*+)><\/\b\1>#', '<\\1\\2>', $html); |
|
1178 | |||
1179 | // ------------------------------------ |
||
1180 | // check if compression worked |
||
1181 | // ------------------------------------ |
||
1182 | |||
1183 | 39 | if ($origHtmlLength < \strlen($html)) { |
|
1184 | 3 | $html = $origHtml; |
|
1185 | } |
||
1186 | |||
1187 | 39 | return $html; |
|
1188 | } |
||
1189 | |||
1190 | /** |
||
1191 | * @param $html |
||
1192 | * @param $decodeUtf8Specials |
||
1193 | * |
||
1194 | * @return string |
||
1195 | */ |
||
1196 | 38 | private function minifyHtmlDom($html, $decodeUtf8Specials): string |
|
1260 | |||
1261 | /** |
||
1262 | * Prevent changes of inline "styles" and "scripts". |
||
1263 | * |
||
1264 | * @param HtmlDomParser $dom |
||
1265 | * |
||
1266 | * @return HtmlDomParser |
||
1267 | */ |
||
1268 | 38 | private function protectTags(HtmlDomParser $dom): HtmlDomParser |
|
1321 | |||
1322 | /** |
||
1323 | * Remove comments in the dom. |
||
1324 | * |
||
1325 | * @param HtmlDomParser $dom |
||
1326 | * |
||
1327 | * @return HtmlDomParser |
||
1328 | */ |
||
1329 | 36 | private function removeComments(HtmlDomParser $dom): HtmlDomParser |
|
1344 | |||
1345 | /** |
||
1346 | * Trim tags in the dom. |
||
1347 | * |
||
1348 | * @param SimpleHtmlDom $element |
||
1349 | * |
||
1350 | * @return void |
||
1351 | */ |
||
1352 | 3 | private function removeWhitespaceAroundTags(SimpleHtmlDom $element) |
|
1377 | |||
1378 | /** |
||
1379 | * Callback function for preg_replace_callback use. |
||
1380 | * |
||
1381 | * @param array $matches PREG matches |
||
1382 | * |
||
1383 | * @return string |
||
1384 | */ |
||
1385 | 5 | private function restoreProtectedHtml($matches): string |
|
1396 | |||
1397 | /** |
||
1398 | * @param array $domainsToRemoveHttpPrefixFromAttributes |
||
1399 | * |
||
1400 | * @return $this |
||
1401 | */ |
||
1402 | 2 | public function setDomainsToRemoveHttpPrefixFromAttributes($domainsToRemoveHttpPrefixFromAttributes): self |
|
1408 | |||
1409 | /** |
||
1410 | * Sum-up extra whitespace from dom-nodes. |
||
1411 | * |
||
1412 | * @param HtmlDomParser $dom |
||
1413 | * |
||
1414 | * @return HtmlDomParser |
||
1415 | */ |
||
1416 | 37 | private function sumUpWhitespace(HtmlDomParser $dom): HtmlDomParser |
|
1443 | |||
1444 | /** |
||
1445 | * WARNING: maybe bad for performance ... |
||
1446 | * |
||
1447 | * @param bool $keepBrokenHtml |
||
1448 | * |
||
1449 | * @return HtmlMin |
||
1450 | */ |
||
1451 | public function useKeepBrokenHtml(bool $keepBrokenHtml): self |
||
1457 | } |
||
1458 |
There are different options of fixing this problem.
If you want to be on the safe side, you can add an additional type-check:
If you are sure that the expression is traversable, you might want to add a doc comment cast to improve IDE auto-completion and static analysis:
Mark the issue as a false-positive: Just hover the remove button, in the top-right corner of this issue for more options.