NewContentElementController.php 18.6 KB
Newer Older
1
2
3
<?php
namespace TYPO3\CMS\Backend\Controller\ContentElement;

4
/*
5
 * This file is part of the TYPO3 CMS project.
6
 *
7
8
9
 * It is free software; you can redistribute it and/or modify it under
 * the terms of the GNU General Public License, either version 2
 * of the License, or any later version.
10
 *
11
12
 * For the full copyright and license information, please read the
 * LICENSE.txt file that was distributed with this source code.
13
 *
14
15
 * The TYPO3 project - inspiring people to share!
 */
16

17
18
use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Message\ResponseInterface;
Nicole Cordes's avatar
Nicole Cordes committed
19
20
use TYPO3\CMS\Backend\Utility\BackendUtility;
use TYPO3\CMS\Backend\Utility\IconUtility;
21
22
use TYPO3\CMS\Core\Imaging\Icon;
use TYPO3\CMS\Core\Imaging\IconFactory;
Nicole Cordes's avatar
Nicole Cordes committed
23
use TYPO3\CMS\Core\Utility\GeneralUtility;
24
use TYPO3\CMS\Core\Utility\PathUtility;
Nicole Cordes's avatar
Nicole Cordes committed
25

26
27
28
/**
 * Script Class for the New Content element wizard
 */
29
class NewContentElementController {
30
31

	/**
32
33
	 * Page id
	 *
34
	 * @var int
35
36
37
38
	 */
	public $id;

	/**
39
40
	 * Sys language
	 *
41
	 * @var int
42
43
44
45
	 */
	public $sys_language = 0;

	/**
46
47
	 * Return URL.
	 *
48
	 * @var string
49
50
51
52
	 */
	public $R_URI = '';

	/**
53
54
	 * If set, the content is destined for a specific column.
	 *
55
	 * @var int|null
56
57
58
59
	 */
	public $colPos;

	/**
60
	 * @var int
61
62
63
64
	 */
	public $uid_pid;

	/**
65
66
	 * Module TSconfig.
	 *
67
	 * @var array
68
69
70
71
72
73
	 */
	public $modTSconfig = array();

	/**
	 * Internal backend template object
	 *
74
	 * @var \TYPO3\CMS\Backend\Template\DocumentTemplate
75
76
77
78
	 */
	public $doc;

	/**
79
80
	 * Used to accumulate the content of the module.
	 *
81
	 * @var string
82
83
84
85
	 */
	public $content;

	/**
86
87
	 * Access boolean.
	 *
88
	 * @var bool
89
90
91
92
	 */
	public $access;

	/**
93
94
	 * config of the wizard
	 *
95
	 * @var array
96
97
98
	 */
	public $config;

99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
	/**
	 * @var array
	 */
	protected $pageInfo;

	/**
	 * @var array
	 */
	protected $elementWrapper;

	/**
	 * @var string
	 */
	protected $onClickEvent;

