Sort by Entityqueue

Created on 29 November 2018, over 6 years ago
Updated 11 October 2023, over 1 year ago

Let me preface this thread with saying that I've tried several related modules related to the idea of "prioritizing results" with Search API, but none of them seem to fit our particular use-case.

Problem:

  1. We have a Search API index for a single Content Type on our site. With that, we have a corresponding View set up to list these.
  2. These results are sorted alphabetically by title.
  3. Before the fallback alphabetical sorting, we would like to have "priority content", which gets pushed to the top of the results. The order of "priority content" (there will be multiple items prioritized) needs to be controlled as well, so a simple "flag as priority" wouldn't work.
  4. There doesn't appear to be any obvious way to create this functionality with Search API or related modules.

We use the Entityqueue module throughout our site to define order of several other entities, so Entityqueue seems like the natural fit for something like this as well. We've been able to accomplish this with Entityqueue + Views outside of Search API, and are quite happy with the results. In addition, we were able to accomplish something similar in a Drupal 7 site with Search API + Nodequeue following some of the advice here:
https://www.drupal.org/project/nodequeue/issues/1480882 β†’

However, it doesn't appear that Entityqueue plays nicely with Search API due to the way data is stored in Search API (which is understandable). I could see this being a question for the Entityqueue module as well.

Mostly, I'm curious if there's a different way to approach this problem as a whole. The desired effect should still be a "prioritization of content" more so than a "flag" system, and a "weight" field on the node feels a bit clunky, but maybe there's something I'm missing or a better solution that would play nicely with Search API.

✨ Feature request
Status

Closed: works as designed

Version

1.0

Component

General code

Created by

πŸ‡ΊπŸ‡ΈUnited States Nuuou Lincoln, NE

Live updates comments and jobs are added and updated live.
Sign in to follow issues

Comments & Activities

Not all content is available!

It's likely this issue predates Contrib.social: some issue and comment data are missing.

  • πŸ‡ΊπŸ‡ΈUnited States thalemn

    We would like to use Entityqueue with Search API...@Nuuou: can you share your solution for D10?

    Thanks!

  • πŸ‡¬πŸ‡§United Kingdom SirClickALot Somerset

    I too would like to do exactly the same thing, i.e. use Entity queue to manage the 'weight' of my nodes globally and then pass that global classification onto Search API for use in sorting.

    So, I second that request for you to share your custom code.

    Once I have that all working over here, my plan this is to see if I can reproduce the whole affair using the ECA module suit which I am trying to use in place of any custom code.

    Thanks

  • Status changed to Active over 1 year ago
  • πŸ‡¬πŸ‡§United Kingdom SirClickALot Somerset
  • πŸ‡ΊπŸ‡ΈUnited States Nuuou Lincoln, NE

    Hey all! It has been quite some time since I worked on something that needed this, so I don't have anything specific to share.
    However, my last post above is pretty accurate for how I approached it at the time.

    1. "Weight" field on nodes that need it (probably just stored this as a number). Default value of something large.
    2. Hook into the entityqueue save and load/update the node "Weights" for all nodes in the queue so they match the "weight" of the queue
    3. Use that "Weight" field to sort in Views as needed

    Likely better approaches now, but the above worked a-okay for me at the time.
    drunken monkey above mentioned a different approach too that would likely be more robust.

    Good luck!

  • πŸ‡ΊπŸ‡ΈUnited States thalemn

    @Nuuou thanks so much for your quick reply. I think we can implement something based on your description, but would you be willing to provide a patch for the entityqueue module that would provide this solution for everyone?

  • πŸ‡ΊπŸ‡ΈUnited States Nuuou Lincoln, NE

    With the approach I took, there's nothing to patch in the Enitityqueue module itself as it's just using existing hooks.

    That approach would be unique to your project's needs without further abstracting it into a standalone feature, which I don't think my approach is suited for. If something were to implement this with Search API/Entityqueue, I think it'd need to be more robust, such as writing a processor mentioned above!

    Here's the code I used though, more-or-less. Looking at it now, this could definitely be improved:

    /**
     * Implements hook_form_alter().
     */
    function YOUR_MODULE_form_alter(&$form, FormStateInterface $form_state, $form_id) {
      // Only run when entity queue saved.
      if ($form_id === 'entity_subqueue_YOUR_QUEUE_edit_form') {
        $form['actions']['submit']['#submit'][] = '_YOUR_MODULE_entityqueue_submit';
      }
    }
    
    function _YOUR_MODULE_entityqueue_submit($form, FormStateInterface $form_state) {
      if ($form_state->hasValue('items')) {
        // Get all Node IDs from items in EntityQueue.
        $items = $form_state->getValue('items');
        $queue_node_ids = [];
        foreach ($items as $item) {
          if (isset($item['target_id'])) {
            $queue_node_ids[$item['target_id']] = $item['_weight'];
          }
        }
    
        // Get all Node IDs of Nodes to consider.
        $all_node_ids = \Drupal::entityQuery('node')
          ->condition('status', 1)
          ->accessCheck(TRUE)
          ->execute();
    
        // Load all Nodes, set weight based on their existence in EntityQueue.
        $all_nodes = Node::loadMultiple($all_node_ids);
        foreach ($all_nodes as $node) {
          $weight = 1000;
          if (isset($queue_node_ids[$node->id()])) {
            $weight = $queue_node_ids[$node->id()];
          }
          $node->set('field_weight', $weight);
          $node->save();
        }
      }
    }
    

    Unfortunately, since this issue is quite old, I don't have the capacity to pursue approach that more robust processor at this time. Hopefully the above approach I took helps for you though! Maybe one day someone smarter than myself can help write some proper Search API/Entityqueue integration haha. 🀞

  • πŸ‡ΊπŸ‡ΈUnited States thalemn

    Thanks for your solution. I'll take a look at our situation and see if we can make it work.

  • πŸ‡ΊπŸ‡ΈUnited States mlncn Minneapolis, MN, USA

    Have the same need, in this case would want to filter as well as sortβ€” only show items that are in a given queue. A single processor plugin should handle both, right?

    Note that this has also been raised from the Entityqueue side, ✨ Search API/ SOLR integration Active

    And probably the processor should live there?

  • Status changed to Closed: duplicate about 1 year ago
  • πŸ‡·πŸ‡΄Romania amateescu

    I agree with #12, this is something that can be added to the Entityqueue module, so I'm closing this as a duplicate.

  • πŸ‡ΊπŸ‡ΈUnited States mlncn Minneapolis, MN, USA

    Advice from the search api side very welcome!

  • πŸ‡ΊπŸ‡ΈUnited States mlncn Minneapolis, MN, USA

    I have an issue fork going over in Entityqueue and would very much appreciate Search API eyes on an initial review: https://git.drupalcode.org/issue/entityqueue-3004722/-/compare/8.x-1.x.....

    It still needs to be made configurable per entityqueue. Can that be done at the Views integration layer (or is it better off done somewhere else anyway)?

  • πŸ‡ΊπŸ‡ΈUnited States mlncn Minneapolis, MN, USA

    The current problem with the code over in ✨ Search API/ SOLR integration Active is that it does not trigger an update to the index when an item is added, removed, or re-ordered in an update queue. How would that be done in Search API?

      \Drupal::getContainer()->get('search_api.entity_datasource.tracking_manager')
        ->entityUpdate($entity);
    

    as in https://git.drupalcode.org/project/search_api/blob/HEAD/search_api.modul... ?

Production build 0.71.5 2024