getContextDefinitions($plugin_definition); if ($context_definitions) { // Check the set of contexts against the requirements. return $this->checkRequirements($contexts, $context_definitions); } // If this plugin doesn't need any context, it is available to use. return TRUE; }); } /** * Returns the context definitions associated with a plugin definition. * * @param array|\Drupal\Component\Plugin\Definition\ContextAwarePluginDefinitionInterface $plugin_definition * The plugin definition. * * @return \Drupal\Component\Plugin\Context\ContextDefinitionInterface[]|null * The context definitions, or NULL if the plugin definition does not * support contexts. */ protected function getContextDefinitions($plugin_definition) { if ($plugin_definition instanceof ContextAwarePluginDefinitionInterface) { return $plugin_definition->getContextDefinitions(); } if (is_array($plugin_definition) && isset($plugin_definition['context'])) { return $plugin_definition['context']; } return NULL; } /** * {@inheritdoc} */ public function checkRequirements(array $contexts, array $requirements) { foreach ($requirements as $requirement) { if ($requirement->isRequired() && !$this->getMatchingContexts($contexts, $requirement)) { return FALSE; } } return TRUE; } /** * {@inheritdoc} */ public function getMatchingContexts(array $contexts, ContextDefinitionInterface $definition) { return array_filter($contexts, function (ContextInterface $context) use ($definition) { return $definition->isSatisfiedBy($context); }); } /** * {@inheritdoc} */ public function applyContextMapping(ContextAwarePluginInterface $plugin, $contexts, $mappings = []) { /** @var $contexts \Drupal\Core\Plugin\Context\ContextInterface[] */ $mappings += $plugin->getContextMapping(); // Loop through each of the expected contexts. $missing_value = []; foreach ($plugin->getContextDefinitions() as $plugin_context_id => $plugin_context_definition) { // If this context was given a specific name, use that. $context_id = isset($mappings[$plugin_context_id]) ? $mappings[$plugin_context_id] : $plugin_context_id; if (!empty($contexts[$context_id])) { // This assignment has been used, remove it. unset($mappings[$plugin_context_id]); // Plugins have their on context objects, only the value is applied. // They also need to know about the cacheability metadata of where that // value is coming from, so pass them through to those objects. $plugin_context = $plugin->getContext($plugin_context_id); if ($plugin_context instanceof ContextInterface && $contexts[$context_id] instanceof CacheableDependencyInterface) { $plugin_context->addCacheableDependency($contexts[$context_id]); } // Pass the value to the plugin if there is one. if ($contexts[$context_id]->hasContextValue()) { $plugin->setContextValue($plugin_context_id, $contexts[$context_id]->getContextData()); } elseif ($plugin_context_definition->isRequired()) { // Collect required contexts that exist but are missing a value. $missing_value[] = $plugin_context_id; } } elseif ($plugin_context_definition->isRequired()) { // Collect required contexts that are missing. $missing_value[] = $plugin_context_id; } else { // Ignore mappings for optional missing context. unset($mappings[$plugin_context_id]); } } // If there are any mappings that were not satisfied, throw an exception. // This is a more severe problem than missing values, so check and throw // this first. if (!empty($mappings)) { throw new ContextException('Assigned contexts were not satisfied: ' . implode(',', array_keys($mappings))); } // If there are any required contexts without a value, throw an exception. if ($missing_value) { throw new MissingValueContextException($missing_value); } } }