	/**
	 * @var array
	 */
	protected $MCONF;

119
120
121
122
123
124
125
126
	/**
	 * Constructor
	 */
	public function __construct() {
		$GLOBALS['SOBE'] = $this;
		$this->init();
	}

127
128
129
130
131
132
	/**
	 * Constructor, initializing internal variables.
	 *
	 * @return void
	 */
	public function init() {
133
134
		$lang = $this->getLanguageService();
		$lang->includeLLFile('EXT:lang/locallang_misc.xlf');
135
		$LOCAL_LANG_orig = $GLOBALS['LOCAL_LANG'];
Wouter Wolters's avatar
Wouter Wolters committed
136
		$lang->includeLLFile('EXT:backend/Resources/Private/Language/locallang_db_new_content_el.xlf');
137
138
139
		\TYPO3\CMS\Core\Utility\ArrayUtility::mergeRecursiveWithOverrule($LOCAL_LANG_orig, $GLOBALS['LOCAL_LANG']);
		$GLOBALS['LOCAL_LANG'] = $LOCAL_LANG_orig;

140
		// Setting internal vars:
141
142
		$this->id = (int)GeneralUtility::_GP('id');
		$this->sys_language = (int)GeneralUtility::_GP('sys_language_uid');
Nicole Cordes's avatar
Nicole Cordes committed
143
		$this->R_URI = GeneralUtility::sanitizeLocalUrl(GeneralUtility::_GP('returnUrl'));
144
		$this->colPos = GeneralUtility::_GP('colPos') === NULL ? NULL : (int)GeneralUtility::_GP('colPos');
145
		$this->uid_pid = (int)GeneralUtility::_GP('uid_pid');
146
		$this->MCONF['name'] = 'xMOD_db_new_content_el';
Nicole Cordes's avatar
Nicole Cordes committed
147
148
		$this->modTSconfig = BackendUtility::getModTSconfig($this->id, 'mod.wizards.newContentElement');
		$config = BackendUtility::getPagesTSconfig($this->id);
149
150
		$this->config = $config['mod.']['wizards.']['newContentElement.'];
		// Starting the document template object:
151
		$this->doc = GeneralUtility::makeInstance(\TYPO3\CMS\Backend\Template\DocumentTemplate::class);
152
		$this->doc->setModuleTemplate('EXT:backend/Resources/Private/Templates/db_new_content_el.html');
153
154
155
156
157
		$this->doc->JScode = '';
		$this->doc->form = '<form action="" name="editForm"><input type="hidden" name="defValues" value="" />';
		// Setting up the context sensitive menu:
		$this->doc->getContextMenuCode();
		// Getting the current page and receiving access information (used in main())
158
159
160
		$perms_clause = $this->getBackendUser()->getPagePermsClause(1);
		$this->pageInfo = BackendUtility::readPageAccess($this->id, $perms_clause);
		$this->access = is_array($this->pageInfo) ? 1 : 0;
161
162
	}

163
164
165
166
	/**
	 * Injects the request object for the current request or subrequest
	 * As this controller goes only through the main() method, it is rather simple for now
	 *
167
168
169
	 * @param ServerRequestInterface $request the current request
	 * @param ResponseInterface $response
	 * @return ResponseInterface the response with the content
170
	 */
171
	public function mainAction(ServerRequestInterface $request, ResponseInterface $response) {
172
173
174
175
176
177
		$this->main();

		$response->getBody()->write($this->content);
		return $response;
	}

178
179
180
181
182
183
	/**
	 * Creating the module output.
	 *
	 * @return void
	 */
	public function main() {
184
		$lang = $this->getLanguageService();
185
186
		if ($this->id && $this->access) {
			// Init position map object:
187
			$posMap = GeneralUtility::makeInstance(\TYPO3\CMS\Backend\Tree\View\ContentCreationPagePositionMap::class);
188
189
			$posMap->cur_sys_language = $this->sys_language;
			// If a column is pre-set:
190
			if (isset($this->colPos)) {
191
192
193
194
195
196
197
198
199
200
201
202
203
				if ($this->uid_pid < 0) {
					$row = array();
					$row['uid'] = abs($this->uid_pid);
				} else {
					$row = '';
				}
				$this->onClickEvent = $posMap->onClickInsertRecord($row, $this->colPos, '', $this->uid_pid, $this->sys_language);
			} else {
				$this->onClickEvent = '';
			}
			// ***************************
			// Creating content
			// ***************************
204
			$this->content .= $this->doc->header($lang->getLL('newContentElement'));
205
			// Wizard
206
			$wizardItems = $this->wizardArray();
207
			// Wrapper for wizards
208
			$this->elementWrapper['section'] = array('', '');
209
210
211
			// Hook for manipulating wizardItems, wrapper, onClickEvent etc.
			if (is_array($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['cms']['db_new_content_el']['wizardItemsHook'])) {
				foreach ($GLOBALS['TYPO3_CONF_VARS']['SC_OPTIONS']['cms']['db_new_content_el']['wizardItemsHook'] as $classData) {
Nicole Cordes's avatar
Nicole Cordes committed
212
					$hookObject = GeneralUtility::getUserObj($classData);
213
					if (!$hookObject instanceof \TYPO3\CMS\Backend\Wizard\NewContentElementWizardHookInterface) {
214
						throw new \UnexpectedValueException('$hookObject must implement interface ' . \TYPO3\CMS\Backend\Wizard\NewContentElementWizardHookInterface::class, 1227834741);
215
216
217
218
219
					}
					$hookObject->manipulateWizardItems($wizardItems, $this);
				}
			}
			// Add document inline javascript
220
			$this->doc->JScode = $this->doc->wrapScriptTags('
221
				function goToalt_doc() {	//
222
					' . $this->onClickEvent . '
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
				}

				if(top.refreshMenu) {
					top.refreshMenu();
				} else {
					top.TYPO3ModuleMenu.refreshMenu();
				}
			');
			// Traverse items for the wizard.
			// An item is either a header or an item rendered with a radio button and title/description and icon:
			$cc = ($key = 0);
			$menuItems = array();
			foreach ($wizardItems as $k => $wInfo) {
				if ($wInfo['header']) {
					$menuItems[] = array(
						'label' => htmlspecialchars($wInfo['header']),
						'content' => $this->elementWrapper['section'][0]
					);
					$key = count($menuItems) - 1;
				} else {
					$content = '';
244
245
246

					if (!$this->onClickEvent) {
						// Radio button:
247
						$oC = 'document.editForm.defValues.value=unescape(' . GeneralUtility::quoteJSvalue(rawurlencode($wInfo['params'])) . ');goToalt_doc();' . (!$this->onClickEvent ? 'window.location.hash=\'#sel2\';' : '');
248
						$content .= '<div class="media-left"><input type="radio" name="tempB" value="' . htmlspecialchars($k) . '" onclick="' . htmlspecialchars($oC) . '" /></div>';
249
250
251
						// Onclick action for icon/title:
						$aOnClick = 'document.getElementsByName(\'tempB\')[' . $cc . '].checked=1;' . $oC . 'return false;';
					} else {
252
						$aOnClick = "document.editForm.defValues.value=unescape('" . rawurlencode($wInfo['params']) . "');goToalt_doc();" . (!$this->onClickEvent?"window.location.hash='#sel2';":'');
253
254
					}

255
					$icon = $wInfo['icon'];
256
					if (strpos($wInfo['icon'], '..') === FALSE && !GeneralUtility::isAbsPath($icon)) {
257
						$icon = GeneralUtility::getFileAbsFileName($icon, TRUE, TRUE);
258
259
260
						$pathInfo = PathUtility::pathinfo($icon);
						$path = PathUtility::getRelativePathTo($pathInfo['dirname']);
						$icon = $path . $pathInfo['basename'];
261
					}
262
263
					$menuItems[$key]['content'] .= '
						<div class="media">
264
265
266
							<a href="#" onclick="' . htmlspecialchars($aOnClick) . '">
								' . $content . '
								<div class="media-left">
267
									<img' . IconUtility::skinImg('', $icon) . ' alt="" />
268
269
270
271
272
273
274
								</div>
								<div class="media-body">
									<strong>' . htmlspecialchars($wInfo['title']) . '</strong>' .
									'<br />' .
									nl2br(htmlspecialchars(trim($wInfo['description']))) .
								'</div>
							</a>
275
						</div>';
276
277
278
279
280
281
282
					$cc++;
				}
			}
			// Add closing section-tag
			foreach ($menuItems as $key => $val) {
				$menuItems[$key]['content'] .= $this->elementWrapper['section'][1];
			}
283
284
285
			// Add the wizard table to the content, wrapped in tabs
			$code = '<p>' . $lang->getLL('sel1', 1) . '</p>' . $this->doc->getDynamicTabMenu($menuItems, 'new-content-element-wizard');

286
			$this->content .= $this->doc->section(!$this->onClickEvent ? $lang->getLL('1_selectType') : '', $code, 0, 1);
287
288
289
290
291
			// If the user must also select a column:
			if (!$this->onClickEvent) {
				// Add anchor "sel2"
				$this->content .= $this->doc->section('', '<a name="sel2"></a>');
				// Select position
292
				$code = '<p>' . $lang->getLL('sel2', 1) . '</p>';
293

294
				// Load SHARED page-TSconfig settings and retrieve column list from there, if applicable:
295
				$colPosArray = GeneralUtility::callUserFunction(\TYPO3\CMS\Backend\View\BackendLayoutView::class . '->getColPosListItemsParsed', $this->id, $this);
296
				$colPosIds = array_column($colPosArray, 1);
297
				// Removing duplicates, if any
298
				$colPosList = implode(',', array_unique(array_map('intval', $colPosIds)));
299
300
				// Finally, add the content of the column selector to the content:
				$code .= $posMap->printContentElementColumns($this->id, 0, $colPosList, 1, $this->R_URI);
301
				$this->content .= $this->doc->section($lang->getLL('2_selectPosition'), $code, 0, 1);
302
303
304
305
			}
		} else {
			// In case of no access:
			$this->content = '';
306
			$this->content .= $this->doc->header($lang->getLL('newContentElement'));
307
308
309
310
311
312
313
			$this->content .= $this->doc->spacer(5);
		}
		// Setting up the buttons and markers for docheader
		$docHeaderButtons = $this->getButtons();
		$markers['CSH'] = $docHeaderButtons['csh'];
		$markers['CONTENT'] = $this->content;
		// Build the <body> for the module
314
315
		$this->content = $this->doc->startPage($lang->getLL('newContentElement'));
		$this->content .= $this->doc->moduleBody($this->pageInfo, $docHeaderButtons, $markers);
316
317
318
319
320
321
322
323
324
		$this->content .= $this->doc->sectionEnd();
		$this->content .= $this->doc->endPage();
		$this->content = $this->doc->insertStylesAndJS($this->content);
	}

	/**
	 * Print out the accumulated content:
	 *
	 * @return void
325
	 * @deprecated since TYPO3 CMS 7, will be removed in TYPO3 CMS 8, use mainAction() instead
326
327
	 */
	public function printContent() {
328
		GeneralUtility::logDeprecatedFunction();
329
330
331
332
333
334
335
336
337
338
339
340
341
		echo $this->content;
	}

	/**
	 * Create the panel of buttons for submitting the form or otherwise perform operations.
	 *
	 * @return array All available buttons as an assoc. array
	 */
	protected function getButtons() {
		$buttons = array(
			'csh' => '',
			'back' => ''
		);
342
		$iconFactory = GeneralUtility::makeInstance(IconFactory::class);
343
		if ($this->id && $this->access) {
Benni Mack's avatar
Benni Mack committed
344
			$buttons['csh'] = BackendUtility::cshItem('xMOD_csh_corebe', 'new_ce');
345
			if ($this->R_URI) {
346
				$buttons['back'] = '<a href="' . htmlspecialchars($this->R_URI) . '" class="typo3-goBack" title="' . $this->getLanguageService()->getLL('goBack', TRUE) . '">' . $iconFactory->getIcon('actions-view-go-back', Icon::SIZE_SMALL) . '</a>';
347
348
349
350
351
352
353
354
355
356
357
358
359
360
			}
		}
		return $buttons;
	}

	/***************************
	 *
	 * OTHER FUNCTIONS:
	 *
	 ***************************/
	/**
	 * Returns the content of wizardArray() function...
	 *
	 * @return array Returns the content of wizardArray() function...
361
	 * @deprecated since TYPO3 CMS 7, will be removed in TYPO3 CMS 8, use "wizardArray()" directly
362
363
	 */
	public function getWizardItems() {
364
		GeneralUtility::logDeprecatedFunction();
365
366
367
368
369
370
371
372
373
374
		return $this->wizardArray();
	}

	/**
	 * Returns the array of elements in the wizard display.
	 * For the plugin section there is support for adding elements there from a global variable.
	 *
	 * @return array
	 */
	public function wizardArray() {
375
		$wizardItems = array();
376
377
		if (is_array($this->config)) {
			$wizards = $this->config['wizardItems.'];
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
			$appendWizards = $this->wizard_appendWizards($wizards['elements.']);
			if (is_array($wizards)) {
				foreach ($wizards as $groupKey => $wizardGroup) {
					$groupKey = rtrim($groupKey, '.');
					$showItems = GeneralUtility::trimExplode(',', $wizardGroup['show'], TRUE);
					$showAll = $wizardGroup['show'] === '*';
					$groupItems = array();
					if (is_array($appendWizards[$groupKey . '.']['elements.'])) {
						$wizardElements = array_merge((array)$wizardGroup['elements.'], $appendWizards[$groupKey . '.']['elements.']);
					} else {
						$wizardElements = $wizardGroup['elements.'];
					}
					if (is_array($wizardElements)) {
						foreach ($wizardElements as $itemKey => $itemConf) {
							$itemKey = rtrim($itemKey, '.');
							if ($showAll || in_array($itemKey, $showItems)) {
								$tmpItem = $this->wizard_getItem($groupKey, $itemKey, $itemConf);
								if ($tmpItem) {
									$groupItems[$groupKey . '_' . $itemKey] = $tmpItem;
								}
398
399
400
							}
						}
					}
401
					if (!empty($groupItems)) {
402
403
404
						$wizardItems[$groupKey] = $this->wizard_getGroupHeader($groupKey, $wizardGroup);
						$wizardItems = array_merge($wizardItems, $groupItems);
					}
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
				}
			}
		}
		// Remove elements where preset values are not allowed:
		$this->removeInvalidElements($wizardItems);
		return $wizardItems;
	}

	/**
	 * @param mixed $wizardElements
	 * @return array
	 */
	public function wizard_appendWizards($wizardElements) {
		if (!is_array($wizardElements)) {
			$wizardElements = array();
		}
		if (is_array($GLOBALS['TBE_MODULES_EXT']['xMOD_db_new_content_el']['addElClasses'])) {
			foreach ($GLOBALS['TBE_MODULES_EXT']['xMOD_db_new_content_el']['addElClasses'] as $class => $path) {
				require_once $path;
Nicole Cordes's avatar
Nicole Cordes committed
424
				$modObj = GeneralUtility::makeInstance($class);
425
426
427
428
429
430
431
432
433
434
435
436
437
				$wizardElements = $modObj->proc($wizardElements);
			}
		}
		$returnElements = array();
		foreach ($wizardElements as $key => $wizardItem) {
			preg_match('/^[a-zA-Z0-9]+_/', $key, $group);
			$wizardGroup = $group[0] ? substr($group[0], 0, -1) . '.' : $key;
			$returnElements[$wizardGroup]['elements.'][substr($key, strlen($wizardGroup)) . '.'] = $wizardItem;
		}
		return $returnElements;
	}

	/**
438
439
	 * @param string $groupKey Not used
	 * @param string $itemKey Not used
440
441
442
443
	 * @param array $itemConf
	 * @return array
	 */
	public function wizard_getItem($groupKey, $itemKey, $itemConf) {
444
445
		$itemConf['title'] = $this->getLanguageService()->sL($itemConf['title']);
		$itemConf['description'] = $this->getLanguageService()->sL($itemConf['description']);
446
447
448
449
450
451
		$itemConf['tt_content_defValues'] = $itemConf['tt_content_defValues.'];
		unset($itemConf['tt_content_defValues.']);
		return $itemConf;
	}

	/**
452
	 * @param string $groupKey Not used
453
454
455
456
457
	 * @param array $wizardGroup
	 * @return array
	 */
	public function wizard_getGroupHeader($groupKey, $wizardGroup) {
		return array(
458
			'header' => $this->getLanguageService()->sL($wizardGroup['header'])
459
460
461
462
463
		);
	}

	/**
	 * Checks the array for elements which might contain unallowed default values and will unset them!
464
465
	 * Looks for the "tt_content_defValues" key in each element and if found it will traverse that array as fieldname / value pairs and check.
	 * The values will be added to the "params" key of the array (which should probably be unset or empty by default).
466
467
468
469
470
471
472
	 *
	 * @param array $wizardItems Wizard items, passed by reference
	 * @return void
	 */
	public function removeInvalidElements(&$wizardItems) {
		// Get TCEFORM from TSconfig of current page
		$row = array('pid' => $this->id);
Nicole Cordes's avatar
Nicole Cordes committed
473
		$TCEFORM_TSconfig = BackendUtility::getTCEFORM_TSconfig('tt_content', $row);
474
475
476
477
478
479
		$headersUsed = array();
		// Traverse wizard items:
		foreach ($wizardItems as $key => $cfg) {
			// Exploding parameter string, if any (old style)
			if ($wizardItems[$key]['params']) {
				// Explode GET vars recursively
Nicole Cordes's avatar
Nicole Cordes committed
480
				$tempGetVars = GeneralUtility::explodeUrl2Array($wizardItems[$key]['params'], TRUE);
481
482
483
				// If tt_content values are set, merge them into the tt_content_defValues array,
				// unset them from $tempGetVars and re-implode $tempGetVars into the param string
				// (in case remaining parameters are around).
484
				if (is_array($tempGetVars['defVals']['tt_content'])) {
485
486
487
488
489
490
					$wizardItems[$key]['tt_content_defValues'] = array_merge(
						is_array($wizardItems[$key]['tt_content_defValues'])
							? $wizardItems[$key]['tt_content_defValues']
							: array(),
						$tempGetVars['defVals']['tt_content']
					);
491
					unset($tempGetVars['defVals']['tt_content']);
Nicole Cordes's avatar
Nicole Cordes committed
492
					$wizardItems[$key]['params'] = GeneralUtility::implodeArrayForUrl('', $tempGetVars);
493
494
495
496
				}
			}
			// If tt_content_defValues are defined...:
			if (is_array($wizardItems[$key]['tt_content_defValues'])) {
497
				$backendUser = $this->getBackendUser();
498
499
500
501
				// Traverse field values:
				foreach ($wizardItems[$key]['tt_content_defValues'] as $fN => $fV) {
					if (is_array($GLOBALS['TCA']['tt_content']['columns'][$fN])) {
						// Get information about if the field value is OK:
502
						$config = &$GLOBALS['TCA']['tt_content']['columns'][$fN]['config'];
503
						$authModeDeny = $config['type'] == 'select' && $config['authMode']
504
							&& !$backendUser->checkAuthMode('tt_content', $fN, $fV, $config['authMode']);
505
506
507
508
509
510
511
						// explode TSconfig keys only as needed
						if (!isset($removeItems[$fN])) {
							$removeItems[$fN] = GeneralUtility::trimExplode(',', $TCEFORM_TSconfig[$fN]['removeItems'], TRUE);
						}
						if (!isset($keepItems[$fN])) {
							$keepItems[$fN] = GeneralUtility::trimExplode(',', $TCEFORM_TSconfig[$fN]['keepItems'], TRUE);
						}
512
						$isNotInKeepItems = !empty($keepItems[$fN]) && !in_array($fV, $keepItems[$fN]);
513
						if ($authModeDeny || $fN === 'CType' && in_array($fV, $removeItems[$fN]) || $isNotInKeepItems) {
514
515
516
517
518
							// Remove element all together:
							unset($wizardItems[$key]);
							break;
						} else {
							// Add the parameter:
519
							$wizardItems[$key]['params'] .= '&defVals[tt_content][' . $fN . ']=' . rawurlencode($fV);
520
521
522
523
524
525
526
527
528
529
							$tmp = explode('_', $key);
							$headersUsed[$tmp[0]] = $tmp[0];
						}
					}
				}
			}
		}
		// remove headers without elements
		foreach ($wizardItems as $key => $cfg) {
			$tmp = explode('_', $key);
530
			if ($tmp[0] && !$tmp[1] && !in_array($tmp[0], $headersUsed)) {
531
532
533
534
535
				unset($wizardItems[$key]);
			}
		}
	}

536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
	/**
	 * Returns LanguageService
	 *
	 * @return \TYPO3\CMS\Lang\LanguageService
	 */
	protected function getLanguageService() {
		return $GLOBALS['LANG'];
	}

	/**
	 * Returns the current BE user.
	 *
	 * @return \TYPO3\CMS\Core\Authentication\BackendUserAuthentication
	 */
	protected function getBackendUser() {
		return $GLOBALS['BE_USER'];
	}

554
}