| Code Coverage | ||||||||||
| Classes and Traits | Functions and Methods | Lines | ||||||||
| Total |  | 0.00% | 0 / 1 |  | 0.00% | 0 / 6 | CRAP |  | 0.00% | 0 / 111 | 
| HtmlRenderer |  | 0.00% | 0 / 1 |  | 0.00% | 0 / 6 | 702 |  | 0.00% | 0 / 111 | 
| __construct |  | 0.00% | 0 / 1 | 2 |  | 0.00% | 0 / 8 | |||
| renderResponse |  | 0.00% | 0 / 1 | 12 |  | 0.00% | 0 / 23 | |||
| anonymous function |  | 0.00% | 0 / 1 | 6 |  | 0.00% | 0 / 5 | |||
| prepare |  | 0.00% | 0 / 1 | 132 |  | 0.00% | 0 / 11 | |||
| invokePageAttachmentHooks |  | 0.00% | 0 / 1 | 20 |  | 0.00% | 0 / 15 | |||
| buildPageTopAndBottom |  | 0.00% | 0 / 1 | 30 |  | 0.00% | 0 / 17 | |||
| <?php | |
| /** | |
| * @file | |
| * Contains \Drupal\Core\Render\MainContent\HtmlRenderer. | |
| */ | |
| namespace Drupal\Core\Render\MainContent; | |
| use Drupal\Component\Plugin\PluginManagerInterface; | |
| use Drupal\Core\Cache\Cache; | |
| use Drupal\Core\Controller\TitleResolverInterface; | |
| use Drupal\Core\Display\PageVariantInterface; | |
| use Drupal\Core\Extension\ModuleHandlerInterface; | |
| use Drupal\Core\Display\ContextAwareVariantInterface; | |
| use Drupal\Core\Render\HtmlResponse; | |
| use Drupal\Core\Render\PageDisplayVariantSelectionEvent; | |
| use Drupal\Core\Render\RenderCacheInterface; | |
| use Drupal\Core\Render\RenderContext; | |
| use Drupal\Core\Render\RendererInterface; | |
| use Drupal\Core\Render\RenderEvents; | |
| use Drupal\Core\Routing\RouteMatchInterface; | |
| use Symfony\Component\EventDispatcher\EventDispatcherInterface; | |
| use Symfony\Component\HttpFoundation\Request; | |
| /** | |
| * Default main content renderer for HTML requests. | |
| * | |
| * For attachment handling of HTML responses: | |
| * @see template_preprocess_html() | |
| * @see \Drupal\Core\Render\AttachmentsResponseProcessorInterface | |
| * @see \Drupal\Core\Render\BareHtmlPageRenderer | |
| * @see \Drupal\Core\Render\HtmlResponse | |
| * @see \Drupal\Core\Render\HtmlResponseAttachmentsProcessor | |
| */ | |
| class HtmlRenderer implements MainContentRendererInterface { | |
| /** | |
| * The title resolver. | |
| * | |
| * @var \Drupal\Core\Controller\TitleResolverInterface | |
| */ | |
| protected $titleResolver; | |
| /** | |
| * The display variant manager. | |
| * | |
| * @var \Drupal\Component\Plugin\PluginManagerInterface | |
| */ | |
| protected $displayVariantManager; | |
| /** | |
| * The event dispatcher. | |
| * | |
| * @var \Symfony\Component\EventDispatcher\EventDispatcherInterface | |
| */ | |
| protected $eventDispatcher; | |
| /** | |
| * The module handler. | |
| * | |
| * @var \Drupal\Core\Extension\ModuleHandlerInterface | |
| */ | |
| protected $moduleHandler; | |
| /** | |
| * The renderer service. | |
| * | |
| * @var \Drupal\Core\Render\RendererInterface | |
| */ | |
| protected $renderer; | |
| /** | |
| * The render cache service. | |
| * | |
| * @var \Drupal\Core\Render\RenderCacheInterface | |
| */ | |
| protected $renderCache; | |
| /** | |
| * The renderer configuration array. | |
| * | |
| * @see sites/default/default.services.yml | |
| * | |
| * @var array | |
| */ | |
| protected $rendererConfig; | |
| /** | |
| * Constructs a new HtmlRenderer. | |
| * | |
| * @param \Drupal\Core\Controller\TitleResolverInterface $title_resolver | |
| * The title resolver. | |
| * @param \Drupal\Component\Plugin\PluginManagerInterface $display_variant_manager | |
| * The display variant manager. | |
| * @param \Symfony\Component\EventDispatcher\EventDispatcherInterface $event_dispatcher | |
| * The event dispatcher. | |
| * @param \Drupal\Core\Extension\ModuleHandlerInterface $module_handler | |
| * The module handler. | |
| * @param \Drupal\Core\Render\RendererInterface $renderer | |
| * The renderer service. | |
| * @param \Drupal\Core\Render\RenderCacheInterface $render_cache | |
| * The render cache service. | |
| * @param array $renderer_config | |
| * The renderer configuration array. | |
| */ | |
| public function __construct(TitleResolverInterface $title_resolver, PluginManagerInterface $display_variant_manager, EventDispatcherInterface $event_dispatcher, ModuleHandlerInterface $module_handler, RendererInterface $renderer, RenderCacheInterface $render_cache, array $renderer_config) { | |
| $this->titleResolver = $title_resolver; | |
| $this->displayVariantManager = $display_variant_manager; | |
| $this->eventDispatcher = $event_dispatcher; | |
| $this->moduleHandler = $module_handler; | |
| $this->renderer = $renderer; | |
| $this->renderCache = $render_cache; | |
| $this->rendererConfig = $renderer_config; | |
| } | |
| /** | |
| * {@inheritdoc} | |
| * | |
| * The entire HTML: takes a #type 'page' and wraps it in a #type 'html'. | |
| */ | |
| public function renderResponse(array $main_content, Request $request, RouteMatchInterface $route_match) { | |
| list($page, $title) = $this->prepare($main_content, $request, $route_match); | |
| if (!isset($page['#type']) || $page['#type'] !== 'page') { | |
| throw new \LogicException('Must be #type page'); | |
| } | |
| $page['#title'] = $title; | |
| // Now render the rendered page.html.twig template inside the html.html.twig | |
| // template, and use the bubbled #attached metadata from $page to ensure we | |
| // load all attached assets. | |
| $html = [ | |
| '#type' => 'html', | |
| 'page' => $page, | |
| ]; | |
| // The special page regions will appear directly in html.html.twig, not in | |
| // page.html.twig, hence add them here, just before rendering html.html.twig. | |
| $this->buildPageTopAndBottom($html); | |
| // Render, but don't replace placeholders yet, because that happens later in | |
| // the render pipeline. To not replace placeholders yet, we use | |
| // RendererInterface::render() instead of RendererInterface::renderRoot(). | |
| // @see \Drupal\Core\Render\HtmlResponseAttachmentsProcessor. | |
| $render_context = new RenderContext(); | |
| $this->renderer->executeInRenderContext($render_context, function() use (&$html) { | |
| // RendererInterface::render() renders the $html render array and updates | |
| // it in place. We don't care about the return value (which is just | |
| // $html['#markup']), but about the resulting render array. | |
| // @todo Simplify this when https://www.drupal.org/node/2495001 lands. | |
| $this->renderer->render($html); | |
| }); | |
| // RendererInterface::render() always causes bubbleable metadata to be | |
| // stored in the render context, no need to check it conditionally. | |
| $bubbleable_metadata = $render_context->pop(); | |
| $bubbleable_metadata->applyTo($html); | |
| $content = $this->renderCache->getCacheableRenderArray($html); | |
| // Also associate the required cache contexts. | |
| // (Because we use ::render() above and not ::renderRoot(), we manually must | |
| // ensure the HTML response varies by the required cache contexts.) | |
| $content['#cache']['contexts'] = Cache::mergeContexts($content['#cache']['contexts'], $this->rendererConfig['required_cache_contexts']); | |
| // Also associate the "rendered" cache tag. This allows us to invalidate the | |
| // entire render cache, regardless of the cache bin. | |
| $content['#cache']['tags'][] = 'rendered'; | |
| $response = new HtmlResponse($content, 200, [ | |
| 'Content-Type' => 'text/html; charset=UTF-8', | |
| ]); | |
| return $response; | |
| } | |
| /** | |
| * Prepares the HTML body: wraps the main content in #type 'page'. | |
| * | |
| * @param array $main_content | |
| * The render array representing the main content. | |
| * @param \Symfony\Component\HttpFoundation\Request $request | |
| * The request object, for context. | |
| * @param \Drupal\Core\Routing\RouteMatchInterface $route_match | |
| * The route match, for context. | |
| * | |
| * @return array | |
| * An array with two values: | |
| * 0. A #type 'page' render array. | |
| * 1. The page title. | |
| * | |
| * @throws \LogicException | |
| * If the selected display variant does not implement PageVariantInterface. | |
| */ | |
| protected function prepare(array $main_content, Request $request, RouteMatchInterface $route_match) { | |
| // Determine the title: use the title provided by the main content if any, | |
| // otherwise get it from the routing information. | |
| $get_title = function (array $main_content) use ($request, $route_match) { | |
| return isset($main_content['#title']) ? $main_content['#title'] : $this->titleResolver->getTitle($request, $route_match->getRouteObject()); | |
| }; | |
| // If the _controller result already is #type => page, | |
| // we have no work to do: The "main content" already is an entire "page" | |
| // (see html.html.twig). | |
| if (isset($main_content['#type']) && $main_content['#type'] === 'page') { | |
| $page = $main_content; | |
| $title = $get_title($page); | |
| } | |
| // Otherwise, render it as the main content of a #type => page, by selecting | |
| // page display variant to do that and building that page display variant. | |
| else { | |
| // Select the page display variant to be used to render this main content, | |
| // default to the built-in "simple page". | |
| $event = new PageDisplayVariantSelectionEvent('simple_page', $route_match); | |
| $this->eventDispatcher->dispatch(RenderEvents::SELECT_PAGE_DISPLAY_VARIANT, $event); | |
| $variant_id = $event->getPluginId(); | |
| // We must render the main content now already, because it might provide a | |
| // title. We set its $is_root_call parameter to FALSE, to ensure | |
| // placeholders are not yet replaced. This is essentially "pre-rendering" | |
| // the main content, the "full rendering" will happen in | |
| // ::renderResponse(). | |
| // @todo Remove this once https://www.drupal.org/node/2359901 lands. | |
| if (!empty($main_content)) { | |
| $this->renderer->executeInRenderContext(new RenderContext(), function() use (&$main_content) { | |
| if (isset($main_content['#cache']['keys'])) { | |
| // Retain #title, otherwise, dynamically generated titles would be | |
| // missing for controllers whose entire returned render array is | |
| // render cached. | |
| $main_content['#cache_properties'][] = '#title'; | |
| } | |
| return $this->renderer->render($main_content, FALSE); | |
| }); | |
| $main_content = $this->renderCache->getCacheableRenderArray($main_content) + [ | |
| '#title' => isset($main_content['#title']) ? $main_content['#title'] : NULL | |
| ]; | |
| } | |
| $title = $get_title($main_content); | |
| // Instantiate the page display, and give it the main content. | |
| $page_display = $this->displayVariantManager->createInstance($variant_id); | |
| if (!$page_display instanceof PageVariantInterface) { | |
| throw new \LogicException('Cannot render the main content for this page because the provided display variant does not implement PageVariantInterface.'); | |
| } | |
| $page_display | |
| ->setMainContent($main_content) | |
| ->setTitle($title) | |
| ->addCacheableDependency($event) | |
| ->setConfiguration($event->getPluginConfiguration()); | |
| // Some display variants need to be passed an array of contexts with | |
| // values because they can't get all their contexts globally. For example, | |
| // in Page Manager, you can create a Page which has a specific static | |
| // context (e.g. a context that refers to the Node with nid 6), if any | |
| // such contexts were added to the $event, pass them to the $page_display. | |
| if ($page_display instanceof ContextAwareVariantInterface) { | |
| $page_display->setContexts($event->getContexts()); | |
| } | |
| // Generate a #type => page render array using the page display variant, | |
| // the page display will build the content for the various page regions. | |
| $page = array( | |
| '#type' => 'page', | |
| ); | |
| $page += $page_display->build(); | |
| } | |
| // $page is now fully built. Find all non-empty page regions, and add a | |
| // theme wrapper function that allows them to be consistently themed. | |
| $regions = \Drupal::theme()->getActiveTheme()->getRegions(); | |
| foreach ($regions as $region) { | |
| if (!empty($page[$region])) { | |
| $page[$region]['#theme_wrappers'][] = 'region'; | |
| $page[$region]['#region'] = $region; | |
| } | |
| } | |
| // Allow hooks to add attachments to $page['#attached']. | |
| $this->invokePageAttachmentHooks($page); | |
| return [$page, $title]; | |
| } | |
| /** | |
| * Invokes the page attachment hooks. | |
| * | |
| * @param array &$page | |
| * A #type 'page' render array, for which the page attachment hooks will be | |
| * invoked and to which the results will be added. | |
| * | |
| * @throws \LogicException | |
| * | |
| * @internal | |
| * | |
| * @see hook_page_attachments() | |
| * @see hook_page_attachments_alter() | |
| */ | |
| public function invokePageAttachmentHooks(array &$page) { | |
| // Modules can add attachments. | |
| $attachments = []; | |
| foreach ($this->moduleHandler->getImplementations('page_attachments') as $module) { | |
| $function = $module . '_page_attachments'; | |
| $function($attachments); | |
| } | |
| if (array_diff(array_keys($attachments), ['#attached', '#cache']) !== []) { | |
| throw new \LogicException('Only #attached and #cache may be set in hook_page_attachments().'); | |
| } | |
| // Modules and themes can alter page attachments. | |
| $this->moduleHandler->alter('page_attachments', $attachments); | |
| \Drupal::theme()->alter('page_attachments', $attachments); | |
| if (array_diff(array_keys($attachments), ['#attached', '#cache']) !== []) { | |
| throw new \LogicException('Only #attached and #cache may be set in hook_page_attachments_alter().'); | |
| } | |
| // Merge the attachments onto the $page render array. | |
| $page = $this->renderer->mergeBubbleableMetadata($page, $attachments); | |
| } | |
| /** | |
| * Invokes the page top and bottom hooks. | |
| * | |
| * @param array &$html | |
| * A #type 'html' render array, for which the page top and bottom hooks will | |
| * be invoked, and to which the 'page_top' and 'page_bottom' children (also | |
| * render arrays) will be added (if non-empty). | |
| * | |
| * @throws \LogicException | |
| * | |
| * @internal | |
| * | |
| * @see hook_page_top() | |
| * @see hook_page_bottom() | |
| * @see html.html.twig | |
| */ | |
| public function buildPageTopAndBottom(array &$html) { | |
| // Modules can add render arrays to the top and bottom of the page. | |
| $page_top = []; | |
| $page_bottom = []; | |
| foreach ($this->moduleHandler->getImplementations('page_top') as $module) { | |
| $function = $module . '_page_top'; | |
| $function($page_top); | |
| } | |
| foreach ($this->moduleHandler->getImplementations('page_bottom') as $module) { | |
| $function = $module . '_page_bottom'; | |
| $function($page_bottom); | |
| } | |
| if (!empty($page_top)) { | |
| $html['page_top'] = $page_top; | |
| } | |
| if (!empty($page_bottom)) { | |
| $html['page_bottom'] = $page_bottom; | |
| } | |
| } | |
| } |