1
|
|
|
<?php |
2
|
|
|
/** |
3
|
|
|
* Contains the class for replacing image normal image display with a modal. |
4
|
|
|
* |
5
|
|
|
* @copyright (C) 2018, Tobias Oetterer, Paderborn University |
6
|
|
|
* @license https://www.gnu.org/licenses/gpl-3.0.html GNU General Public License, version 3 (or later) |
7
|
|
|
* |
8
|
|
|
* This file is part of the MediaWiki extension BootstrapComponents. |
9
|
|
|
* The BootstrapComponents extension is free software: you can redistribute it |
10
|
|
|
* and/or modify it under the terms of the GNU General Public License as published |
11
|
|
|
* by the Free Software Foundation, either version 3 of the License, or |
12
|
|
|
* (at your option) any later version. |
13
|
|
|
* |
14
|
|
|
* The BootstrapComponents extension is distributed in the hope that it will be useful, |
15
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
16
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
17
|
|
|
* GNU General Public License for more details. |
18
|
|
|
* |
19
|
|
|
* You should have received a copy of the GNU General Public License |
20
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>. |
21
|
|
|
* |
22
|
|
|
* @file |
23
|
|
|
* @ingroup BootstrapComponents |
24
|
|
|
* @author Tobias Oetterer |
25
|
|
|
*/ |
26
|
|
|
|
27
|
|
|
namespace BootstrapComponents; |
28
|
|
|
|
29
|
|
|
use \Linker; |
30
|
|
|
use \Html; |
31
|
|
|
use \MediaWiki\MediaWikiServices; |
32
|
|
|
use \RequestContext; |
33
|
|
|
use \Title; |
34
|
|
|
|
35
|
|
|
/** |
36
|
|
|
* Class ImageModal |
37
|
|
|
* |
38
|
|
|
* @since 1.0 |
39
|
|
|
*/ |
40
|
|
|
class ImageModal implements NestableInterface { |
41
|
|
|
|
42
|
|
|
const CSS_CLASS_PREVENTING_MODAL = 'no-modal'; |
43
|
|
|
|
44
|
|
|
/** |
45
|
|
|
* The components listed here prevent the generation of an image modal. |
46
|
|
|
* @var array |
47
|
|
|
*/ |
48
|
|
|
const PARENTS_PREVENTING_MODAL = [ 'button', 'collapse ', 'image_modal', 'modal', 'popover', 'tooltip' ]; |
49
|
|
|
|
50
|
|
|
/** |
51
|
|
|
* @var \DummyLinker $dummyLinker |
52
|
|
|
*/ |
53
|
|
|
private $dummyLinker; |
54
|
|
|
|
55
|
|
|
/** |
56
|
|
|
* @var \File $file |
57
|
|
|
*/ |
58
|
|
|
private $file; |
59
|
|
|
|
60
|
|
|
/** |
61
|
|
|
* @var string $id |
62
|
|
|
*/ |
63
|
|
|
private $id; |
64
|
|
|
|
65
|
|
|
/** |
66
|
|
|
* @var NestingController $nestingController |
67
|
|
|
*/ |
68
|
|
|
private $nestingController; |
69
|
|
|
|
70
|
|
|
/** |
71
|
|
|
* @var NestableInterface|false $parentComponent |
72
|
|
|
*/ |
73
|
|
|
private $parentComponent; |
74
|
|
|
|
75
|
|
|
/** |
76
|
|
|
* @var ParserOutputHelper $parserOutputHelper |
77
|
|
|
*/ |
78
|
|
|
private $parserOutputHelper; |
79
|
|
|
|
80
|
|
|
/** |
81
|
|
|
* @var bool $disableSourceLink |
82
|
|
|
*/ |
83
|
|
|
private $disableSourceLink; |
84
|
|
|
|
85
|
|
|
/** |
86
|
|
|
* @var Title $title |
87
|
|
|
*/ |
88
|
|
|
private $title; |
89
|
|
|
|
90
|
|
|
/** |
91
|
|
|
* ImageModal constructor. |
92
|
|
|
* |
93
|
|
|
* @param \DummyLinker $dummyLinker |
94
|
|
|
* @param \Title $title |
95
|
|
|
* @param \File $file |
96
|
|
|
* @param NestingController $nestingController |
97
|
|
|
* @param ParserOutputHelper $parserOutputHelper DI for unit testing |
98
|
|
|
* |
99
|
|
|
* @throws \MWException cascading {@see \BootstrapComponents\ApplicationFactory} methods |
100
|
|
|
*/ |
101
|
23 |
|
public function __construct( $dummyLinker, $title, $file, $nestingController = null, $parserOutputHelper = null ) { |
102
|
23 |
|
$this->file = $file; |
103
|
23 |
|
$this->dummyLinker = $dummyLinker; |
104
|
23 |
|
$this->title = $title; |
105
|
|
|
|
106
|
23 |
|
$this->nestingController = is_null( $nestingController ) |
107
|
23 |
|
? ApplicationFactory::getInstance()->getNestingController() |
108
|
1 |
|
: $nestingController; |
109
|
23 |
|
$this->parserOutputHelper = is_null( $parserOutputHelper ) |
110
|
23 |
|
? ApplicationFactory::getInstance()->getParserOutputHelper() |
111
|
11 |
|
: $parserOutputHelper ; |
112
|
|
|
|
113
|
23 |
|
$this->parentComponent = $this->getNestingController()->getCurrentElement(); |
114
|
23 |
|
$this->id = $this->getNestingController()->generateUniqueId( |
115
|
23 |
|
$this->getComponentName() |
116
|
23 |
|
); |
117
|
23 |
|
$this->disableSourceLink = false; |
118
|
23 |
|
} |
119
|
|
|
|
120
|
|
|
/** |
121
|
|
|
* @inheritdoc |
122
|
|
|
*/ |
123
|
23 |
|
public function getComponentName() { |
124
|
23 |
|
return "image_modal"; |
125
|
|
|
} |
126
|
|
|
|
127
|
|
|
/** |
128
|
|
|
* @inheritdoc |
129
|
|
|
*/ |
130
|
15 |
|
public function getId() { |
131
|
15 |
|
return $this->id; |
132
|
|
|
} |
133
|
|
|
|
134
|
|
|
/** |
135
|
|
|
* @param array $frameParams Associative array of parameters external to the media handler. |
136
|
|
|
* Boolean parameters are indicated by presence or absence, the value is arbitrary and |
137
|
|
|
* will often be false. |
138
|
|
|
* thumbnail If present, downscale and frame |
139
|
|
|
* manualthumb Image name to use as a thumbnail, instead of automatic scaling |
140
|
|
|
* framed Shows image in original size in a frame |
141
|
|
|
* frameless Downscale but don't frame |
142
|
|
|
* upright If present, tweak default sizes for portrait orientation |
143
|
|
|
* upright_factor Fudge factor for "upright" tweak (default 0.75) |
144
|
|
|
* border If present, show a border around the image |
145
|
|
|
* align Horizontal alignment (left, right, center, none) |
146
|
|
|
* valign Vertical alignment (baseline, sub, super, top, text-top, middle, |
147
|
|
|
* bottom, text-bottom) |
148
|
|
|
* alt Alternate text for image (i.e. alt attribute). Plain text. |
149
|
|
|
* class HTML for image classes. Plain text. |
150
|
|
|
* caption HTML for image caption. |
151
|
|
|
* link-url URL to link to |
152
|
|
|
* link-title Title object to link to |
153
|
|
|
* link-target Value for the target attribute, only with link-url |
154
|
|
|
* no-link Boolean, suppress description link |
155
|
|
|
* @param array $handlerParams Associative array of media handler parameters, to be passed |
156
|
|
|
* to transform(). Typical keys are "width" and "page". |
157
|
|
|
* @param string|bool $time Timestamp of the file, set as false for current |
158
|
|
|
* @param string $res Final HTML output, used if this returns false |
159
|
|
|
* |
160
|
|
|
* @throws \MWException cascading {@see \BootstrapComponents\NestingController::open} |
161
|
|
|
* @throws \ConfigException cascading {@see \BootstrapComponents\ImageModal::generateTrigger} |
162
|
|
|
* |
163
|
|
|
* @return bool |
164
|
|
|
*/ |
165
|
22 |
|
public function parse( &$frameParams, &$handlerParams, &$time, &$res ) { |
|
|
|
|
166
|
22 |
|
if ( !$this->assertResponsibility( $this->getFile(), $frameParams ) ) { |
167
|
10 |
|
wfDebugLog( 'BootstrapComponents', 'Image modal relegating image rendering back to Linker.php.' ); |
168
|
10 |
|
return true; |
169
|
|
|
} |
170
|
|
|
|
171
|
|
|
// it's on us, let's do some modal-ing |
172
|
15 |
|
$this->augmentParserOutput(); |
173
|
15 |
|
$this->getNestingController()->open( $this ); |
174
|
|
|
|
175
|
15 |
|
$sanitizedFrameParams = $this->sanitizeFrameParams( $frameParams ); |
176
|
15 |
|
$handlerParams['page'] = isset( $handlerParams['page'] ) ? $handlerParams['page'] : false; |
177
|
|
|
|
178
|
15 |
|
$res = $this->turnParamsIntoModal( $sanitizedFrameParams, $handlerParams ); |
179
|
|
|
|
180
|
15 |
|
$this->getNestingController()->close( |
181
|
15 |
|
$this->getId() |
182
|
15 |
|
); |
183
|
|
|
|
184
|
15 |
|
if ( $res === '' ) { |
185
|
|
|
// ImageModal::turnParamsIntoModal returns the empty string, when something went wrong |
186
|
3 |
|
return true; |
187
|
|
|
} |
188
|
12 |
|
return false; |
189
|
|
|
} |
190
|
|
|
|
191
|
|
|
/** |
192
|
|
|
* After this, all bool params ( 'thumbnail', 'framed', 'frameless', 'border' ) are true, if they were present before, false otherwise and all |
193
|
|
|
* string params are set (to the original value or the empty string). |
194
|
|
|
* |
195
|
|
|
* This method is public, because it is used in {@see \BootstrapComponents\Tests\ImageModalTest::doTestCompareTriggerWithOriginalThumb} |
196
|
|
|
* |
197
|
|
|
* @param array $frameParams |
198
|
|
|
* |
199
|
|
|
* @return array |
200
|
|
|
*/ |
201
|
15 |
|
public function sanitizeFrameParams( $frameParams ) { |
202
|
15 |
|
foreach ( [ 'thumbnail', 'framed', 'frameless', 'border' ] as $boolField ) { |
203
|
15 |
|
$frameParams[$boolField] = isset( $frameParams[$boolField] ); |
204
|
15 |
|
} |
205
|
15 |
|
foreach ( [ 'align', 'alt', 'caption', 'class', 'title', 'valign' ] as $stringField ) { |
206
|
15 |
|
$frameParams[$stringField] = !empty( $frameParams[$stringField] ) ? $frameParams[$stringField] : false; |
207
|
15 |
|
} |
208
|
15 |
|
$frameParams['caption'] = $this->preventModalInception( $frameParams['caption'] ); |
209
|
15 |
|
$frameParams['title'] = $this->preventModalInception( $frameParams['title'] ); |
210
|
15 |
|
return $frameParams; |
211
|
|
|
} |
212
|
|
|
|
213
|
|
|
/** |
214
|
|
|
* Disables the source link in modal content. |
215
|
|
|
*/ |
216
|
|
|
public function disableSourceLink() { |
217
|
|
|
$this->disableSourceLink = true; |
218
|
|
|
} |
219
|
|
|
|
220
|
|
|
/** |
221
|
|
|
* Runs various tests, to see, if we delegate processing back to {@see \Linker::makeImageLink} |
222
|
|
|
* After this, we can assume: |
223
|
|
|
* * file is a {@see \File} and exists |
224
|
|
|
* * there is no link param set (link-url, link-title, link-target, no-link) |
225
|
|
|
* * file allows inline display (ref {@see \File::allowInlineDisplay}) |
226
|
|
|
* * we are not inside an image modal or an otherwise compromising component (thanks to {@see ImageModal::getNestingController}) |
227
|
|
|
* * no magic word suppressing image modals is on the page |
228
|
|
|
* * image does not have the "no-modal" class {@see ImageModal::CSS_CLASS_PREVENTING_MODAL} |
229
|
|
|
* |
230
|
|
|
* @param \File $file |
231
|
|
|
* @param array $frameParams |
232
|
|
|
* |
233
|
|
|
* @return bool true, if all assertions hold, false if one fails (see above) |
234
|
|
|
*/ |
235
|
22 |
|
protected function assertResponsibility( $file, $frameParams ) { |
236
|
22 |
|
if ( !$this->assertImageTagValid( $file, $frameParams ) ) { |
237
|
9 |
|
return false; |
238
|
|
|
} |
239
|
20 |
|
return $this->assertImageModalNotSuppressed( $frameParams ); |
240
|
|
|
} |
241
|
|
|
|
242
|
|
|
/** |
243
|
|
|
* @param \File $file |
244
|
|
|
* @param array $sanitizedFrameParams |
245
|
|
|
* @param array $handlerParams |
246
|
|
|
* |
247
|
|
|
* @return array bool|string bool (large image yes or no) |
248
|
|
|
*/ |
249
|
13 |
|
protected function generateContent( $file, $sanitizedFrameParams, $handlerParams ) { |
250
|
|
|
|
251
|
|
|
/** @var \MediaTransformOutput $img $img */ |
252
|
13 |
|
$img = $file->getUnscaledThumb( |
253
|
13 |
|
[ 'page' => $handlerParams['page'] ] |
254
|
13 |
|
); |
255
|
13 |
|
if ( !$img ) { |
256
|
1 |
|
return [ false, false ]; |
257
|
|
|
} |
258
|
|
|
return [ |
259
|
12 |
|
$this->buildContentImageString( $img, $sanitizedFrameParams ), |
260
|
12 |
|
$img->getWidth() > 600 |
261
|
12 |
|
]; |
262
|
|
|
} |
263
|
|
|
|
264
|
|
|
/** |
265
|
|
|
* @return \DummyLinker |
266
|
|
|
*/ |
267
|
|
|
/** @scrutinizer ignore-unused */ |
268
|
|
|
protected function getDummyLinker() { |
269
|
|
|
return $this->dummyLinker; |
270
|
|
|
} |
271
|
|
|
|
272
|
|
|
/** |
273
|
|
|
* @return \File |
274
|
|
|
*/ |
275
|
22 |
|
protected function getFile() { |
276
|
22 |
|
return $this->file; |
277
|
|
|
} |
278
|
|
|
|
279
|
|
|
/** |
280
|
|
|
* @return NestingController |
281
|
|
|
*/ |
282
|
23 |
|
protected function getNestingController() { |
283
|
23 |
|
return $this->nestingController; |
284
|
|
|
} |
285
|
|
|
|
286
|
|
|
/** |
287
|
|
|
* @return null|NestableInterface |
288
|
|
|
*/ |
289
|
20 |
|
protected function getParentComponent() { |
290
|
20 |
|
return $this->parentComponent; |
291
|
|
|
} |
292
|
|
|
|
293
|
|
|
/** |
294
|
|
|
* @return ParserOutputHelper |
295
|
|
|
*/ |
296
|
15 |
|
protected function getParserOutputHelper() { |
297
|
15 |
|
return $this->parserOutputHelper; |
298
|
|
|
} |
299
|
|
|
|
300
|
|
|
/** |
301
|
|
|
* @return Title |
302
|
|
|
*/ |
303
|
12 |
|
protected function getTitle() { |
304
|
12 |
|
return $this->title; |
305
|
|
|
} |
306
|
|
|
|
307
|
|
|
/** |
308
|
|
|
* @param $sanitizedFrameParams |
309
|
|
|
* @param $handlerParams |
310
|
|
|
* |
311
|
|
|
* @throws \ConfigException |
312
|
|
|
* |
313
|
|
|
* @return string rendered modal on success, empty string on failure. |
314
|
|
|
*/ |
315
|
15 |
|
protected function turnParamsIntoModal( $sanitizedFrameParams, $handlerParams ) { |
316
|
15 |
|
$trigger = new ImageModalTrigger( |
317
|
15 |
|
$this->getId(), |
318
|
15 |
|
$this->getFile() |
319
|
15 |
|
); |
320
|
|
|
|
321
|
15 |
|
$triggerString = $trigger->generate( $sanitizedFrameParams, $handlerParams ); |
322
|
|
|
|
323
|
15 |
|
if ( $triggerString === false ) { |
324
|
|
|
// something wrong with the trigger. Relegating back |
325
|
2 |
|
return ''; |
326
|
|
|
} |
327
|
|
|
|
328
|
13 |
|
list ( $content, $largeDialog ) = $this->generateContent( |
329
|
13 |
|
$this->getFile(), |
330
|
13 |
|
$sanitizedFrameParams, |
331
|
|
|
$handlerParams |
332
|
13 |
|
); |
333
|
|
|
|
334
|
13 |
|
if ( $content === false ) { |
335
|
|
|
// could not create content image. Relegating back |
336
|
1 |
|
return ''; |
337
|
|
|
} |
338
|
|
|
|
339
|
12 |
|
$modal = ApplicationFactory::getInstance()->getModalBuilder( |
340
|
12 |
|
$this->getId(), |
341
|
12 |
|
$triggerString, |
342
|
12 |
|
$content, |
343
|
12 |
|
$this->getParserOutputHelper() |
344
|
12 |
|
); |
345
|
12 |
|
$modal->setHeader( |
346
|
12 |
|
$this->getTitle()->getBaseText() |
347
|
12 |
|
); |
348
|
|
|
|
349
|
12 |
|
if ( !$this->disableSourceLink ) { |
350
|
12 |
|
$modal->setFooter( |
351
|
12 |
|
$this->generateButtonToSource( |
352
|
12 |
|
$this->getTitle(), |
353
|
|
|
$handlerParams |
354
|
12 |
|
) |
355
|
12 |
|
); |
356
|
12 |
|
}; |
357
|
|
|
|
358
|
12 |
|
if ( $largeDialog ) { |
359
|
8 |
|
$modal->setDialogClass( 'modal-lg' ); |
360
|
8 |
|
} |
361
|
|
|
|
362
|
12 |
|
return $modal->parse(); |
363
|
|
|
} |
364
|
|
|
|
365
|
|
|
/** |
366
|
|
|
* @param \File $file |
367
|
|
|
* @param array $frameParams |
368
|
|
|
* |
369
|
|
|
* @return bool |
370
|
|
|
*/ |
371
|
22 |
|
private function assertImageTagValid( $file, $frameParams ) { |
372
|
22 |
|
if ( !$file || !$file->exists() ) { |
373
|
8 |
|
return false; |
374
|
|
|
} |
375
|
21 |
|
if ( isset( $frameParams['link-url'] ) || isset( $frameParams['link-title'] ) |
376
|
21 |
|
|| isset( $frameParams['link-target'] ) || isset( $frameParams['no-link'] ) |
377
|
21 |
|
) { |
378
|
2 |
|
return false; |
379
|
|
|
} |
380
|
21 |
|
if ( !$file->allowInlineDisplay() ) { |
381
|
|
|
// let Linker.php handle these cases as well |
382
|
1 |
|
return false; |
383
|
|
|
} |
384
|
20 |
|
return true; |
385
|
|
|
} |
386
|
|
|
|
387
|
|
|
/** |
388
|
|
|
* @param array $frameParams |
389
|
|
|
* |
390
|
|
|
* @return bool |
391
|
|
|
*/ |
392
|
20 |
|
private function assertImageModalNotSuppressed( $frameParams ) { |
393
|
20 |
|
if ( $this->getParentComponent() && in_array( $this->getParentComponent()->getComponentName(), self::PARENTS_PREVENTING_MODAL ) ) { |
394
|
5 |
|
return false; |
395
|
|
|
} |
396
|
16 |
|
if ( isset( $frameParams['class'] ) && in_array( self::CSS_CLASS_PREVENTING_MODAL, explode( ' ', $frameParams['class'] ) ) ) { |
397
|
1 |
|
return false; |
398
|
|
|
} |
399
|
|
|
/** @see ParserOutputHelper::areImageModalsSuppressed as to why we need to use the global parser1 */ |
400
|
16 |
|
$parser = $GLOBALS['wgParser']; |
401
|
|
|
// the is_null test has to be added because otherwise some unit tests will fail |
402
|
16 |
|
return is_null( $parser->getOutput() ) || !$parser->getOutput()->getExtensionData( 'bsc_no_image_modal' ); |
403
|
|
|
} |
404
|
|
|
|
405
|
|
|
/** |
406
|
|
|
* Performs all the mandatory actions on the parser output for the component class |
407
|
|
|
* |
408
|
|
|
* @throws \MWException cascading {@see \BootstrapComponents\ApplicationFactory::getComponentLibrary} |
409
|
|
|
*/ |
410
|
15 |
|
private function augmentParserOutput() { |
411
|
15 |
|
$skin = $this->getParserOutputHelper()->getNameOfActiveSkin(); |
412
|
15 |
|
$this->getParserOutputHelper()->loadBootstrapModules(); |
413
|
15 |
|
$this->getParserOutputHelper()->addModules( |
414
|
15 |
|
ApplicationFactory::getInstance()->getComponentLibrary()->getModulesFor( 'modal', $skin ) |
415
|
15 |
|
); |
416
|
15 |
|
} |
417
|
|
|
|
418
|
|
|
/** |
419
|
|
|
* @param \MediaTransformOutput $img |
420
|
|
|
* @param array $sanitizedFrameParams |
421
|
|
|
* |
422
|
|
|
* @return string |
423
|
|
|
*/ |
424
|
12 |
|
private function buildContentImageString( $img, $sanitizedFrameParams ) { |
425
|
|
|
$imgParams = [ |
426
|
12 |
|
'alt' => $sanitizedFrameParams['alt'], |
427
|
12 |
|
'title' => $sanitizedFrameParams['title'], |
428
|
12 |
|
'img-class' => trim( $sanitizedFrameParams['class'] . ' img-responsive' ), |
429
|
12 |
|
]; |
430
|
12 |
|
$imgString = $img->toHtml( $imgParams ); |
431
|
12 |
|
if ( $sanitizedFrameParams['caption'] ) { |
432
|
3 |
|
$imgString .= ' ' . Html::rawElement( |
433
|
3 |
|
'div', |
434
|
3 |
|
[ 'class' => 'modal-caption' ], |
435
|
3 |
|
$this->sanitizeCaption( $sanitizedFrameParams['caption'] ) |
436
|
3 |
|
); |
437
|
3 |
|
} |
438
|
12 |
|
return $imgString; |
439
|
|
|
} |
440
|
|
|
|
441
|
|
|
/** |
442
|
|
|
* @param Title $title |
443
|
|
|
* @param array $handlerParams |
444
|
|
|
* |
445
|
|
|
* @return string |
446
|
|
|
*/ |
447
|
12 |
|
private function generateButtonToSource( $title, $handlerParams ) { |
448
|
12 |
|
$url = $title->getLocalURL(); |
449
|
12 |
|
if ( isset( $handlerParams['page'] ) ) { |
450
|
12 |
|
$url = wfAppendQuery( $url, [ 'page' => $handlerParams['page'] ] ); |
451
|
12 |
|
} |
452
|
12 |
|
return Html::rawElement( |
453
|
12 |
|
'a', |
454
|
|
|
[ |
455
|
12 |
|
'class' => 'btn btn-primary', |
456
|
12 |
|
'role' => 'button', |
457
|
12 |
|
'href' => $url, |
458
|
12 |
|
], |
459
|
12 |
|
wfMessage( 'bootstrap-components-image-modal-source-button' )->inContentLanguage()->text() |
460
|
12 |
|
); |
461
|
|
|
} |
462
|
|
|
|
463
|
|
|
/** |
464
|
|
|
* We don't want a modal inside a modal. Unfortunately, the caption (and title) are parsed, before the modal is generated. So instead of |
465
|
|
|
* building the modal from the outside, it is build from the inside. This method tries to detect this construct and removes any modal from |
466
|
|
|
* the supplied text and replaces it with the image tag found inside the modal caption content. |
467
|
|
|
* |
468
|
|
|
* @param string $text |
469
|
|
|
* |
470
|
|
|
* @return string |
471
|
|
|
*/ |
472
|
15 |
|
private function preventModalInception( $text ) { |
473
|
15 |
|
if ( preg_match( |
474
|
|
|
'~div class="modal-dialog.+div class="modal-content.+div class="modal-body.+' |
475
|
15 |
|
. '(<img[^>]*/>).+ class="modal-footer.+~Ds', $text, $matches ) ) { |
476
|
|
|
$text = $matches[1]; |
477
|
|
|
} |
478
|
15 |
|
return $text; |
479
|
|
|
} |
480
|
|
|
|
481
|
|
|
/** |
482
|
|
|
* @param string $caption |
483
|
|
|
* |
484
|
|
|
* @return string |
485
|
|
|
*/ |
486
|
3 |
|
private function sanitizeCaption( $caption ) { |
487
|
3 |
|
return preg_replace( '/([^\n])\n([^\n])/m', '\1\2', $caption ); |
488
|
|
|
} |
489
|
|
|
} |
This check looks from parameters that have been defined for a function or method, but which are not used in the method body.