vendor/terminal42/dc_multilingual/src/Model/MultilingualTrait.php line 159

Open in your IDE?
  1. <?php
  2. declare(strict_types=1);
  3. namespace Terminal42\DcMultilingualBundle\Model;
  4. use Contao\Database;
  5. use Contao\Database\Result;
  6. use Contao\DcaExtractor;
  7. use Contao\DcaLoader;
  8. use Contao\Model\Collection;
  9. use Contao\System;
  10. use Doctrine\DBAL\Query\QueryBuilder;
  11. use Terminal42\DcMultilingualBundle\QueryBuilder\MultilingualQueryBuilderFactoryInterface;
  12. use Terminal42\DcMultilingualBundle\QueryBuilder\MultilingualQueryBuilderInterface;
  13. trait MultilingualTrait
  14. {
  15.     /**
  16.      * Returns the ID of the fallback language.
  17.      */
  18.     public function getLanguageId()
  19.     {
  20.         $pidColumn = static::getPidColumn();
  21.         if ($this->{$pidColumn} > 0) {
  22.             return $this->{$pidColumn};
  23.         }
  24.         return $this->id;
  25.     }
  26.     /**
  27.      * Get the alias of a multilingual model.
  28.      *
  29.      * @param string $language
  30.      * @param string $aliasColumnName
  31.      *
  32.      * @return mixed
  33.      */
  34.     public function getAlias($language$aliasColumnName 'alias')
  35.     {
  36.         // Do not load any translation if already target language
  37.         $langColumn = static::getLangColumn();
  38.         $fallbackLang = static::getFallbackLanguage();
  39.         if ($language === $fallbackLang && !$this->{$langColumn}) {
  40.             return $this->{$aliasColumnName};
  41.         }
  42.         if ($language === $this->{$langColumn}) {
  43.             return $this->{$aliasColumnName};
  44.         }
  45.         // Try to load the translated model
  46.         $translatedModel = static::findByPk($this->getLanguageId(), ['language' => $language]);
  47.         if (null === $translatedModel) {
  48.             // Get fallback
  49.             if ($language === $fallbackLang) {
  50.                 return $this->{$aliasColumnName};
  51.             }
  52.             $fallbackModel = static::findByPk($this->getLanguageId(), ['language' => $fallbackLang]);
  53.             return $fallbackModel->{$aliasColumnName};
  54.         }
  55.         return $translatedModel->{$aliasColumnName};
  56.     }
  57.     /**
  58.      * Find a model by its alias.
  59.      *
  60.      * @param        $alias
  61.      * @param string $aliasColumnName
  62.      * @param array  $options
  63.      *
  64.      * @return mixed
  65.      */
  66.     public static function findByAlias($alias$aliasColumnName 'alias'$options = [])
  67.     {
  68.         $table = static::getTable();
  69.         $options array_merge(
  70.             [
  71.                 'limit' => 1,
  72.                 'column' => ["$table.$aliasColumnName=?"],
  73.                 'value' => [$alias],
  74.                 'return' => 'Model',
  75.             ],
  76.             $options
  77.         );
  78.         return static::find($options);
  79.     }
  80.     /**
  81.      * Find a model by its alias when using multilingal aliases.
  82.      *
  83.      * @param        $alias
  84.      * @param string $aliasColumnName
  85.      * @param array  $options
  86.      *
  87.      * @return mixed
  88.      */
  89.     public static function findByMultilingualAlias($alias$aliasColumnName 'alias'$options = [])
  90.     {
  91.         $table = static::getTable();
  92.         $options array_merge(
  93.             [
  94.                 'limit' => 1,
  95.                 'column' => ["($table.$aliasColumnName=? OR translation.$aliasColumnName=?)"],
  96.                 'value' => [$alias$alias],
  97.                 'return' => 'Model',
  98.             ],
  99.             $options
  100.         );
  101.         return static::find($options);
  102.     }
  103.     /**
  104.      * Get the language column.
  105.      *
  106.      * @return string
  107.      */
  108.     public static function getLangColumn()
  109.     {
  110.         static::ensureDataContainerIsLoaded();
  111.         return $GLOBALS['TL_DCA'][static::getTable()]['config']['langColumnName'] ?? 'language';
  112.     }
  113.     /**
  114.      * Get the fallback language if available.
  115.      *
  116.      * @return string|null
  117.      */
  118.     public static function getFallbackLanguage()
  119.     {
  120.         static::ensureDataContainerIsLoaded();
  121.         return $GLOBALS['TL_DCA'][static::getTable()]['config']['fallbackLang'] ?? null;
  122.     }
  123.     /**
  124.      * Build a query based on the given options.
  125.      * The method returns a QueryBuilder instance so you can easily modify
  126.      * the query in your child class. We can just return the instance as the
  127.      * QueryBuilder implements the __toString() method so we don't have to call
  128.      * ->getSql() manually.
  129.      *
  130.      * @param array $options The options array
  131.      *
  132.      * @return QueryBuilder
  133.      */
  134.     protected static function buildFindQuery(array $options)
  135.     {
  136.         $mlqb = static::getMultilingualQueryBuilder();
  137.         // Use the current language if none provided
  138.         if (!isset($options['language'])) {
  139.             $options['language'] = str_replace('-''_'$GLOBALS['TL_LANGUAGE']);
  140.         }
  141.         // Consider the fallback language
  142.         $fallbackLang = static::getFallbackLanguage();
  143.         if (null !== $fallbackLang && $fallbackLang === $options['language']) {
  144.             $options['language'] = '';
  145.         }
  146.         $mlqb->buildQueryBuilderForFind($options['language']);
  147.         static::applyOptionsToQueryBuilder($mlqb->getQueryBuilder(), $options);
  148.         return $mlqb->getQueryBuilder();
  149.     }
  150.     /**
  151.      * Build a query based on the given options to count the number of records.
  152.      * The method returns a QueryBuilder instance so you can easily modify
  153.      * the query in your child class. We can just return the instance as the
  154.      * QueryBuilder implements the __toString() method so we don't have to call
  155.      * ->getSql() manually.
  156.      *
  157.      * @param array $options The options array
  158.      *
  159.      * @return QueryBuilder
  160.      */
  161.     protected static function buildCountQuery(array $options)
  162.     {
  163.         $mlqb = static::getMultilingualQueryBuilder();
  164.         if (isset($options['having'])) {
  165.             $mlqb->buildQueryBuilderForCountWithSubQuery(static::buildFindQuery($options));
  166.         } else {
  167.             $mlqb->buildQueryBuilderForCount();
  168.             static::applyOptionsToQueryBuilder($mlqb->getQueryBuilder(), $options);
  169.         }
  170.         return $mlqb->getQueryBuilder();
  171.     }
  172.     /**
  173.      * Prevent model from saving when creating a model from a database result. See #51.
  174.      *
  175.      * @param Result $objResult The database result object
  176.      *
  177.      * @return static The model
  178.      */
  179.     protected static function createModelFromDbResult(Result $objResult)
  180.     {
  181.         $model = new static($objResult);
  182.         $model->preventSaving(false);
  183.         return $model;
  184.     }
  185.     /**
  186.      * Prevent new models from saving when creating a new collection from a database result. See #51.
  187.      *
  188.      * @param Result $objResult The database result object
  189.      * @param string $strTable  The table name
  190.      *
  191.      * @return Collection The model collection
  192.      */
  193.     protected static function createCollectionFromDbResult(Result $objResult$strTable)
  194.     {
  195.         $collection Collection::createFromDbResult($objResult$strTable);
  196.         /** @var self $model */
  197.         foreach ($collection as $model) {
  198.             $model->preventSaving(false);
  199.         }
  200.         return $collection->reset();
  201.     }
  202.     /**
  203.      * Apply the model options to the query builder.
  204.      */
  205.     protected static function applyOptionsToQueryBuilder(QueryBuilder $qb, array $options): void
  206.     {
  207.         // Columns
  208.         if (!empty($options['column'])) {
  209.             if (\is_array($options['column'])) {
  210.                 foreach ($options['column'] as $column) {
  211.                     $qb->andWhere($column);
  212.                 }
  213.             } else {
  214.                 // Default is likely fallback table
  215.                 $table = static::getTable();
  216.                 $qb->andWhere("$table.{$options['column']}=?");
  217.             }
  218.         }
  219.         // Group by
  220.         if (!empty($options['group'])) {
  221.             $qb->groupBy($options['group']);
  222.         }
  223.         // Having
  224.         if (!empty($options['having'])) {
  225.             $qb->having($options['having']);
  226.         }
  227.         // Order by
  228.         if (!empty($options['order'])) {
  229.             $qb->add('orderBy'$options['order']);
  230.         }
  231.     }
  232.     /**
  233.      * Get the MultilingualQueryBuilder.
  234.      *
  235.      * @return MultilingualQueryBuilderInterface
  236.      */
  237.     protected static function getMultilingualQueryBuilder()
  238.     {
  239.         /** @var MultilingualQueryBuilderFactoryInterface $factory */
  240.         $factory System::getContainer()->get('terminal42.dc_multilingual.querybuilder_factory');
  241.         return $factory->build(
  242.             static::getTable(),
  243.             static::getPidColumn(),
  244.             static::getLangColumn(),
  245.             static::getRegularFields(),
  246.             static::getTranslatableFields()
  247.         );
  248.     }
  249.     /**
  250.      * Get the regular fields.
  251.      *
  252.      * @return array
  253.      */
  254.     protected static function getRegularFields()
  255.     {
  256.         $extractor DcaExtractor::getInstance(static::getTable());
  257.         $tableColumns Database::getInstance()->getFieldNames(static::getTable());
  258.         return array_intersect($tableColumnsarray_keys($extractor->getFields()));
  259.     }
  260.     /**
  261.      * Get the fields that are translatable.
  262.      *
  263.      * @return array
  264.      */
  265.     protected static function getTranslatableFields()
  266.     {
  267.         static::ensureDataContainerIsLoaded();
  268.         $fields = [];
  269.         $tableColumns Database::getInstance()->getFieldNames(static::getTable());
  270.         foreach ($GLOBALS['TL_DCA'][static::getTable()]['fields'] as $field => $data) {
  271.             if (!isset($data['eval']['translatableFor']) || !\in_array($field$tableColumnstrue)) {
  272.                 continue;
  273.             }
  274.             $fields[] = $field;
  275.         }
  276.         return $fields;
  277.     }
  278.     /**
  279.      * Get the PID column.
  280.      *
  281.      * @return string
  282.      */
  283.     protected static function getPidColumn()
  284.     {
  285.         static::ensureDataContainerIsLoaded();
  286.         return $GLOBALS['TL_DCA'][static::getTable()]['config']['langPid'] ?? 'langPid';
  287.     }
  288.     /**
  289.      * Ensure the data container is loaded.
  290.      */
  291.     protected static function ensureDataContainerIsLoaded(): void
  292.     {
  293.         if (!isset($GLOBALS['TL_DCA'][static::getTable()])) {
  294.             $loader = new DcaLoader(static::getTable());
  295.             $loader->load();
  296.         }
  297.     }
  298. }