From 8943e7ed6d1dba1d49e76fd6c2f6b30c16438017 Mon Sep 17 00:00:00 2001
From: Fabian Galinski <fabian@sgalinski.de>
Date: Thu, 14 Sep 2017 02:55:27 +0200
Subject: [PATCH] [FEATURE] Better implementation for the TYPO3 8 integration

---
 .../Form/Element/CkEditorRichTextElement.php  | 376 ------------------
 ...ichTextElement.php => RichTextElement.php} | 100 ++---
 .../Form/Resolver/RichTextNodeResolver.php    |   7 +-
 3 files changed, 53 insertions(+), 430 deletions(-)
 delete mode 100644 Classes/Form/Element/CkEditorRichTextElement.php
 rename Classes/Form/Element/{RtehtmlareaRichTextElement.php => RichTextElement.php} (95%)

diff --git a/Classes/Form/Element/CkEditorRichTextElement.php b/Classes/Form/Element/CkEditorRichTextElement.php
deleted file mode 100644
index 4a2336b..0000000
--- a/Classes/Form/Element/CkEditorRichTextElement.php
+++ /dev/null
@@ -1,376 +0,0 @@
-<?php
-declare(strict_types=1);
-
-namespace SGalinski\Tinymce4Rte\Form\Element;
-
-/*
- * This file is part of the TYPO3 CMS project.
- *
- * 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.
- *
- * For the full copyright and license information, please read the
- * LICENSE.txt file that was distributed with this source code.
- *
- * The TYPO3 project - inspiring people to share!
- */
-
-use SGalinski\Tinymce\Loader;
-use TYPO3\CMS\Backend\Form\Element\AbstractFormElement;
-use TYPO3\CMS\Backend\Routing\UriBuilder;
-use TYPO3\CMS\Core\Authentication\BackendUserAuthentication;
-use TYPO3\CMS\Core\Localization\Locales;
-use TYPO3\CMS\Core\Utility\GeneralUtility;
-use TYPO3\CMS\Core\Utility\PathUtility;
-
-/**
- * Render rich text editor in FormEngine
- */
-class CkEditorRichTextElement extends AbstractFormElement {
-	/**
-	 * Default field wizards enabled for this element.
-	 *
-	 * @var array
-	 */
-	protected $defaultFieldWizard = [
-		'localizationStateSelector' => [
-			'renderType' => 'localizationStateSelector',
-		],
-		'otherLanguageContent' => [
-			'renderType' => 'otherLanguageContent',
-			'after' => [
-				'localizationStateSelector'
-			],
-		],
-		'defaultLanguageDifferences' => [
-			'renderType' => 'defaultLanguageDifferences',
-			'after' => [
-				'otherLanguageContent',
-			],
-		],
-	];
-
-	/**
-	 * This property contains configuration related to the RTE
-	 * But only the .editor configuration part
-	 *
-	 * @var array
-	 */
-	protected $rteConfiguration = [];
-
-	/**
-	 * Renders the ckeditor element
-	 *
-	 * @return array
-	 * @throws \UnexpectedValueException
-	 * @throws \BadFunctionCallException
-	 * @throws \InvalidArgumentException
-	 */
-	public function render(): array {
-		$resultArray = $this->initializeResultArray();
-		$parameterArray = $this->data['parameterArray'];
-		$config = $parameterArray['fieldConf']['config'];
-
-		$fieldId = $this->sanitizeFieldId($parameterArray['itemFormElName']);
-		$itemFormElementName = $this->data['parameterArray']['itemFormElName'];
-
-		$value = $this->data['parameterArray']['itemFormElValue'] ?? '';
-
-		$legacyWizards = $this->renderWizards();
-		$legacyFieldControlHtml = implode(LF, $legacyWizards['fieldControl']);
-		$legacyFieldWizardHtml = implode(LF, $legacyWizards['fieldWizard']);
-
-		$fieldInformationResult = $this->renderFieldInformation();
-		$fieldInformationHtml = $fieldInformationResult['html'];
-		$resultArray = $this->mergeChildReturnIntoExistingResult($resultArray, $fieldInformationResult, FALSE);
-
-		$fieldControlResult = $this->renderFieldControl();
-		$fieldControlHtml = $legacyFieldControlHtml . $fieldControlResult['html'];
-		$resultArray = $this->mergeChildReturnIntoExistingResult($resultArray, $fieldControlResult, FALSE);
-
-		$fieldWizardResult = $this->renderFieldWizard();
-		$fieldWizardHtml = $legacyFieldWizardHtml . $fieldWizardResult['html'];
-		$resultArray = $this->mergeChildReturnIntoExistingResult($resultArray, $fieldWizardResult, FALSE);
-
-//		$attributes = [
-//			'style' => 'display:none',
-//			'data-formengine-validation-rules' => $this->getValidationDataAsJsonString($config),
-//			'id' => $fieldId,
-//			'name' => htmlspecialchars($itemFormElementName),
-//		];
-
-		// Modified by fgalinski - Start
-		$attributes = [
-			'id' => 'RTEarea' . $fieldId,
-			'name' => htmlspecialchars($itemFormElementName),
-			'class' => 'tinymce4_rte',
-		];
-		// Modified by fgalinski - End
-
-		$html = [];
-		$html[] = '<div class="formengine-field-item t3js-formengine-field-item">';
-		$html[] = $fieldInformationHtml;
-		$html[] = '<div class="form-control-wrap">';
-		$html[] = '<div class="form-wizards-wrap">';
-		$html[] = '<div class="form-wizards-element">';
-		$html[] = '<textarea ' . GeneralUtility::implodeAttributes($attributes, TRUE) . '>';
-		$html[] = htmlspecialchars($value);
-		$html[] = '</textarea>';
-		$html[] = '</div>';
-		$html[] = '<div class="form-wizards-items-aside">';
-		$html[] = '<div class="btn-group">';
-		$html[] = $fieldControlHtml;
-		$html[] = '</div>';
-		$html[] = '</div>';
-		$html[] = '<div class="form-wizards-items-bottom">';
-		$html[] = $fieldWizardHtml;
-		$html[] = '</div>';
-		$html[] = '</div>';
-		$html[] = '</div>';
-		$html[] = '</div>';
-
-		$resultArray['html'] = implode(LF, $html);
-
-		$this->rteConfiguration = $config['richtextConfiguration']['editor'];
-
-		// Modified by fgalinski - Start
-		/** @var Loader $tinyMCE */
-		$tinyMCE = GeneralUtility::makeInstance(Loader::class);
-		$tinyMCE->loadConfiguration($config['richtextConfiguration']['tinymceConfiguration']);
-		$contentCssArray = $config['richtextConfiguration']['contentCSS.'];
-		if (is_array($contentCssArray) && count($contentCssArray) > 0) {
-			$contentCssFileArray = [];
-			foreach ($contentCssArray as $contentCssKey => $contentCssFile) {
-				$contentCssFileAbs = GeneralUtility::getFileAbsFileName(trim($contentCssFile));
-				if (is_file($contentCssFileAbs)) {
-					$contentCssFileArray[] = GeneralUtility::getIndpEnv('TYPO3_SITE_URL') . PathUtility::stripPathSitePrefix($contentCssFileAbs) . '?' . filemtime($contentCssFileAbs);
-				}
-			}
-			$tinyMCE->addConfigurationOption('content_css', implode(',', $contentCssFileArray));
-
-		}
-
-		$tinyMCE->addConfigurationOption(
-			'changeMethod', 'function() {
-				var TBE_EDITOR = window.TBE_EDITOR || null;
-				if (TBE_EDITOR && TBE_EDITOR.fieldChanged && typeof TBE_EDITOR.fieldChanged === \'function\') {
-					TBE_EDITOR.fieldChanged();
-				}
-			}'
-		);
-
-		$tinyMCE->addConfigurationOption('editornumber', $fieldId);
-
-		// IRRE
-		$resultArray['requireJsModules'] = $tinyMCE->loadJsViaRequireJS();
-		// Modified by fgalinski - End
-
-		return $resultArray;
-	}
-
-	/**
-	 * @param string $itemFormElementName
-	 *
-	 * @return string
-	 */
-	protected function sanitizeFieldId(string $itemFormElementName): string {
-		$fieldId = preg_replace('/[^a-zA-Z0-9_:.-]/', '_', $itemFormElementName);
-		return htmlspecialchars(preg_replace('/^[^a-zA-Z]/', 'x', $fieldId));
-	}
-
-	/**
-	 * Gets the JavaScript code for CKEditor module
-	 * Compiles the configuration, and then adds plugins
-	 *
-	 * @param string $fieldId
-	 *
-	 * @return string
-	 */
-	protected function getCkEditorRequireJsModuleCode(string $fieldId): string {
-		$configuration = $this->prepareConfigurationForEditor();
-
-		$externalPlugins = '';
-		foreach ($this->getExtraPlugins() as $pluginName => $config) {
-			$configuration[$pluginName] = $config['config'];
-			$configuration['extraPlugins'] .= ',' . $pluginName;
-
-			$externalPlugins .= 'CKEDITOR.plugins.addExternal(';
-			$externalPlugins .= GeneralUtility::quoteJSvalue($pluginName) . ',';
-			$externalPlugins .= GeneralUtility::quoteJSvalue($config['resource']) . ',';
-			$externalPlugins .= '\'\');';
-		}
-
-		return 'function(CKEDITOR) {
-                ' . $externalPlugins . '
-                $(function(){
-                    CKEDITOR.replace("' . $fieldId . '", ' . json_encode($configuration) . ');
-                    require([\'jquery\', \'TYPO3/CMS/Backend/FormEngine\'], function($, FormEngine) {
-                        CKEDITOR.instances["' . $fieldId . '"].on(\'change\', function() {
-                            CKEDITOR.instances["' . $fieldId . '"].updateElement();
-                            FormEngine.Validation.validate();
-                            FormEngine.Validation.markFieldAsChanged($(\'#' . $fieldId . '\'));
-                        });
-                    });
-                });
-        }';
-	}
-
-	/**
-	 * Compiles the configuration set from the outside
-	 * to have it easily injected into the CKEditor.
-	 *
-	 * @return array the configuration
-	 */
-	protected function prepareConfigurationForEditor(): array {
-		// Ensure custom config is empty so nothing additional is loaded
-		// Of course this can be overridden by the editor configuration below
-		$configuration = [
-			'customConfig' => '',
-		];
-
-		if (is_array($this->rteConfiguration['config'])) {
-			$configuration = array_replace_recursive($configuration, $this->rteConfiguration['config']);
-		}
-		// Set the UI language of the editor if not hard-coded by the existing configuration
-		if (empty($configuration['language'])) {
-			$configuration['language'] = $this->getBackendUser()->uc['lang'] ?: ($this->getBackendUser()->user['lang'] ?: 'en');
-		}
-		$configuration['contentsLanguage'] = $this->getLanguageIsoCodeOfContent();
-
-		// Replace all label references
-		$configuration = $this->replaceLanguageFileReferences($configuration);
-		// Replace all paths
-		$configuration = $this->replaceAbsolutePathsToRelativeResourcesPath($configuration);
-
-		// there are some places where we define an array, but it needs to be a list in order to work
-		if (is_array($configuration['extraPlugins'])) {
-			$configuration['extraPlugins'] = implode(',', $configuration['extraPlugins']);
-		}
-		if (is_array($configuration['removePlugins'])) {
-			$configuration['removePlugins'] = implode(',', $configuration['removePlugins']);
-		}
-		if (is_array($configuration['removeButtons'])) {
-			$configuration['removeButtons'] = implode(',', $configuration['removeButtons']);
-		}
-
-		return $configuration;
-	}
-
-	/**
-	 * @return BackendUserAuthentication
-	 */
-	protected function getBackendUser() {
-		return $GLOBALS['BE_USER'];
-	}
-
-	/**
-	 * Determine the contents language iso code
-	 *
-	 * @return string
-	 */
-	protected function getLanguageIsoCodeOfContent(): string {
-		$currentLanguageUid = $this->data['databaseRow']['sys_language_uid'];
-		if (is_array($currentLanguageUid)) {
-			$currentLanguageUid = $currentLanguageUid[0];
-		}
-		$contentLanguageUid = (int) max($currentLanguageUid, 0);
-		if ($contentLanguageUid) {
-			$contentLanguage = $this->data['systemLanguageRows'][$currentLanguageUid]['iso'];
-		} else {
-			$contentLanguage = $this->rteConfiguration['config']['defaultContentLanguage'] ?? 'en_US';
-			$languageCodeParts = explode('_', $contentLanguage);
-			$contentLanguage = strtolower($languageCodeParts[0]) . ($languageCodeParts[1] ? '_' . strtoupper($languageCodeParts[1]) : '');
-			// Find the configured language in the list of localization locales
-			$locales = GeneralUtility::makeInstance(Locales::class);
-			// If not found, default to 'en'
-			if (!in_array($contentLanguage, $locales->getLocales(), TRUE)) {
-				$contentLanguage = 'en';
-			}
-		}
-		return $contentLanguage;
-	}
-
-	/**
-	 * Add configuration to replace LLL: references with the translated value
-	 *
-	 * @param array $configuration
-	 *
-	 * @return array
-	 */
-	protected function replaceLanguageFileReferences(array $configuration): array {
-		foreach ($configuration as $key => $value) {
-			if (is_array($value)) {
-				$configuration[$key] = $this->replaceLanguageFileReferences($value);
-			} elseif (is_string($value) && stripos($value, 'LLL:') === 0) {
-				$configuration[$key] = $this->getLanguageService()->sL($value);
-			}
-		}
-		return $configuration;
-	}
-
-	/**
-	 * Add configuration to replace absolute EXT: paths with relative ones
-	 *
-	 * @param array $configuration
-	 *
-	 * @return array
-	 */
-	protected function replaceAbsolutePathsToRelativeResourcesPath(array $configuration): array {
-		foreach ($configuration as $key => $value) {
-			if (is_array($value)) {
-				$configuration[$key] = $this->replaceAbsolutePathsToRelativeResourcesPath($value);
-			} elseif (is_string($value) && stripos($value, 'EXT:') === 0) {
-				$configuration[$key] = $this->resolveUrlPath($value);
-			}
-		}
-		return $configuration;
-	}
-
-	/**
-	 * Resolves an EXT: syntax file to an absolute web URL
-	 *
-	 * @param string $value
-	 *
-	 * @return string
-	 */
-	protected function resolveUrlPath(string $value): string {
-		$value = GeneralUtility::getFileAbsFileName($value);
-		return PathUtility::getAbsoluteWebPath($value);
-	}
-
-	/**
-	 * Get configuration of external/additional plugins
-	 *
-	 * @return array
-	 */
-	protected function getExtraPlugins(): array {
-		$urlParameters = [
-			'P' => [
-				'table' => $this->data['tableName'],
-				'uid' => $this->data['databaseRow']['uid'],
-				'fieldName' => $this->data['fieldName'],
-				'recordType' => $this->data['recordTypeValue'],
-				'pid' => $this->data['effectivePid'],
-			]
-		];
-
-		$pluginConfiguration = [];
-		if (isset($this->rteConfiguration['externalPlugins']) && is_array($this->rteConfiguration['externalPlugins'])) {
-			$uriBuilder = GeneralUtility::makeInstance(UriBuilder::class);
-			foreach ($this->rteConfiguration['externalPlugins'] as $pluginName => $configuration) {
-				$pluginConfiguration[$pluginName] = [
-					'resource' => $this->resolveUrlPath($configuration['resource'])
-				];
-				unset($configuration['resource']);
-
-				if ($configuration['route']) {
-					$configuration['routeUrl'] = (string) $uriBuilder->buildUriFromRoute($configuration['route'], $urlParameters);
-				}
-
-				$pluginConfiguration[$pluginName]['config'] = $configuration;
-			}
-		}
-		return $pluginConfiguration;
-	}
-}
diff --git a/Classes/Form/Element/RtehtmlareaRichTextElement.php b/Classes/Form/Element/RichTextElement.php
similarity index 95%
rename from Classes/Form/Element/RtehtmlareaRichTextElement.php
rename to Classes/Form/Element/RichTextElement.php
index 3ceee7f..74c807d 100644
--- a/Classes/Form/Element/RtehtmlareaRichTextElement.php
+++ b/Classes/Form/Element/RichTextElement.php
@@ -39,7 +39,7 @@ use TYPO3\CMS\Lang\LanguageService;
 /**
  * Render rich text editor in FormEngine
  */
