mirror of
https://github.com/eugeny/tabby
synced 2025-12-11 18:25:57 +00:00
112 lines
4.2 KiB
TypeScript
112 lines
4.2 KiB
TypeScript
import { firstBy } from 'thenby'
|
|
import { Component, Input, HostListener, ViewChildren, QueryList, ElementRef } from '@angular/core' // eslint-disable-line @typescript-eslint/no-unused-vars
|
|
import { NgbActiveModal } from '@ng-bootstrap/ng-bootstrap'
|
|
import FuzzySearch from 'fuzzy-search'
|
|
import { SelectorOption } from '../api/selector'
|
|
|
|
/** @hidden */
|
|
@Component({
|
|
selector: 'selector-modal',
|
|
template: require('./selectorModal.component.pug'),
|
|
styles: [require('./selectorModal.component.scss')],
|
|
})
|
|
export class SelectorModalComponent<T> {
|
|
@Input() options: SelectorOption<T>[]
|
|
@Input() filteredOptions: SelectorOption<T>[]
|
|
@Input() filter = ''
|
|
@Input() name: string
|
|
@Input() selectedIndex = 0
|
|
hasGroups = false
|
|
@ViewChildren('item') itemChildren: QueryList<ElementRef>
|
|
|
|
constructor (
|
|
public modalInstance: NgbActiveModal,
|
|
) { }
|
|
|
|
ngOnInit (): void {
|
|
this.onFilterChange()
|
|
this.hasGroups = this.options.some(x => x.group)
|
|
}
|
|
|
|
@HostListener('keydown', ['$event']) onKeyUp (event: KeyboardEvent): void {
|
|
if (event.key === 'PageUp' || event.key === 'ArrowUp' && event.metaKey) {
|
|
this.selectedIndex -= 10
|
|
event.preventDefault()
|
|
} else if (event.key === 'PageDown' || event.key === 'ArrowDown' && event.metaKey) {
|
|
this.selectedIndex += 10
|
|
event.preventDefault()
|
|
} else if (event.key === 'ArrowUp') {
|
|
this.selectedIndex--
|
|
event.preventDefault()
|
|
} else if (event.key === 'ArrowDown') {
|
|
this.selectedIndex++
|
|
event.preventDefault()
|
|
} else if (event.key === 'Enter') {
|
|
this.selectOption(this.filteredOptions[this.selectedIndex])
|
|
} else if (event.key === 'Escape') {
|
|
this.close()
|
|
}
|
|
if (event.key === 'Backspace' && this.canEditSelected()) {
|
|
this.filter = this.filteredOptions[this.selectedIndex].freeInputEquivalent!
|
|
this.onFilterChange()
|
|
}
|
|
|
|
this.selectedIndex = (this.selectedIndex + this.filteredOptions.length) % this.filteredOptions.length
|
|
Array.from(this.itemChildren)[this.selectedIndex]?.nativeElement.scrollIntoView({
|
|
behavior: 'smooth',
|
|
block: 'nearest',
|
|
})
|
|
}
|
|
|
|
onFilterChange (): void {
|
|
const f = this.filter.trim().toLowerCase()
|
|
if (!f) {
|
|
this.filteredOptions = this.options.slice().sort(
|
|
firstBy<SelectorOption<T>, number>(x => x.weight ?? 0)
|
|
.thenBy<SelectorOption<T>, string>(x => x.group ?? '')
|
|
.thenBy<SelectorOption<T>, string>(x => x.name)
|
|
)
|
|
.filter(x => !x.freeInputPattern)
|
|
} else {
|
|
// eslint-disable-next-line @typescript-eslint/restrict-plus-operands
|
|
this.filteredOptions = new FuzzySearch(
|
|
this.options,
|
|
['name', 'group', 'description'],
|
|
{ sort: true },
|
|
).search(f)
|
|
|
|
const freeOption = this.options.find(x => x.freeInputPattern)
|
|
if (freeOption && !this.filteredOptions.includes(freeOption)) {
|
|
this.filteredOptions.push(freeOption)
|
|
}
|
|
}
|
|
this.selectedIndex = Math.max(0, this.selectedIndex)
|
|
this.selectedIndex = Math.min(this.filteredOptions.length - 1, this.selectedIndex)
|
|
}
|
|
|
|
filterMatches (option: SelectorOption<T>, terms: string[]): boolean {
|
|
const content = (option.group ?? '') + option.name + (option.description ?? '')
|
|
return terms.every(term => content.toLowerCase().includes(term))
|
|
}
|
|
|
|
getOptionText (option: SelectorOption<T>): string {
|
|
if (option.freeInputPattern) {
|
|
return option.freeInputPattern.replace('%s', this.filter)
|
|
}
|
|
return option.name
|
|
}
|
|
|
|
selectOption (option: SelectorOption<T>): void {
|
|
option.callback?.(this.filter)
|
|
this.modalInstance.close(option.result)
|
|
}
|
|
|
|
canEditSelected (): boolean {
|
|
return !this.filter && !!this.filteredOptions[this.selectedIndex].freeInputEquivalent && this.options.some(x => x.freeInputPattern)
|
|
}
|
|
|
|
close (): void {
|
|
this.modalInstance.dismiss()
|
|
}
|
|
}
|