NewContentElementController.php 18.1 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

Nicole Cordes's avatar
Nicole Cordes committed
17
18
19
20
use TYPO3\CMS\Backend\Utility\BackendUtility;
use TYPO3\CMS\Backend\Utility\IconUtility;
use TYPO3\CMS\Core\Utility\GeneralUtility;

21
22
23
24
25
26
/**
 * Script Class for the New Content element wizard
 */
class NewContentElementController {

	/**
27
28
	 * Page id
	 *
29
	 * @var int
30
31
32
33
	 */
	public $id;

	/**
34
35
	 * Sys language
	 *
36
	 * @var int
37
38
39
40
	 */
	public $sys_language = 0;

	/**
41
42
	 * Return URL.
	 *
43
	 * @var string
44
45
46
47
	 */
	public $R_URI = '';

	/**
48
49
	 * If set, the content is destined for a specific column.
	 *
50
	 * @var int|null
51
52
53
54
	 */
	public $colPos;

	/**
55
	 * @var int
56
57
58
59
	 */
	public $uid_pid;

	/**
60
61
	 * Module TSconfig.
	 *
62
	 * @var array
63
64
65
66
67
68
	 */
	public $modTSconfig = array();

	/**
	 * Internal backend template object
	 *
69
	 * @var \TYPO3\CMS\Backend\Template\DocumentTemplate
70
71
72
73
	 */
	public $doc;

	/**
74
75
	 * Used to accumulate the content of the module.
	 *
76
	 * @var string
77
78
79
80
	 */
	public $content;

	/**
81
82
	 * Access boolean.
	 *
83
	 * @var bool
84
85
86
87
	 */
	public $access;

	/**
88
89
	 * config of the wizard
	 *
90
	 * @var array
91
92
93
	 */
	public $config;

94
95
96
97
98
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 array
	 */
	protected $elementWrapperForTabs;

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

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

119
120
121
122
123
124
	/**
	 * Constructor, initializing internal variables.
	 *
	 * @return void
	 */
	public function init() {
125
126
		$lang = $this->getLanguageService();
		$lang->includeLLFile('EXT:lang/locallang_misc.xlf');
127
		$LOCAL_LANG_orig = $GLOBALS['LOCAL_LANG'];
Wouter Wolters's avatar
Wouter Wolters committed
128
		$lang->includeLLFile('EXT:backend/Resources/Private/Language/locallang_db_new_content_el.xlf');
129
130
131
		\TYPO3\CMS\Core\Utility\ArrayUtility::mergeRecursiveWithOverrule($LOCAL_LANG_orig, $GLOBALS['LOCAL_LANG']);
		$GLOBALS['LOCAL_LANG'] = $LOCAL_LANG_orig;

132
		// Setting internal vars:
133
134
		$this->id = (int)GeneralUtility::_GP('id');
		$this->sys_language = (int)GeneralUtility::_GP('sys_language_uid');
Nicole Cordes's avatar
Nicole Cordes committed
135
		$this->R_URI = GeneralUtility::sanitizeLocalUrl(GeneralUtility::_GP('returnUrl'));
136
		$this->colPos = GeneralUtility::_GP('colPos') === NULL ? NULL : (int)GeneralUtility::_GP('colPos');
137
		$this->uid_pid = (int)GeneralUtility::_GP('uid_pid');
138
		$this->MCONF['name'] = 'xMOD_db_new_content_el';
Nicole Cordes's avatar
Nicole Cordes committed
139
140
		$this->modTSconfig = BackendUtility::getModTSconfig($this->id, 'mod.wizards.newContentElement');
		$config = BackendUtility::getPagesTSconfig($this->id);
141
142
		$this->config = $config['mod.']['wizards.']['newContentElement.'];
		// Starting the document template object:
143
		$this->doc = GeneralUtility::makeInstance(\TYPO3\CMS\Backend\Template\DocumentTemplate::class);
144
		$this->doc->backPath = $GLOBALS['BACK_PATH'];
145
		$this->doc->setModuleTemplate('EXT:backend/Resources/Private/Templates/db_new_content_el.html');
146
147
148
149
150
		$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())
151
152
153
		$perms_clause = $this->getBackendUser()->getPagePermsClause(1);
		$this->pageInfo = BackendUtility::readPageAccess($this->id, $perms_clause);
		$this->access = is_array($this->pageInfo) ? 1 : 0;
154
155
156
157
158
159
160
161
	}

	/**
	 * Creating the module output.
	 *
	 * @return void
	 */
	public function main() {
162
		$lang = $this->getLanguageService();
163
164
		if ($this->id && $this->access) {
			// Init position map object:
165
			$posMap = GeneralUtility::makeInstance(\TYPO3\CMS\Backend\Tree\View\ContentCreationPagePositionMap::class);
166
167
168
			$posMap->cur_sys_language = $this->sys_language;
			$posMap->backPath = $GLOBALS['BACK_PATH'];
			// If a column is pre-set:
169
			if (isset($this->colPos)) {
170
171
172
173
174
175
176
177
178
179
180
181
182
				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
			// ***************************
183
			$this->content .= $this->doc->header($lang->getLL('newContentElement'));
184
185
186
			// Wizard
			$wizardItems = $this->getWizardItems();
			// Wrapper for wizards
187
			$this->elementWrapper['section'] = array('', '');
188
189
190
191
192
			// Copy wrapper for tabs
			$this->elementWrapperForTabs = $this->elementWrapper;
			// 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
193
					$hookObject = GeneralUtility::getUserObj($classData);
194
					if (!$hookObject instanceof \TYPO3\CMS\Backend\Wizard\NewContentElementWizardHookInterface) {
195
						throw new \UnexpectedValueException('$hookObject must implement interface ' . \TYPO3\CMS\Backend\Wizard\NewContentElementWizardHookInterface::class, 1227834741);
196
197
198
199
200
201
202
203
204
					}
					$hookObject->manipulateWizardItems($wizardItems, $this);
				}
			}
			if ($this->config['renderMode'] == 'tabs' && $this->elementWrapperForTabs != $this->elementWrapper) {
				// Restore wrapper for tabs if they are overwritten in hook
				$this->elementWrapper = $this->elementWrapperForTabs;
			}
			// Add document inline javascript
205
			$this->doc->JScode = $this->doc->wrapScriptTags('
206
				function goToalt_doc() {	//
207
					' . $this->onClickEvent . '
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
				}

				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 = '';
229
230
231

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

240
241
242
243
244
					$icon = $wInfo['icon'];
					if (strpos($wInfo['icon'], '..') === FALSE) {
						$icon = GeneralUtility::getFileAbsFileName($icon, TRUE, TRUE);
						$icon = substr($icon, strlen(PATH_typo3));
					}
245
246
247
248
249
					$menuItems[$key]['content'] .= '
						<div class="media">
							' . $content . '
							<div class="media-left">
								<a href="#" onclick="' . htmlspecialchars($aOnClick) . '">
250
									<img' . IconUtility::skinImg($this->doc->backPath, $icon) . ' alt="" />
251
								</a>
252
							</div>
253
254
255
256
257
258
259
							<div class="media-body">
								<a href="#" onclick="' . htmlspecialchars($aOnClick) . '">
									<strong>' . htmlspecialchars($wInfo['title']) . '</strong>
									<br />' . nl2br(htmlspecialchars(trim($wInfo['description']))) .
								'</a>
							</div>
						</div>';
260
261
262
263
264
265
266
267
268
					$cc++;
				}
			}
			// Add closing section-tag
			foreach ($menuItems as $key => $val) {
				$menuItems[$key]['content'] .= $this->elementWrapper['section'][1];
			}
			// Add the wizard table to the content, wrapped in tabs:
			if ($this->config['renderMode'] == 'tabs') {
269
				$code = '<p>' . $lang->getLL('sel1', 1) . '</p>' . $this->doc->getDynamicTabMenu($menuItems, 'new-content-element-wizard');
270
			} else {
271
				$code = '<p>' . $lang->getLL('sel1', 1) . '</p>';
272
				foreach ($menuItems as $section) {
273
					$code .= '<h3 class="divider">' . $section['label'] . '</h3>' . $section['content'];
274
275
				}
			}
