vendor/shopware/storefront/Theme/ThemeService.php line 186

  1. <?php declare(strict_types=1);
  2. namespace Shopware\Storefront\Theme;
  3. use Doctrine\DBAL\Connection;
  4. use Shopware\Core\Framework\Context;
  5. use Shopware\Core\Framework\DataAbstractionLayer\EntityRepository;
  6. use Shopware\Core\Framework\DataAbstractionLayer\Exception\InconsistentCriteriaIdsException;
  7. use Shopware\Core\Framework\DataAbstractionLayer\Search\Criteria;
  8. use Shopware\Core\Framework\DataAbstractionLayer\Search\EntitySearchResult;
  9. use Shopware\Core\Framework\Log\Package;
  10. use Shopware\Core\Framework\Uuid\Uuid;
  11. use Shopware\Storefront\Theme\ConfigLoader\AbstractConfigLoader;
  12. use Shopware\Storefront\Theme\Event\ThemeAssignedEvent;
  13. use Shopware\Storefront\Theme\Event\ThemeConfigChangedEvent;
  14. use Shopware\Storefront\Theme\Event\ThemeConfigResetEvent;
  15. use Shopware\Storefront\Theme\Exception\InvalidThemeConfigException;
  16. use Shopware\Storefront\Theme\Exception\InvalidThemeException;
  17. use Shopware\Storefront\Theme\StorefrontPluginConfiguration\StorefrontPluginConfigurationCollection;
  18. use Symfony\Contracts\EventDispatcher\EventDispatcherInterface;
  19. #[Package('storefront')]
  20. class ThemeService
  21. {
  22.     /**
  23.      * @internal
  24.      */
  25.     public function __construct(
  26.         private readonly StorefrontPluginRegistryInterface $extensionRegistry,
  27.         private readonly EntityRepository $themeRepository,
  28.         private readonly EntityRepository $themeSalesChannelRepository,
  29.         private readonly ThemeCompilerInterface $themeCompiler,
  30.         private readonly EventDispatcherInterface $dispatcher,
  31.         private readonly AbstractConfigLoader $configLoader,
  32.         private readonly Connection $connection
  33.     ) {
  34.     }
  35.     /**
  36.      * Only compiles a single theme/saleschannel combination.
  37.      * Use `compileThemeById` to compile all dependend saleschannels
  38.      */
  39.     public function compileTheme(
  40.         string $salesChannelId,
  41.         string $themeId,
  42.         Context $context,
  43.         ?StorefrontPluginConfigurationCollection $configurationCollection null,
  44.         bool $withAssets true
  45.     ): void {
  46.         $this->themeCompiler->compileTheme(
  47.             $salesChannelId,
  48.             $themeId,
  49.             $this->configLoader->load($themeId$context),
  50.             $configurationCollection ?? $this->extensionRegistry->getConfigurations(),
  51.             $withAssets,
  52.             $context
  53.         );
  54.     }
  55.     /**
  56.      * Compiles all dependend saleschannel/Theme combinations
  57.      *
  58.      * @return array<int, string>
  59.      */
  60.     public function compileThemeById(
  61.         string $themeId,
  62.         Context $context,
  63.         ?StorefrontPluginConfigurationCollection $configurationCollection null,
  64.         bool $withAssets true
  65.     ): array {
  66.         $mappings $this->getThemeDependencyMapping($themeId);
  67.         $compiledThemeIds = [];
  68.         /** @var ThemeSalesChannel $mapping */
  69.         foreach ($mappings as $mapping) {
  70.             $this->themeCompiler->compileTheme(
  71.                 $mapping->getSalesChannelId(),
  72.                 $mapping->getThemeId(),
  73.                 $this->configLoader->load($mapping->getThemeId(), $context),
  74.                 $configurationCollection ?? $this->extensionRegistry->getConfigurations(),
  75.                 $withAssets,
  76.                 $context
  77.             );
  78.             $compiledThemeIds[] = $mapping->getThemeId();
  79.         }
  80.         return $compiledThemeIds;
  81.     }
  82.     /**
  83.      * @param array<string, mixed>|null $config
  84.      */
  85.     public function updateTheme(string $themeId, ?array $config, ?string $parentThemeIdContext $context): void
  86.     {
  87.         $criteria = new Criteria([$themeId]);
  88.         $criteria->addAssociation('salesChannels');
  89.         /** @var ThemeEntity|null $theme */
  90.         $theme $this->themeRepository->search($criteria$context)->get($themeId);
  91.         if (!$theme) {
  92.             throw new InvalidThemeException($themeId);
  93.         }
  94.         $data = ['id' => $themeId];
  95.         if ($config) {
  96.             foreach ($config as $key => $value) {
  97.                 $data['configValues'][$key] = $value;
  98.             }
  99.         }
  100.         if ($parentThemeId) {
  101.             $data['parentThemeId'] = $parentThemeId;
  102.         }
  103.         if (\array_key_exists('configValues'$data)) {
  104.             $this->dispatcher->dispatch(new ThemeConfigChangedEvent($themeId$data['configValues']));
  105.         }
  106.         if (\array_key_exists('configValues'$data) && $theme->getConfigValues()) {
  107.             $submittedChanges $data['configValues'];
  108.             $currentConfig $theme->getConfigValues();
  109.             $data['configValues'] = array_replace_recursive($currentConfig$data['configValues']);
  110.             foreach ($submittedChanges as $key => $changes) {
  111.                 if (isset($changes['value']) && \is_array($changes['value']) && isset($currentConfig[(string) $key]) && \is_array($currentConfig[(string) $key])) {
  112.                     $data['configValues'][$key]['value'] = array_unique($changes['value']);
  113.                 }
  114.             }
  115.         }
  116.         $this->themeRepository->update([$data], $context);
  117.         if ($theme->getSalesChannels() === null) {
  118.             return;
  119.         }
  120.         $this->compileThemeById($themeId$contextnullfalse);
  121.     }
  122.     public function assignTheme(string $themeIdstring $salesChannelIdContext $contextbool $skipCompile false): bool
  123.     {
  124.         if (!$skipCompile) {
  125.             $this->compileTheme($salesChannelId$themeId$context);
  126.         }
  127.         $this->themeSalesChannelRepository->upsert([[
  128.             'themeId' => $themeId,
  129.             'salesChannelId' => $salesChannelId,
  130.         ]], $context);
  131.         $this->dispatcher->dispatch(new ThemeAssignedEvent($themeId$salesChannelId));
  132.         return true;
  133.     }
  134.     public function resetTheme(string $themeIdContext $context): void
  135.     {
  136.         $criteria = new Criteria([$themeId]);
  137.         $theme $this->themeRepository->search($criteria$context)->get($themeId);
  138.         if (!$theme) {
  139.             throw new InvalidThemeException($themeId);
  140.         }
  141.         $data = ['id' => $themeId];
  142.         $data['configValues'] = null;
  143.         $this->dispatcher->dispatch(new ThemeConfigResetEvent($themeId));
  144.         $this->themeRepository->update([$data], $context);
  145.     }
  146.     /**
  147.      * @throws InvalidThemeConfigException
  148.      * @throws InvalidThemeException
  149.      * @throws InconsistentCriteriaIdsException
  150.      *
  151.      * @return array<string, mixed>
  152.      */
  153.     public function getThemeConfiguration(string $themeIdbool $translateContext $context): array
  154.     {
  155.         $criteria = new Criteria();
  156.         $criteria->setTitle('theme-service::load-config');
  157.         $themes $this->themeRepository->search($criteria$context);
  158.         $theme $themes->get($themeId);
  159.         /** @var ThemeEntity|null $theme */
  160.         if (!$theme) {
  161.             throw new InvalidThemeException($themeId);
  162.         }
  163.         /** @var ThemeEntity $baseTheme */
  164.         $baseTheme $themes->filter(fn (ThemeEntity $themeEntry) => $themeEntry->getTechnicalName() === StorefrontPluginRegistry::BASE_THEME_NAME)->first();
  165.         $baseThemeConfig $this->mergeStaticConfig($baseTheme);
  166.         $themeConfigFieldFactory = new ThemeConfigFieldFactory();
  167.         $configFields = [];
  168.         $labels array_replace_recursive($baseTheme->getLabels() ?? [], $theme->getLabels() ?? []);
  169.         $helpTexts array_replace_recursive($baseTheme->getHelpTexts() ?? [], $theme->getHelpTexts() ?? []);
  170.         if ($theme->getParentThemeId()) {
  171.             $parentThemes $this->getParentThemeIds($themes$theme);
  172.             foreach ($parentThemes as $parentTheme) {
  173.                 $configuredParentTheme $this->mergeStaticConfig($parentTheme);
  174.                 $baseThemeConfig array_replace_recursive($baseThemeConfig$configuredParentTheme);
  175.                 $labels array_replace_recursive($labels$parentTheme->getLabels() ?? []);
  176.                 $helpTexts array_replace_recursive($helpTexts$parentTheme->getHelpTexts() ?? []);
  177.             }
  178.         }
  179.         $configuredTheme $this->mergeStaticConfig($theme);
  180.         $themeConfig array_replace_recursive($baseThemeConfig$configuredTheme);
  181.         foreach ($themeConfig['fields'] ?? [] as $name => &$item) {
  182.             $configFields[$name] = $themeConfigFieldFactory->create($name$item);
  183.             if (
  184.                 isset($item['value'])
  185.                 && isset($configuredTheme['fields'])
  186.                 && \is_array($item['value'])
  187.                 && \array_key_exists($name$configuredTheme['fields'])
  188.             ) {
  189.                 $configFields[$name]->setValue($configuredTheme['fields'][$name]['value']);
  190.             }
  191.         }
  192.         $configFields json_decode((string) json_encode($configFields\JSON_THROW_ON_ERROR), true512\JSON_THROW_ON_ERROR);
  193.         if ($translate && !empty($labels)) {
  194.             $configFields $this->translateLabels($configFields$labels);
  195.         }
  196.         if ($translate && !empty($helpTexts)) {
  197.             $configFields $this->translateHelpTexts($configFields$helpTexts);
  198.         }
  199.         $themeConfig['fields'] = $configFields;
  200.         $themeConfig['currentFields'] = [];
  201.         $themeConfig['baseThemeFields'] = [];
  202.         foreach ($themeConfig['fields'] as $field => $fieldItem) {
  203.             $isInherited $this->fieldIsInherited($field$configuredTheme);
  204.             $themeConfig['currentFields'][$field]['isInherited'] = $isInherited;
  205.             if ($isInherited) {
  206.                 $themeConfig['currentFields'][$field]['value'] = null;
  207.             } elseif (\array_key_exists('value'$fieldItem)) {
  208.                 $themeConfig['currentFields'][$field]['value'] = $fieldItem['value'];
  209.             }
  210.             $isInherited $this->fieldIsInherited($field$baseThemeConfig);
  211.             $themeConfig['baseThemeFields'][$field]['isInherited'] = $isInherited;
  212.             if ($isInherited) {
  213.                 $themeConfig['baseThemeFields'][$field]['value'] = null;
  214.             } elseif (\array_key_exists('value'$fieldItem) && isset($baseThemeConfig['fields'][$field]['value'])) {
  215.                 $themeConfig['baseThemeFields'][$field]['value'] = $baseThemeConfig['fields'][$field]['value'];
  216.             }
  217.         }
  218.         return $themeConfig;
  219.     }
  220.     /**
  221.      * @return array<string, mixed>
  222.      */
  223.     public function getThemeConfigurationStructuredFields(string $themeIdbool $translateContext $context): array
  224.     {
  225.         $mergedConfig $this->getThemeConfiguration($themeId$translate$context)['fields'];
  226.         $translations = [];
  227.         if ($translate) {
  228.             $translations $this->getTranslations($themeId$context);
  229.             $mergedConfig $this->translateLabels($mergedConfig$translations);
  230.         }
  231.         $outputStructure = [];
  232.         foreach ($mergedConfig as $fieldName => $fieldConfig) {
  233.             $tab $this->getTab($fieldConfig);
  234.             $tabLabel $this->getTabLabel($tab$translations);
  235.             $block $this->getBlock($fieldConfig);
  236.             $blockLabel $this->getBlockLabel($block$translations);
  237.             $section $this->getSection($fieldConfig);
  238.             $sectionLabel $this->getSectionLabel($section$translations);
  239.             // set default tab
  240.             $outputStructure['tabs']['default']['label'] = '';
  241.             // set labels
  242.             $outputStructure['tabs'][$tab]['label'] = $tabLabel;
  243.             $outputStructure['tabs'][$tab]['blocks'][$block]['label'] = $blockLabel;
  244.             $outputStructure['tabs'][$tab]['blocks'][$block]['sections'][$section]['label'] = $sectionLabel;
  245.             // add fields to sections
  246.             $outputStructure['tabs'][$tab]['blocks'][$block]['sections'][$section]['fields'][$fieldName] = [
  247.                 'label' => $fieldConfig['label'],
  248.                 'helpText' => $fieldConfig['helpText'] ?? null,
  249.                 'type' => $fieldConfig['type'],
  250.                 'custom' => $fieldConfig['custom'],
  251.                 'fullWidth' => $fieldConfig['fullWidth'],
  252.             ];
  253.         }
  254.         return $outputStructure;
  255.     }
  256.     public function getThemeDependencyMapping(string $themeId): ThemeSalesChannelCollection
  257.     {
  258.         $mappings = new ThemeSalesChannelCollection();
  259.         $themeData $this->connection->fetchAllAssociative(
  260.             'SELECT LOWER(HEX(theme.id)) as id, LOWER(HEX(childTheme.id)) as dependentId,
  261.             LOWER(HEX(tsc.sales_channel_id)) as saleschannelId,
  262.             LOWER(HEX(dtsc.sales_channel_id)) as dsaleschannelId
  263.             FROM theme
  264.             LEFT JOIN theme as childTheme ON childTheme.parent_theme_id = theme.id
  265.             LEFT JOIN theme_sales_channel as tsc ON theme.id = tsc.theme_id
  266.             LEFT JOIN theme_sales_channel as dtsc ON childTheme.id = dtsc.theme_id
  267.             WHERE theme.id = :id',
  268.             ['id' => Uuid::fromHexToBytes($themeId)]
  269.         );
  270.         foreach ($themeData as $data) {
  271.             if (isset($data['id']) && isset($data['saleschannelId']) && $data['id'] === $themeId) {
  272.                 $mappings->add(new ThemeSalesChannel($data['id'], $data['saleschannelId']));
  273.             }
  274.             if (isset($data['dependentId']) && isset($data['dsaleschannelId'])) {
  275.                 $mappings->add(new ThemeSalesChannel($data['dependentId'], $data['dsaleschannelId']));
  276.             }
  277.         }
  278.         return $mappings;
  279.     }
  280.     /**
  281.      * @param array<string, mixed> $parentThemes
  282.      *
  283.      * @return array<string, mixed>
  284.      */
  285.     private function getParentThemeIds(EntitySearchResult $themesThemeEntity $mainTheme, array $parentThemes = []): array
  286.     {
  287.         foreach ($this->getConfigInheritance($mainTheme) as $parentThemeName) {
  288.             $parentTheme $themes->filter(fn (ThemeEntity $themeEntry) => $themeEntry->getTechnicalName() === str_replace('@''', (string) $parentThemeName))->first();
  289.             if ($parentTheme instanceof ThemeEntity && !\array_key_exists($parentTheme->getId(), $parentThemes)) {
  290.                 $parentThemes[$parentTheme->getId()] = $parentTheme;
  291.                 if ($parentTheme->getParentThemeId()) {
  292.                     $parentThemes $this->getParentThemeIds($themes$mainTheme$parentThemes);
  293.                 }
  294.             }
  295.         }
  296.         if ($mainTheme->getParentThemeId()) {
  297.             $parentTheme $themes->filter(fn (ThemeEntity $themeEntry) => $themeEntry->getId() === $mainTheme->getParentThemeId())->first();
  298.             if ($parentTheme instanceof ThemeEntity && !\array_key_exists($parentTheme->getId(), $parentThemes)) {
  299.                 $parentThemes[$parentTheme->getId()] = $parentTheme;
  300.                 if ($parentTheme->getParentThemeId()) {
  301.                     $parentThemes $this->getParentThemeIds($themes$mainTheme$parentThemes);
  302.                 }
  303.             }
  304.         }
  305.         return $parentThemes;
  306.     }
  307.     /**
  308.      * @return array<string, mixed>
  309.      */
  310.     private function getConfigInheritance(ThemeEntity $mainTheme): array
  311.     {
  312.         if (\is_array($mainTheme->getBaseConfig())
  313.             && \array_key_exists('configInheritance'$mainTheme->getBaseConfig())
  314.             && \is_array($mainTheme->getBaseConfig()['configInheritance'])
  315.             && !empty($mainTheme->getBaseConfig()['configInheritance'])
  316.         ) {
  317.             return $mainTheme->getBaseConfig()['configInheritance'];
  318.         }
  319.         return [];
  320.     }
  321.     /**
  322.      * @return array<string, mixed>
  323.      */
  324.     private function mergeStaticConfig(ThemeEntity $theme): array
  325.     {
  326.         $configuredTheme = [];
  327.         $pluginConfig null;
  328.         if ($theme->getTechnicalName()) {
  329.             $pluginConfig $this->extensionRegistry->getConfigurations()->getByTechnicalName($theme->getTechnicalName());
  330.         }
  331.         if ($pluginConfig !== null) {
  332.             $configuredTheme $pluginConfig->getThemeConfig();
  333.         }
  334.         if ($theme->getBaseConfig() !== null) {
  335.             $configuredTheme array_replace_recursive($configuredTheme ?? [], $theme->getBaseConfig());
  336.         }
  337.         if ($theme->getConfigValues() !== null) {
  338.             foreach ($theme->getConfigValues() as $fieldName => $configValue) {
  339.                 if (\array_key_exists('value'$configValue)) {
  340.                     $configuredTheme['fields'][$fieldName]['value'] = $configValue['value'];
  341.                 }
  342.             }
  343.         }
  344.         return $configuredTheme ?: [];
  345.     }
  346.     /**
  347.      * @param array<string, mixed> $fieldConfig
  348.      */
  349.     private function getTab(array $fieldConfig): string
  350.     {
  351.         $tab 'default';
  352.         if (isset($fieldConfig['tab'])) {
  353.             $tab $fieldConfig['tab'];
  354.         }
  355.         return $tab;
  356.     }
  357.     /**
  358.      * @param array<string, mixed> $fieldConfig
  359.      */
  360.     private function getBlock(array $fieldConfig): string
  361.     {
  362.         $block 'default';
  363.         if (isset($fieldConfig['block'])) {
  364.             $block $fieldConfig['block'];
  365.         }
  366.         return $block;
  367.     }
  368.     /**
  369.      * @param array<string, mixed> $fieldConfig
  370.      */
  371.     private function getSection(array $fieldConfig): string
  372.     {
  373.         $section 'default';
  374.         if (isset($fieldConfig['section'])) {
  375.             $section $fieldConfig['section'];
  376.         }
  377.         return $section;
  378.     }
  379.     /**
  380.      * @param array<string, mixed> $translations
  381.      */
  382.     private function getTabLabel(string $tabName, array $translations): string
  383.     {
  384.         if ($tabName === 'default') {
  385.             return '';
  386.         }
  387.         return $translations['tabs.' $tabName] ?? $tabName;
  388.     }
  389.     /**
  390.      * @param array<string, mixed> $translations
  391.      */
  392.     private function getBlockLabel(string $blockName, array $translations): string
  393.     {
  394.         if ($blockName === 'default') {
  395.             return '';
  396.         }
  397.         return $translations['blocks.' $blockName] ?? $blockName;
  398.     }
  399.     /**
  400.      * @param array<string, mixed> $translations
  401.      */
  402.     private function getSectionLabel(string $sectionName, array $translations): string
  403.     {
  404.         if ($sectionName === 'default') {
  405.             return '';
  406.         }
  407.         return $translations['sections.' $sectionName] ?? $sectionName;
  408.     }
  409.     /**
  410.      * @param array<string, mixed> $themeConfiguration
  411.      * @param array<string, mixed> $translations
  412.      *
  413.      * @return array<string, mixed>
  414.      */
  415.     private function translateLabels(array $themeConfiguration, array $translations): array
  416.     {
  417.         foreach ($themeConfiguration as $key => &$value) {
  418.             $value['label'] = $translations['fields.' $key] ?? $key;
  419.         }
  420.         return $themeConfiguration;
  421.     }
  422.     /**
  423.      * @param array<string, mixed> $themeConfiguration
  424.      * @param array<string, mixed> $translations
  425.      *
  426.      * @return array<string, mixed>
  427.      */
  428.     private function translateHelpTexts(array $themeConfiguration, array $translations): array
  429.     {
  430.         foreach ($themeConfiguration as $key => &$value) {
  431.             $value['helpText'] = $translations['fields.' $key] ?? null;
  432.         }
  433.         return $themeConfiguration;
  434.     }
  435.     /**
  436.      * @return array<string, mixed>
  437.      */
  438.     private function getTranslations(string $themeIdContext $context): array
  439.     {
  440.         /** @var ThemeEntity $theme */
  441.         $theme $this->themeRepository->search(new Criteria([$themeId]), $context)->get($themeId);
  442.         $translations $theme->getLabels() ?: [];
  443.         if ($theme->getParentThemeId()) {
  444.             $criteria = new Criteria();
  445.             $criteria->setTitle('theme-service::load-translations');
  446.             $themes $this->themeRepository->search($criteria$context);
  447.             $parentThemes $this->getParentThemeIds($themes$theme);
  448.             foreach ($parentThemes as $parentTheme) {
  449.                 $parentTranslations $parentTheme->getLabels() ?: [];
  450.                 $translations array_replace_recursive($parentTranslations$translations);
  451.             }
  452.         }
  453.         return $translations;
  454.     }
  455.     /**
  456.      * @param array<string, mixed> $configuration
  457.      */
  458.     private function fieldIsInherited(string $fieldName, array $configuration): bool
  459.     {
  460.         if (!isset($configuration['fields'])) {
  461.             return true;
  462.         }
  463.         if (!\is_array($configuration['fields'])) {
  464.             return true;
  465.         }
  466.         if (!\array_key_exists($fieldName$configuration['fields'])) {
  467.             return true;
  468.         }
  469.         return false;
  470.     }
  471. }