diff --git a/config/vufind/combined.ini b/config/vufind/combined.ini index 1e1f124502290c4bf749e397874936a36e06bb16..48625a997ba82d8e92ac615eefe94871dd613d39 100644 --- a/config/vufind/combined.ini +++ b/config/vufind/combined.ini @@ -1,6 +1,10 @@ ; This file controls the "combined search" module. Each section is named for a -; search backend (e.g. "Solr", "Summon", "WorldCat", etc.) and contains the -; following settings: +; search backend (e.g. "Solr", "Summon", "WorldCat", etc.). If you want to create +; multiple columns using the same backend but different settings, you may add a +; colon and a suffix (e.g. "Solr:filter1", "Solr:filter2") to differentiate the +; sections. + +; Each section contains some or all of the following settings: ; ; label = The header on the column ; sublabel = Text to display below the header (optional) @@ -22,6 +26,8 @@ ; specified recommendations will be loaded into the top ; area of the column. If set to false, recommendations ; will be suppressed (default). +; filter = One or more filters to apply to search results displayed in the column. +; Use multiple "filter[] = ..." lines if multiple filters are needed. ; ; All display text is subject to translation and may be added to the language ; .ini files. diff --git a/module/VuFind/src/VuFind/Controller/CombinedController.php b/module/VuFind/src/VuFind/Controller/CombinedController.php index 98ff2988a3c3d6270072a13c213d751393afd162..7ff244878f783c10dba578013d7c39144449fa52 100644 --- a/module/VuFind/src/VuFind/Controller/CombinedController.php +++ b/module/VuFind/src/VuFind/Controller/CombinedController.php @@ -70,13 +70,14 @@ class CombinedController extends AbstractSearch $this->getSearchMemory()->disable(); // Validate configuration: - $searchClassId = $this->params()->fromQuery('id'); + $sectionId = $this->params()->fromQuery('id'); $config = $this->getServiceLocator()->get('VuFind\Config')->get('combined') ->toArray(); $tabConfig = $this->getTabConfig($config); - if (!isset($tabConfig[$searchClassId])) { + if (!isset($tabConfig[$sectionId])) { throw new \Exception('Illegal ID'); } + list($searchClassId) = explode(':', $sectionId); // Retrieve results: $options = $this->getServiceLocator() @@ -84,7 +85,7 @@ class CombinedController extends AbstractSearch $currentOptions = $options->get($searchClassId); list($controller, $action) = explode('-', $currentOptions->getSearchAction()); - $settings = $tabConfig[$searchClassId]; + $settings = $tabConfig[$sectionId]; $this->adjustQueryForSettings($settings); $settings['view'] = $this->forwardTo($controller, $action); @@ -151,8 +152,9 @@ class CombinedController extends AbstractSearch $supportsCart = false; $supportsCartOptions = []; foreach ($this->getTabConfig($config) as $current => $settings) { + list($searchClassId) = explode(':', $current); $this->adjustQueryForSettings($settings); - $currentOptions = $options->get($current); + $currentOptions = $options->get($searchClassId); $supportsCartOptions[] = $currentOptions->supportsCart(); if ($currentOptions->supportsCart()) { $supportsCart = true; @@ -160,13 +162,19 @@ class CombinedController extends AbstractSearch list($controller, $action) = explode('-', $currentOptions->getSearchAction()); $combinedResults[$current] = $settings; + + // Calculate a unique DOM id for this section of the search results; + // $searchClassId may contain colons, which must be converted. + $combinedResults[$current]['domId'] + = 'combined_' . str_replace(':', '____', $searchClassId); + $combinedResults[$current]['view'] = (!isset($settings['ajax']) || !$settings['ajax']) ? $this->forwardTo($controller, $action) : $this->createViewModel(['results' => $results]); // Special case: include appropriate "powered by" message: - if (strtolower($current) == 'summon') { + if (strtolower($searchClassId) == 'summon') { $this->layout()->poweredBy = 'Powered by Summonâ„¢ from Serials ' . 'Solutions, a division of ProQuest.'; } @@ -253,6 +261,10 @@ class CombinedController extends AbstractSearch $query = $this->getRequest()->getQuery(); $query->limit = isset($settings['limit']) ? $settings['limit'] : null; + // Apply filters, if any: + $query->filter = isset($settings['filter']) + ? (array)$settings['filter'] : null; + // Reset override to avoid bleed-over from one section to the next! $query->recommendOverride = false; diff --git a/themes/bootstrap3/templates/combined/results-ajax.phtml b/themes/bootstrap3/templates/combined/results-ajax.phtml index d2c9eb3217da7836fe4dfe8bf47a747e9180afa8..6089b75de4241a9da4b9fe76aee35904c70029e6 100644 --- a/themes/bootstrap3/templates/combined/results-ajax.phtml +++ b/themes/bootstrap3/templates/combined/results-ajax.phtml @@ -9,14 +9,14 @@ // Set up Javascript for use below: $searchClassIdEncoded = urlencode($searchClassId); - $searchClassIdHtmlEscaped = $this->escapeHtml($searchClassId); + $targetIdHtmlEscaped = $this->escapeHtml('#' . $currentSearch['domId']); $lookforEncoded = urlencode($lookfor); $loadJs = <<<JS $(document).ready(function(){ var url = path + '/Combined/Result?id=$searchClassIdEncoded&lookfor=$lookforEncoded'; - $('#combined_$searchClassIdHtmlEscaped').load(url, '', function(responseText) { + $('$targetIdHtmlEscaped').load(url, '', function(responseText) { if (responseText.length == 0) { - $('#combined_$searchClassIdHtmlEscaped').hide(); + $('$targetIdHtmlEscaped').hide(); } $('a.openUrlEmbed').click(function() { embedOpenUrlLinks($(this)); diff --git a/themes/bootstrap3/templates/combined/stack-distributed.phtml b/themes/bootstrap3/templates/combined/stack-distributed.phtml index d077ad4d748516adacfee6c52ee6e1e8f9725e90..bfd7281646e5c6338efdb687cabee2c5eb7c15cb 100644 --- a/themes/bootstrap3/templates/combined/stack-distributed.phtml +++ b/themes/bootstrap3/templates/combined/stack-distributed.phtml @@ -21,7 +21,7 @@ // Enable bulk options if appropriate: $viewParams['showBulkOptions'] = $this->supportsCartOptions[$columnIndex] && $this->showBulkOptions; ?> - <div id="combined_<?=$this->escapeHtmlAttr($searchClassId)?>"> + <div id="<?=$this->escapeHtmlAttr($currentSearch['domId'])?>"> <? $templateSuffix = (isset($currentSearch['ajax']) && $currentSearch['ajax']) ? 'ajax' : 'list'; ?> <?=$this->render('combined/results-' . $templateSuffix . '.phtml', $viewParams)?> </div> diff --git a/themes/bootstrap3/templates/combined/stack-left.phtml b/themes/bootstrap3/templates/combined/stack-left.phtml index c6228f0fe375165cdf6776a1856e036214417b56..b75aa6bc471dcf2fe3c98422e042da496a227a31 100644 --- a/themes/bootstrap3/templates/combined/stack-left.phtml +++ b/themes/bootstrap3/templates/combined/stack-left.phtml @@ -14,7 +14,7 @@ // Enable bulk options if appropriate: $viewParams['showBulkOptions'] = $this->supportsCartOptions[$columnIndex] && $this->showBulkOptions; ?> - <div id="combined_<?=$this->escapeHtmlAttr($searchClassId)?>"> + <div id="<?=$this->escapeHtmlAttr($currentSearch['domId'])?>"> <? $templateSuffix = (isset($currentSearch['ajax']) && $currentSearch['ajax']) ? 'ajax' : 'list'; ?> <?=$this->render('combined/results-' . $templateSuffix . '.phtml', $viewParams)?> </div> diff --git a/themes/bootstrap3/templates/combined/stack-right.phtml b/themes/bootstrap3/templates/combined/stack-right.phtml index 34340ab9bb35a4e5f5c65747f2ed670534b8e08b..bd6126dc369716963530e4cc2134e19800667da4 100644 --- a/themes/bootstrap3/templates/combined/stack-right.phtml +++ b/themes/bootstrap3/templates/combined/stack-right.phtml @@ -14,7 +14,7 @@ // Enable bulk options if appropriate: $viewParams['showBulkOptions'] = $this->supportsCartOptions[$columnIndex] && $this->showBulkOptions; ?> - <div id="combined_<?=$this->escapeHtmlAttr($searchClassId)?>"> + <div id="<?=$this->escapeHtmlAttr($currentSearch['domId'])?>"> <? $templateSuffix = (isset($currentSearch['ajax']) && $currentSearch['ajax']) ? 'ajax' : 'list'; ?> <?=$this->render('combined/results-' . $templateSuffix . '.phtml', $viewParams)?> </div> diff --git a/themes/jquerymobile/templates/combined/results-ajax.phtml b/themes/jquerymobile/templates/combined/results-ajax.phtml index eaf0b46e08404e4ad4c316757148aa94c7df3400..840e63c31c63a72e1a47d04d82d64c14d621b004 100644 --- a/themes/jquerymobile/templates/combined/results-ajax.phtml +++ b/themes/jquerymobile/templates/combined/results-ajax.phtml @@ -7,7 +7,7 @@ // Set up Javascript for use below: $loadJs = 'var url = path + "/Combined/Result?id=' . urlencode($searchClassId) . '&lookfor=' . urlencode($lookfor) . '";' - . "\$('#combined_" . $this->escapeHtml($searchClassId) . "').load(url, '', function(responseText) {" + . "\$('#" . $this->escapeHtml($currentSearch['domId']) . "').load(url, '', function(responseText) {" . "if (responseText.length == 0) $('#combined_" . $this->escapeHtml($searchClassId) . "').hide();" . "$('.combinedButton').button(); });"; ?> diff --git a/themes/jquerymobile/templates/combined/results.phtml b/themes/jquerymobile/templates/combined/results.phtml index 3bd8b746e2690f17e9e2c60debf18b21f2d3e627..2620b26674210a15fa875e55336ff1ea5cc5aacd 100644 --- a/themes/jquerymobile/templates/combined/results.phtml +++ b/themes/jquerymobile/templates/combined/results.phtml @@ -13,7 +13,7 @@ <?=$this->flashmessages()?> <? foreach ($this->combinedResults as $searchClassId => $currentSearch): ?> <? if ((!isset($currentSearch['ajax']) || !$currentSearch['ajax']) && isset($currentSearch['hide_if_empty']) && $currentSearch['hide_if_empty'] && $currentSearch['view']->results->getResultTotal() == 0) { continue; } ?> - <div class="combinedResult" id="combined_<?=$this->escapeHtml($searchClassId)?>"> + <div class="combinedResult" id="<?=$this->escapeHtmlAttr($currentSearch['domId'])?>"> <? if (isset($currentSearch['ajax']) && $currentSearch['ajax']): ?> <?=$this->render('combined/results-ajax.phtml', array('searchClassId' => $searchClassId, 'currentSearch' => $currentSearch))?> <? else: ?>