276
			$this->content .= $this->doc->section(!$this->onClickEvent ? $lang->getLL('1_selectType') : '', $code, 0, 1);
277
278
279
280
281
			// 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
282
				$code = '<p>' . $lang->getLL('sel2', 1) . '</p>';
283

284
				// Load SHARED page-TSconfig settings and retrieve column list from there, if applicable:
285
				$colPosArray = GeneralUtility::callUserFunction(\TYPO3\CMS\Backend\View\BackendLayoutView::class . '->getColPosListItemsParsed', $this->id, $this);
286
				$colPosIds = array_column($colPosArray, 1);
287
				// Removing duplicates, if any
288
				$colPosList = implode(',', array_unique(array_map('intval', $colPosIds)));
289
290
				// Finally, add the content of the column selector to the content:
				$code .= $posMap->printContentElementColumns($this->id, 0, $colPosList, 1, $this->R_URI);
291
				$this->content .= $this->doc->section($lang->getLL('2_selectPosition'), $code, 0, 1);
292
293
294
295
			}
		} else {
			// In case of no access:
			$this->content = '';
296
			$this->content .= $this->doc->header($lang->getLL('newContentElement'));
297
298
299
300
301
302
303
			$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
304
305
		$this->content = $this->doc->startPage($lang->getLL('newContentElement'));
		$this->content .= $this->doc->moduleBody($this->pageInfo, $docHeaderButtons, $markers);
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
		$this->content .= $this->doc->sectionEnd();
		$this->content .= $this->doc->endPage();
		$this->content = $this->doc->insertStylesAndJS($this->content);
	}

	/**
	 * Print out the accumulated content:
	 *
	 * @return void
	 */
	public function printContent() {
		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' => ''
		);
		if ($this->id && $this->access) {
Benni Mack's avatar
Benni Mack committed
331
			$buttons['csh'] = BackendUtility::cshItem('xMOD_csh_corebe', 'new_ce');
332
			if ($this->R_URI) {
333
				$buttons['back'] = '<a href="' . htmlspecialchars($this->R_URI) . '" class="typo3-goBack" title="' . $this->getLanguageService()->getLL('goBack', TRUE) . '">' . IconUtility::getSpriteIcon('actions-view-go-back') . '</a>';
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
			}
		}
		return $buttons;
	}

	/***************************
	 *
	 * OTHER FUNCTIONS:
	 *
	 ***************************/
	/**
	 * Returns the content of wizardArray() function...
	 *
	 * @return array Returns the content of wizardArray() function...
	 */
	public function getWizardItems() {
		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() {
360
		$wizardItems = array();
361
362
		if (is_array($this->config)) {
			$wizards = $this->config['wizardItems.'];
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
			$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;
								}
383
384
385
							}
						}
					}