-class RtehtmlareaRichTextElement extends AbstractFormElement {
+class RichTextElement extends AbstractFormElement {
 	/**
 	 * Main result array as defined in initializeResultArray() of AbstractNode
 	 *
@@ -359,55 +359,55 @@ class RtehtmlareaRichTextElement extends AbstractFormElement {
 	 * @return void
 	 */
 	protected function enableRegisteredPlugins() {
-		$plugins = [
-			'TYPO3Image' => [
-				'objectReference' => Typo3Image::class,
-			],
-			'TYPO3Link' => [
-				'objectReference' => TYPO3Link::class,
-			],
-		];
-		foreach ($plugins as $pluginId => $pluginObjectConfiguration) {
-			if (is_array($pluginObjectConfiguration) && isset($pluginObjectConfiguration['objectReference'])) {
-				/** @var RteHtmlAreaApi $plugin */
-				$plugin = GeneralUtility::makeInstance($pluginObjectConfiguration['objectReference']);
-				$configuration = array(
-					'language' => $this->language,
-					'contentTypo3Language' => $this->contentTypo3Language,
-					'contentISOLanguage' => $this->contentISOLanguage,
-					'contentLanguageUid' => $this->contentLanguageUid,
-					'RTEsetup' => $this->vanillaRteTsConfig,
-					'client' => $this->client,
-					'thisConfig' => $this->processedRteConfiguration,
-					'specConf' => $this->defaultExtras,
-				);
-				if ($plugin->main($configuration)) {
-					$this->registeredPlugins[$pluginId] = $plugin;
-					// Override buttons from previously registered plugins
-					$pluginButtons = GeneralUtility::trimExplode(',', $plugin->getPluginButtons(), TRUE);
-					foreach ($this->pluginButton as $previousPluginId => $buttonList) {
-						$this->pluginButton[$previousPluginId] = implode(',', array_diff(GeneralUtility::trimExplode(',', $this->pluginButton[$previousPluginId], TRUE), $pluginButtons));
-					}
-					$this->pluginButton[$pluginId] = $plugin->getPluginButtons();
-					$pluginLabels = GeneralUtility::trimExplode(',', $plugin->getPluginLabels(), TRUE);
-					foreach ($this->pluginLabel as $previousPluginId => $labelList) {
-						$this->pluginLabel[$previousPluginId] = implode(',', array_diff(GeneralUtility::trimExplode(',', $this->pluginLabel[$previousPluginId], TRUE), $pluginLabels));
-					}
-					$this->pluginLabel[$pluginId] = $plugin->getPluginLabels();
-					$this->pluginEnabledArray[] = $pluginId;
-				}
-			}
-		}
-
-		// Process overrides
-		$hidePlugins = array();
-		foreach ($this->registeredPlugins as $pluginId => $plugin) {
-			/** @var RteHtmlAreaApi $plugin */
-			if ($plugin->addsButtons() && !$this->pluginButton[$pluginId]) {
-				$hidePlugins[] = $pluginId;
-			}
-		}
-		$this->pluginEnabledArray = array_unique(array_diff($this->pluginEnabledArray, $hidePlugins));
+//		$plugins = [
+//			'TYPO3Image' => [
+//				'objectReference' => Typo3Image::class,
+//			],
+//			'TYPO3Link' => [
+//				'objectReference' => TYPO3Link::class,
+//			],
+//		];
+//		foreach ($plugins as $pluginId => $pluginObjectConfiguration) {
+//			if (is_array($pluginObjectConfiguration) && isset($pluginObjectConfiguration['objectReference'])) {
+//				/** @var RteHtmlAreaApi $plugin */
+//				$plugin = GeneralUtility::makeInstance($pluginObjectConfiguration['objectReference']);
+//				$configuration = array(
+//					'language' => $this->language,
+//					'contentTypo3Language' => $this->contentTypo3Language,
+//					'contentISOLanguage' => $this->contentISOLanguage,
+//					'contentLanguageUid' => $this->contentLanguageUid,
+//					'RTEsetup' => $this->vanillaRteTsConfig,
+//					'client' => $this->client,
+//					'thisConfig' => $this->processedRteConfiguration,
+//					'specConf' => $this->defaultExtras,
+//				);
+//				if ($plugin->main($configuration)) {
+//					$this->registeredPlugins[$pluginId] = $plugin;
+//					// Override buttons from previously registered plugins
+//					$pluginButtons = GeneralUtility::trimExplode(',', $plugin->getPluginButtons(), TRUE);
+//					foreach ($this->pluginButton as $previousPluginId => $buttonList) {
+//						$this->pluginButton[$previousPluginId] = implode(',', array_diff(GeneralUtility::trimExplode(',', $this->pluginButton[$previousPluginId], TRUE), $pluginButtons));
+//					}
+//					$this->pluginButton[$pluginId] = $plugin->getPluginButtons();
+//					$pluginLabels = GeneralUtility::trimExplode(',', $plugin->getPluginLabels(), TRUE);
+//					foreach ($this->pluginLabel as $previousPluginId => $labelList) {
+//						$this->pluginLabel[$previousPluginId] = implode(',', array_diff(GeneralUtility::trimExplode(',', $this->pluginLabel[$previousPluginId], TRUE), $pluginLabels));
+//					}
+//					$this->pluginLabel[$pluginId] = $plugin->getPluginLabels();
+//					$this->pluginEnabledArray[] = $pluginId;
+//				}
+//			}
+//		}
+//
+//		// Process overrides
+//		$hidePlugins = array();
+//		foreach ($this->registeredPlugins as $pluginId => $plugin) {
+//			/** @var RteHtmlAreaApi $plugin */
+//			if ($plugin->addsButtons() && !$this->pluginButton[$pluginId]) {
+//				$hidePlugins[] = $pluginId;
+//			}
+//		}
+//		$this->pluginEnabledArray = array_unique(array_diff($this->pluginEnabledArray, $hidePlugins));
 	}
 
 	/**
diff --git a/Classes/Form/Resolver/RichTextNodeResolver.php b/Classes/Form/Resolver/RichTextNodeResolver.php
index 2501ab1..c86c393 100644
--- a/Classes/Form/Resolver/RichTextNodeResolver.php
+++ b/Classes/Form/Resolver/RichTextNodeResolver.php
@@ -15,8 +15,7 @@ namespace SGalinski\Tinymce4Rte\Form\Resolver;
  * The TYPO3 project - inspiring people to share!
  */
 
-use SGalinski\Tinymce4Rte\Form\Element\CkEditorRichTextElement;
-use SGalinski\Tinymce4Rte\Form\Element\RtehtmlareaRichTextElement;
+use SGalinski\Tinymce4Rte\Form\Element\RichTextElement;
 use SGalinski\Tinymce4Rte\Utility\VersionUtility;
 use TYPO3\CMS\Backend\Form\NodeFactory;
 use TYPO3\CMS\Backend\Form\NodeResolverInterface;
@@ -71,7 +70,7 @@ class RichTextNodeResolver implements NodeResolverInterface {
 				// If RTE is not disabled on configuration level
 				&& !$parameterArray['fieldConf']['config']['richtextConfiguration']['disabled']
 			) {
-				return CkEditorRichTextElement::class;
+				return RichTextElement::class;
 			}
 		} else {
 			if (// This field is not read only
@@ -96,7 +95,7 @@ class RichTextNodeResolver implements NodeResolverInterface {
 						);
 						if (!$rteSetupConfiguration['disabled']) {
 							// Finally, we're sure the editor should really be rendered ...
-							return RtehtmlareaRichTextElement::class;
+							return RichTextElement::class;
 						}
 					}
 				}
-- 
GitLab