Как программно изменить активную тему Drupal 8?
В Drupal 6 мы использовали следующий код.
global $custom_theme;
$custom_theme = 'garland';
В Drupal 7 мы использовали hook_custom_theme()
.
В Drupal 8, как правильно это сделать?
В Drupal 8 вы используете согласователи тем , которые по сути являются сервисами, использующими определенный тег. Посмотрите тему переговорщиков, реализованную Drupal, чтобы точно понять, как они работают; пример, приведенный в записи изменений, не обновляется.
theme.negotiator.admin_theme:
class: Drupal\user\Theme\AdminNegotiator
arguments: ['@current_user', '@config.factory', '@entity.manager', '@router.admin_context']
tags:
- { name: theme_negotiator, priority: -40 }
namespace Drupal\user\Theme;
use Drupal\Core\Config\ConfigFactoryInterface;
use Drupal\Core\Entity\EntityManagerInterface;
use Drupal\Core\Routing\AdminContext;
use Drupal\Core\Routing\RouteMatchInterface;
use Drupal\Core\Session\AccountInterface;
use Drupal\Core\Theme\ThemeNegotiatorInterface;
/**
* Sets the active theme on admin pages.
*/
class AdminNegotiator implements ThemeNegotiatorInterface {
/**
* The current user.
*
* @var \Drupal\Core\Session\AccountInterface
*/
protected $user;
/**
* The config factory.
*
* @var \Drupal\Core\Config\ConfigFactoryInterface
*/
protected $configFactory;
/**
* The entity manager.
*
* @var \Drupal\Core\Entity\EntityManagerInterface
*/
protected $entityManager;
/**
* The route admin context to determine whether a route is an admin one.
*
* @var \Drupal\Core\Routing\AdminContext
*/
protected $adminContext;
/**
* Creates a new AdminNegotiator instance.
*
* @param \Drupal\Core\Session\AccountInterface $user
* The current user.
* @param \Drupal\Core\Config\ConfigFactoryInterface $config_factory
* The config factory.
* @param \Drupal\Core\Entity\EntityManagerInterface $entity_manager
* The entity manager.
* @param \Drupal\Core\Routing\AdminContext $admin_context
* The route admin context to determine whether the route is an admin one.
*/
public function __construct(AccountInterface $user, ConfigFactoryInterface $config_factory, EntityManagerInterface $entity_manager, AdminContext $admin_context) {
$this->user = $user;
$this->configFactory = $config_factory;
$this->entityManager = $entity_manager;
$this->adminContext = $admin_context;
}
/**
* {@inheritdoc}
*/
public function applies(RouteMatchInterface $route_match) {
return ($this->entityManager->hasHandler('user_role', 'storage') && $this->user->hasPermission('view the administration theme') && $this->adminContext->isAdminRoute($route_match->getRouteObject()));
}
/**
* {@inheritdoc}
*/
public function determineActiveTheme(RouteMatchInterface $route_match) {
return $this->configFactory->get('system.theme')->get('admin');
}
}
Код довольно прост для понимания: applies()
метод возвращается, TRUE
когда текущий маршрут является тем, для которого ваш модуль хочет изменить тему; determineActiveTheme()
метод возвращает имя темы машинной темы применения.
См. Также ThemeNegotiator :: defineActiveTheme () не должна требовать передачи RouteMatch для возможного изменения аргументов, полученных из методов, используемых участниками переговоров по теме; если этот патч будет применен, вам также потребуется изменить код посредника по теме.