From f3ed25045c5b6ff81bd0db9ff18b1fbf577f0632 Mon Sep 17 00:00:00 2001 From: Ahmed Awan Date: Tue, 4 Jun 2024 14:12:29 -0500 Subject: [PATCH] [24.1] Disable state filter for collections in `HistoryFilters` Adds a `disablesFilters` property to the `ValidFilter` class which allows for the functionality of disabling other filters given certain values for a given filter. Note: This only works for ranged, input-field and dropdown type filters in the `FilterMenu` Fixes https://github.com/galaxyproject/galaxy/issues/18264 --- client/src/components/Common/FilterMenu.vue | 24 ++++++++++++++++++- .../components/Common/FilterMenuDropdown.vue | 6 ++++- .../src/components/Common/FilterMenuInput.vue | 11 +++++++-- .../components/Common/FilterMenuRanged.vue | 7 ++++-- .../History/Content/model/StatesInfo.vue | 2 ++ .../History/CurrentHistory/HistoryPanel.vue | 1 + .../src/components/History/HistoryFilters.js | 2 ++ client/src/utils/filtering.ts | 6 +++++ 8 files changed, 53 insertions(+), 6 deletions(-) diff --git a/client/src/components/Common/FilterMenu.vue b/client/src/components/Common/FilterMenu.vue index 0f0596e2f422..bec385b5454a 100644 --- a/client/src/components/Common/FilterMenu.vue +++ b/client/src/components/Common/FilterMenu.vue @@ -4,7 +4,7 @@ import { faAngleDoubleUp, faQuestion, faSearch } from "@fortawesome/free-solid-s import { FontAwesomeIcon } from "@fortawesome/vue-fontawesome"; import { BButton, BModal, BPopover } from "bootstrap-vue"; import { kebabCase } from "lodash"; -import { computed, ref } from "vue"; +import { computed, ref, set, watch } from "vue"; import type Filtering from "@/utils/filtering"; import { type Alias, type ErrorType, getOperatorForAlias, type ValidFilter } from "@/utils/filtering"; @@ -91,6 +91,8 @@ const toggleMenuButton = computed(() => { // Boolean for showing the help modal for the whole filter menu (if provided) const showHelp = ref(false); +const isDisabled = ref>({}); + const formattedSearchError = computed(() => { if (props.searchError) { const { column, col, operation, op, value, val, err_msg, ValueError } = props.searchError; @@ -144,6 +146,8 @@ function getValidFilter(filter: string): ValidFilter { function onOption(filter: string, value: any) { filters.value[filter] = value; + setDisabled(filter, value); + // for the compact view, we want to immediately search if (props.view === "compact") { onSearch(); @@ -177,6 +181,21 @@ function onToggle() { emit("update:show-advanced", !props.showAdvanced); } +function setDisabled(filter: string, newVal: any) { + const disablesFilters = validFilters.value[filter]?.disablesFilters; + const type = validFilters.value[filter]?.type; + if (disablesFilters && type !== Boolean) { + for (const [disabledFilter, disablingValues] of Object.entries(disablesFilters)) { + if (newVal && (disablingValues === null || disablingValues.includes(newVal))) { + set(isDisabled.value, disabledFilter, true); + filters.value[disabledFilter] = undefined; + } else { + set(isDisabled.value, disabledFilter, false); + } + } + } +} + function updateFilterText(newFilterText: string) { emit("update:filter-text", newFilterText); } @@ -243,6 +262,7 @@ function updateFilterText(newFilterText: string) { :filters="filters" :error="formattedSearchError || undefined" :identifier="identifier" + :disabled="isDisabled[filter] || false" @change="onOption" @on-enter="onSearch" @on-esc="onToggle" /> @@ -269,6 +289,7 @@ function updateFilterText(newFilterText: string) { :filter="getValidFilter(filter)" :filters="filters" :identifier="identifier" + :disabled="isDisabled[filter] || false" @change="onOption" /> diff --git a/client/src/components/Common/FilterMenuDropdown.vue b/client/src/components/Common/FilterMenuDropdown.vue index cdcfd41c4c5b..b86b5ca5663e 100644 --- a/client/src/components/Common/FilterMenuDropdown.vue +++ b/client/src/components/Common/FilterMenuDropdown.vue @@ -31,6 +31,7 @@ interface Props { [k: string]: FilterValue; }; identifier: string; + disabled?: boolean; } const props = defineProps(); @@ -76,7 +77,9 @@ const helpToggle = ref(false); const modalTitle = `${capitalize(props.filter.placeholder)} Help`; function onHelp(_: string, value: string) { helpToggle.value = false; - localValue.value = value; + if (!props.disabled) { + localValue.value = value; + } } // Quota Source refs and operations @@ -140,6 +143,7 @@ function setValue(val: string | QuotaUsage | undefined) { menu-class="w-100" size="sm" boundary="window" + :disabled="props.disabled" :toggle-class="props.error ? 'text-danger' : ''"> (any) diff --git a/client/src/components/Common/FilterMenuInput.vue b/client/src/components/Common/FilterMenuInput.vue index 011e12a73ae0..71e6e496bf78 100644 --- a/client/src/components/Common/FilterMenuInput.vue +++ b/client/src/components/Common/FilterMenuInput.vue @@ -28,6 +28,7 @@ interface Props { filters: { [k: string]: FilterType; }; + disabled?: boolean; } const props = defineProps(); @@ -47,14 +48,18 @@ const modalTitle = `${capitalize(props.filter.placeholder)} Help`; function onHelp(_: string, value: string) { helpToggle.value = false; - localValue.value = value; + + if (!props.disabled) { + localValue.value = value; + } } watch( () => localValue.value, (newFilter) => { emit("change", props.name, newFilter); - } + }, + { immediate: true } ); watch( @@ -79,6 +84,7 @@ watch( size="sm" :state="props.error ? false : null" :placeholder="`any ${props.filter.placeholder}`" + :disabled="props.disabled" :list="props.filter.datalist ? `${identifier}-${props.name}-selectList` : null" @keyup.enter="emit('on-enter')" @keyup.esc="emit('on-esc')" /> @@ -99,6 +105,7 @@ watch( v-model="localValue" reset-button button-only + :disabled="props.disabled" size="sm" /> diff --git a/client/src/components/Common/FilterMenuRanged.vue b/client/src/components/Common/FilterMenuRanged.vue index 91e362bad359..72f013212dc5 100644 --- a/client/src/components/Common/FilterMenuRanged.vue +++ b/client/src/components/Common/FilterMenuRanged.vue @@ -14,6 +14,7 @@ interface Props { filters: { [k: string]: FilterType; }; + disabled?: boolean; } const props = defineProps(); @@ -93,11 +94,12 @@ watch( size="sm" :state="hasError(localNameGt) ? false : null" :placeholder="localPlaceholder('gt')" + :disabled="props.disabled" @keyup.enter="emit('on-enter')" @keyup.esc="emit('on-esc')" /> - + @@ -109,11 +111,12 @@ watch( size="sm" :state="hasError(localNameLt) ? false : null" :placeholder="localPlaceholder('lt')" + :disabled="props.disabled" @keyup.enter="emit('on-enter')" @keyup.esc="emit('on-esc')" /> - + diff --git a/client/src/components/History/Content/model/StatesInfo.vue b/client/src/components/History/Content/model/StatesInfo.vue index 2cd7ab525f1a..14d2267e5c7a 100644 --- a/client/src/components/History/Content/model/StatesInfo.vue +++ b/client/src/components/History/Content/model/StatesInfo.vue @@ -39,6 +39,8 @@ function onFilter(value: string) { (Note that the colors for each state correspond to content item state colors in the history, and if it exists, hovering over the icon on a history item will display the state message.) +
+ You cannot filter a history for collections given a state.

diff --git a/client/src/components/History/CurrentHistory/HistoryPanel.vue b/client/src/components/History/CurrentHistory/HistoryPanel.vue index f1e94fd4ea68..ca6738f303dd 100644 --- a/client/src/components/History/CurrentHistory/HistoryPanel.vue +++ b/client/src/components/History/CurrentHistory/HistoryPanel.vue @@ -509,6 +509,7 @@ function setItemDragstart( = { helpInfo?: DefineComponent | string; /** A default value (will make this a default filter for an empty `filterText`) */ default?: T; + /** A dict of filters and corresponding values for this filter that disable them. + * Note: if value is null, the filter is disabled for any value of this filter. + */ + disablesFilters?: { + [filter: string]: T[] | null; + }; }; /** Converts user input to backend compatible date