Skip to content

Commit

Permalink
feat(dashboards): add bulk adding via comma seperated values to `…
Browse files Browse the repository at this point in the history
…manage validators`

Old implementation of `<BcSearchbarMain>` was `buggy`
and was removed.

See: BEDS-547
  • Loading branch information
marcel-bitfly committed Nov 20, 2024
1 parent 4d6992d commit b6b40af
Show file tree
Hide file tree
Showing 13 changed files with 800 additions and 455 deletions.
8 changes: 7 additions & 1 deletion frontend/assets/css/main.scss
Original file line number Diff line number Diff line change
Expand Up @@ -3,14 +3,20 @@
@import "~/assets/css/fonts.scss";
@import "~/assets/css/utils.scss";

html {
transition-behavior: allow-discrete;
interpolate-size: allow-keywords;
}

html,
h1,h2,h3,h4,h5,h6,
input,
ul,
li
{
margin: 0;
padding: 0;
font: inherit;
font: var(--standard_text_font_family);
}
ul {
padding-inline-start: 1.5rem;
Expand Down
1 change: 0 additions & 1 deletion frontend/assets/css/prime.scss
Original file line number Diff line number Diff line change
Expand Up @@ -503,7 +503,6 @@ div.p-select {
height: 30px;

.p-select-label {
font-size: var(--tiny_text_font_size);
color: var(--input-active-text-color);
}

Expand Down
322 changes: 322 additions & 0 deletions frontend/components/bc/BcInputSearch.vue
Original file line number Diff line number Diff line change
@@ -0,0 +1,322 @@
<script setup lang="ts" generic="T">
const props = withDefaults(defineProps<{
hasError?: boolean,
hasFocus?: boolean,
isLoading: boolean,
results: T[] | undefined,
shouldClearOnSubmit?: boolean,
shouldSelectFirstResult?: boolean,
}>(), {
shouldClearOnSubmit: true,
})
const emit = defineEmits<{
(e: 'search', value: string): void,
(e: 'submit', value: T): void,
}>()
const { t: $t } = useTranslation()
const input = defineModel<string>()
const elementInput = ref<HTMLElement >()
const hasInput = computed(() => input.value?.length)
const idResults = useId()
const elementResults = ref<HTMLElement>()
const idElementList = useId()
const elementList = ref<Array<HTMLElement>>()
const resultItems = computed(() => {
if (!elementList.value) return []
const result = elementList.value
return result
})
const hasResults = computed(() => props.results?.length)
const hasPopover = ref(false)
const currentIndex = ref(-1)
const changeSelectedIndex = (step: number) => {
showPopover()
currentIndex.value = currentIndex.value + step
if (currentIndex.value >= resultItems.value.length) {
currentIndex.value = 0
return
}
if (currentIndex.value < 0) {
currentIndex.value = resultItems.value.length - 1
return
}
}
const {
bottom,
left,
right,
update: updatePositionOfInput,
} = useElementBounding(elementInput, {
immediate: false,
})
const showPopover = () => {
updatePositionOfInput()
hasPopover.value = true
if (!elementResults.value || elementResults.value.matches(':popover-open')) return
if (props.shouldSelectFirstResult) currentIndex.value = 0
elementResults.value.showPopover()
}
const hidePopover = () => {
hasPopover.value = false
currentIndex.value = -1
if (!elementResults.value || !elementResults.value.matches(':popover-open')) return
elementResults.value.hidePopover()
}
const isEmpty = computed(() => !props.results?.length)
watchDebounced(input, async () => {
emit('search', input.value ?? '')
})
watch(hasInput, () => {
if (!hasInput.value) {
hidePopover()
}
})
const hasErrorOrNoResults = computed(() => props.hasError || props.results === undefined)
watch([
() => props.results,
() => props.hasError,
], () => {
if (!hasInput.value) return
showPopover()
})
watchEffect(() => {
elementResults.value?.style.setProperty('--position-top', bottom.value + 'px')
elementResults.value?.style.setProperty('--position-left', left.value + 'px')
elementResults.value?.style.setProperty('--position-right', right.value + 'px')
})
const handleEsc = (event: KeyboardEvent) => {
if (!hasInput.value) return
event.stopPropagation()
if (!hasPopover.value) return
event.preventDefault()
hidePopover()
}
const handleFocus = (event: Event) => {
if (!(event.target as HTMLInputElement).value) return
if (hasResults.value) showPopover()
}
const selectedResult = computed(() => props.results?.[currentIndex.value])
const handleSubmit = () => {
if (!selectedResult.value) return
emit('submit', toRaw(selectedResult.value))
if (props.shouldClearOnSubmit) input.value = ''
hidePopover()
}
const handleClick = (index: number) => {
currentIndex.value = index
elementInput.value?.focus()
handleSubmit()
}
const isResultsHovered = () => elementResults.value?.matches(':hover')
const handleBlur = () => {
if (isResultsHovered()) {
return
}
hidePopover()
}
onMounted(() => {
if (props.hasFocus) elementInput.value?.focus()
})
const ariaActivedescendant = computed(() => {
if (props.hasError) return `${idResults}-error`
if (isEmpty.value) return `${idResults}-empty`
return `${idElementList}-${currentIndex.value}`
})
</script>

<template>
<form
role="search"
class="bc-input-search__form"
@keydown.arrow-up.prevent="hasResults && changeSelectedIndex(-1)"
@keydown.arrow-down.prevent="hasResults && changeSelectedIndex(+1)"
@keydown.arrow-down.alt.exact.prevent="hasResults && showPopover()"
@keydown.esc="handleEsc"
@submit.prevent="handleSubmit"
>
<input
ref="elementInput"
v-model.trim="input"
role="combobox"
:aria-expanded="hasPopover"
:aria-busy="isLoading"
aria-autocomplete="none"
:aria-controls="idResults"
:aria-label="$t('dashboard.validator.management.search.label')"
:aria-activedescendant
:placeholder="$t('dashboard.validator.management.search.placeholder')"
type="search"
class="bc-input-search__input"
:class="{ 'bc-input-search__input--has-popover': hasPopover }"
@blur="handleBlur"
@focus="handleFocus"
>
<span
class="bc-input-search__loading-indicator"
>
<BcLoadingSpinner
v-if="isLoading"
size="full"
loading
/>
</span>
<div
:id="idResults"
ref="elementResults"
popover="manual"
class="bc-input-search__results"
:class="{ 'bc-input-search__results--has-popover': hasPopover }"
>
<div
class="bc-input-search__results_content"
>
<div
v-if="hasErrorOrNoResults"
:id="`${idResults}-error`"
class="bc-input-search__list-content-error"
>
<slot name="error">
{{ $t('common.search_error') }}
</slot>
</div>
<div
v-else-if="isEmpty"
:id="`${idResults}-empty`"
class="bc-input-search__list-content-empty"
>
<slot name="empty">
{{ $t('common.search_empty') }}
</slot>
</div>
<slot
v-else
name="results"
:results
role="listbox"
>
<ul
role="listbox"
aria-label="Add the following validators to your dashbard"
class="bc-input-search__list"
>
<BcInputSearchItem
v-for="(item, index) in results"
:id="`${idElementList}-${index}`"
:key="`${item}-${index}`"
ref="elementList"
:is-aria-selected="currentIndex === index"
tabindex="-1"
@click="handleClick(index)"
>
<slot name="result" :item />
</BcInputSearchItem>
</ul>
</slot>
</div>
</div>
</form>
</template>

<style scoped lang="scss">
.bc-input-search__form {
--padding: .375rem;
position: relative;
}
.bc-input-search__input {
width: 100%;
flex: 1;
padding: var(--padding);
padding-right: 1.625rem;
border: none;
border: .0625rem solid var(--input-border-color);
border-radius: var(--corner-radius, .25rem);
background-color: var(--input-background);
color: var(--input-active-text-color);
transition: border-radius 200ms ease-in;
&::-webkit-search-cancel-button {
display: none;
}
&:focus-visible,
&:focus-within,
&--has-popover
{
border-color: var(--primary-color);
outline: none;
}
}
.bc-input-search__input--has-popover {
--missing-border-bottom: .0625rem;
border-bottom: unset;
padding-bottom: calc(var(--padding) + var(--missing-border-bottom) );
border-bottom-left-radius: unset;
border-bottom-right-radius: unset;
}
.bc-input-search__loading-indicator{
position: absolute;
inset: var(--padding);
left: unset;
aspect-ratio: 1;
}
.bc-input-search__results {
--position-top: 0;
--position-left: 0;
--position-right: 0;
border: .0625rem solid var(--primary-color);
border-top: none;
border-bottom-left-radius: var(--corner-radius, .25rem);
border-bottom-right-radius: var(--corner-radius, .25rem);
background-color: var(--input-background);
inset: unset;
top: calc(var(--position-top) - .0625rem);
left: var(--position-left);
width: calc(var(--position-right) - var(--position-left));
color: var(--input-active-text-color);
padding: 0;
overflow: clip;
height: 0;
transition: height 200ms ease-in, display 200ms;
transition-delay: 200ms;
height: auto;
&:popover-open{
@starting-style {
height: 0;
}
}
}
.bc-input-search__results_content {
border-top: .0625rem solid var(--input-border-color);
}
.bc-input-search__list {
overflow: auto;
max-height: 13.75rem;
list-style: none;
padding-inline-start: 0;
display: grid;
grid-auto-flow: row;
grid-template-columns: min-content max-content 2fr max-content;
column-gap: .625rem;
}
.bc-input-search__list-content-error,
.bc-input-search__list-content-empty {
padding: var(--padding);
text-align: center;
color: var(--text-color-disabled);
}
</style>
32 changes: 32 additions & 0 deletions frontend/components/bc/BcInputSearchItem.vue
Original file line number Diff line number Diff line change
@@ -0,0 +1,32 @@
<script setup lang="ts">
defineProps<{
isAriaSelected: boolean,
}>()
</script>

<template>
<li
:aria-selected="isAriaSelected"
role="option"
class="bc-input-search__list-item"
tabindex="-1"
>
<slot />
</li>
</template>

<style scoped lang="scss">
.bc-input-search__list-item {
grid-column: 1 / -1;
display: grid;
align-items: center;
grid-template-columns: subgrid;
padding: var(--padding);
width: 100%;
cursor: pointer;
&:hover,
&[aria-selected="true"] {
background: var(--list-hover-background);
}
}
</style>
Loading

0 comments on commit b6b40af

Please sign in to comment.