386
					if (!empty($groupItems)) {
387
388
389
						$wizardItems[$groupKey] = $this->wizard_getGroupHeader($groupKey, $wizardGroup);
						$wizardItems = array_merge($wizardItems, $groupItems);
					}
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
				}
			}
		}
		// 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
409
				$modObj = GeneralUtility::makeInstance($class);
410
411
412
413
414
415
416
417
418
419
420
421
422
				$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;
	}

	/**
423
424
	 * @param string $groupKey Not used
	 * @param string $itemKey Not used
425
426
427
428
	 * @param array $itemConf
	 * @return array
	 */
	public function wizard_getItem($groupKey, $itemKey, $itemConf) {
429
430
		$itemConf['title'] = $this->getLanguageService()->sL($itemConf['title']);
		$itemConf['description'] = $this->getLanguageService()->sL($itemConf['description']);
431
432
433
434
435
436
		$itemConf['tt_content_defValues'] = $itemConf['tt_content_defValues.'];
		unset($itemConf['tt_content_defValues.']);
		return $itemConf;
	}

	/**
437
	 * @param string $groupKey Not used
438
439
440
441
442
	 * @param array $wizardGroup
	 * @return array
	 */
	public function wizard_getGroupHeader($groupKey, $wizardGroup) {
		return array(
443
			'header' => $this->getLanguageService()->sL($wizardGroup['header'])
444
445
446
447
448
		);
	}

	/**
	 * Checks the array for elements which might contain unallowed default values and will unset them!
449
450
	 * 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).
451
452
453
454
455
456
457
	 *
	 * @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
458
		$TCEFORM_TSconfig = BackendUtility::getTCEFORM_TSconfig('tt_content', $row);
459
460
461
462
463
464
		$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
465
				$tempGetVars = GeneralUtility::explodeUrl2Array($wizardItems[$key]['params'], TRUE);
466
467
468
				// 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).
469
				if (is_array($tempGetVars['defVals']['tt_content'])) {
470
471
472
473
474
475
					$wizardItems[$key]['tt_content_defValues'] = array_merge(
						is_array($wizardItems[$key]['tt_content_defValues'])
							? $wizardItems[$key]['tt_content_defValues']
							: array(),
						$tempGetVars['defVals']['tt_content']
					);
476
					unset($tempGetVars['defVals']['tt_content']);
Nicole Cordes's avatar
Nicole Cordes committed
477
					$wizardItems[$key]['params'] = GeneralUtility::implodeArrayForUrl('', $tempGetVars);
478
479
480
481
				}
			}
			// If tt_content_defValues are defined...:
			if (is_array($wizardItems[$key]['tt_content_defValues'])) {
482
				$backendUser = $this->getBackendUser();
483
484
485
486
				// 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:
487
						$config = &$GLOBALS['TCA']['tt_content']['columns'][$fN]['config'];
488
						$authModeDeny = $config['type'] == 'select' && $config['authMode']
489
							&& !$backendUser->checkAuthMode('tt_content', $fN, $fV, $config['authMode']);
490
491
492
493
494
495
496
						// 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);
						}
497
						$isNotInKeepItems = !empty($keepItems[$fN]) && !in_array($fV, $keepItems[$fN]);
498
						if ($authModeDeny || $fN === 'CType' && in_array($fV, $removeItems[$fN]) || $isNotInKeepItems) {
499
500
501
502
503
							// Remove element all together:
							unset($wizardItems[$key]);
							break;
						} else {
							// Add the parameter:
504
							$wizardItems[$key]['params'] .= '&defVals[tt_content][' . $fN . ']=' . rawurlencode($fV);
505
506
507
508
509
510
511
512
513
514
							$tmp = explode('_', $key);
							$headersUsed[$tmp[0]] = $tmp[0];
						}
					}
				}
			}
		}
		// remove headers without elements
		foreach ($wizardItems as $key => $cfg) {
			$tmp = explode('_', $key);
515
			if ($tmp[0] && !$tmp[1] && !in_array($tmp[0], $headersUsed)) {
516
517
518
519
520
				unset($wizardItems[$key]);
			}
		}
	}

521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
	/**
	 * 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'];
	}

539
}