You are here

class RouteSubscriber in Facets Pretty Paths 8

Alter facet source routes, adding a parameter.

Hierarchy

Expanded class hierarchy of RouteSubscriber

1 string reference to 'RouteSubscriber'
facets_pretty_paths.services.yml in ./facets_pretty_paths.services.yml
facets_pretty_paths.services.yml
1 service uses RouteSubscriber
facets_pretty_paths.route_subscriber in ./facets_pretty_paths.services.yml
Drupal\facets_pretty_paths\RouteSubscriber

File

src/RouteSubscriber.php, line 13

Namespace

Drupal\facets_pretty_paths
View source
class RouteSubscriber extends RouteSubscriberBase {

  /**
   * Service plugin.manager.facet_source.
   *
   * @var \Drupal\facets\FacetSource\FacetSourcePluginManager
   */
  protected $facetSourcePluginManager;

  /**
   * Constructs a RouteSubscriber object.
   *
   * @param Drupal\facets\FacetSource\FacetSourcePluginManager $facetSourcePluginManager
   *   The plugin.manager.facets.facet_source service.
   */
  public function __construct(FacetSourcePluginManager $facetSourcePluginManager) {
    $this->facetSourcePluginManager = $facetSourcePluginManager;
  }

  /**
   * {@inheritdoc}
   */
  public function alterRoutes(RouteCollection $collection) {
    $sources = $this->facetSourcePluginManager
      ->getDefinitions();
    foreach ($sources as $source) {
      $sourcePlugin = $this->facetSourcePluginManager
        ->createInstance($source['id']);
      $path = $sourcePlugin
        ->getPath();
      $storage = \Drupal::entityTypeManager()
        ->getStorage('facets_facet_source');
      $source_id = str_replace(':', '__', $sourcePlugin
        ->getPluginId());
      $facet_source = $storage
        ->load($source_id);
      if (!$facet_source || $facet_source
        ->getUrlProcessorName() != 'facets_pretty_paths') {

        // If no custom configuration is set for the facet source, it is not
        // using pretty_paths. If there is custom configuration, ensure the url
        // processor is pretty paths.
        continue;
      }
      try {
        $url = Url::fromUri('internal:' . $path);
        $sourceRoute = $collection
          ->get($url
          ->getRouteName());

        // Ensure this only triggers once per route.
        // See https://www.drupal.org/project/facets_pretty_paths/issues/2984105
        if ($sourceRoute && strpos($sourceRoute
          ->getPath(), '/{facets_query}') === FALSE) {
          $sourceRoute
            ->setPath($sourceRoute
            ->getPath() . '/{facets_query}');
          $sourceRoute
            ->setDefault('facets_query', '');
          $sourceRoute
            ->setRequirement('facets_query', '.*');

          // Core improperly checks for route parameters that can have a slash
          // in them, only making the route match for parameters that don't
          // have a slash.
          // Workaround that here by adding fake optional parameters to the
          // route, that'll never be filled, and won't get any value set because
          // {facets_query} will already have matched the whole path.
          // Note that until the core bug is resolved, the path maximum length
          // of 255 in the router table induces a limit to the number of facets
          // that can be triggered, which will depend on the facets source path
          // length. For a base path of "/search", 40 placeholders can be added,
          // which means 20 active filter pairs.
          // See https://www.drupal.org/project/drupal/issues/2741939
          $routePath = $sourceRoute
            ->getPath();
          for ($i = 0; strlen($routePath) < 250; $i++) {
            $sourceRoute
              ->setDefault('f' . $i, '');
            $routePath .= "/{f{$i}}";
          }
          $sourceRoute
            ->setPath($routePath);
        }
      } catch (\Exception $e) {
      }
    }
  }

}

Members

Namesort descending Modifiers Type Description Overrides
RouteSubscriber::$facetSourcePluginManager protected property Service plugin.manager.facet_source.
RouteSubscriber::alterRoutes public function Alters existing routes for a specific collection. Overrides RouteSubscriberBase::alterRoutes
RouteSubscriber::__construct public function Constructs a RouteSubscriber object.
RouteSubscriberBase::getSubscribedEvents public static function Returns an array of event names this subscriber wants to listen to. 5
RouteSubscriberBase::onAlterRoutes public function Delegates the route altering to self::alterRoutes(). 1