client/src/app/shared/shared-main/buttons/action-dropdown.component.ts
import { ChangeDetectionStrategy, Component, Input } from '@angular/core'
import { Params, RouterLink } from '@angular/router'
import { GlobalIconName } from '@app/shared/shared-icons/global-icon.component'
import { GlobalIconComponent } from '../../shared-icons/global-icon.component'
import { NgbDropdown, NgbDropdownToggle, NgbDropdownMenu, NgbTooltip } from '@ng-bootstrap/ng-bootstrap'
import { NgIf, NgClass, NgFor, NgTemplateOutlet } from '@angular/common'
export type DropdownAction<T> = {
label?: string
iconName?: GlobalIconName
description?: string
title?: string
handler?: (a: T) => any
linkBuilder?: (a: T) => (string | number)[]
queryParamsBuilder?: (a: T) => Params
isDisplayed?: (a: T) => boolean
class?: string[]
isHeader?: boolean
ownerOrModeratorPrivilege?: () => string
}
export type DropdownButtonSize = 'normal' | 'small'
export type DropdownTheme = 'orange' | 'grey'
export type DropdownDirection = 'horizontal' | 'vertical'
@Component({
selector: 'my-action-dropdown',
styleUrls: [ './action-dropdown.component.scss' ],
templateUrl: './action-dropdown.component.html',
changeDetection: ChangeDetectionStrategy.OnPush,
standalone: true,
imports: [
NgIf,
NgbTooltip,
NgbDropdown,
NgbDropdownToggle,
NgClass,
GlobalIconComponent,
NgbDropdownMenu,
NgFor,
RouterLink,
NgTemplateOutlet
]
})
export class ActionDropdownComponent<T> {
@Input() actions: DropdownAction<T>[] | DropdownAction<T>[][] = []
@Input() entry: T
@Input() placement = 'bottom-left auto'
@Input() container: null | 'body'
@Input() buttonSize: DropdownButtonSize = 'normal'
@Input() buttonDirection: DropdownDirection = 'horizontal'
@Input() buttonStyled = true
@Input() label: string
@Input() theme: DropdownTheme = 'grey'
getActions (): DropdownAction<T>[][] {
if (this.actions.length !== 0 && Array.isArray(this.actions[0])) return this.actions as DropdownAction<T>[][]
return [ this.actions as DropdownAction<T>[] ]
}
getQueryParams (action: DropdownAction<T>, entry: T) {
if (action.queryParamsBuilder) return action.queryParamsBuilder(entry)
return {}
}
areActionsDisplayed (actions: (DropdownAction<T> | DropdownAction<T>[])[], entry: T): boolean {
return actions.some(a => {
if (Array.isArray(a)) return this.areActionsDisplayed(a, entry)
return a.isHeader !== true && (a.isDisplayed === undefined || a.isDisplayed(entry))
})
}
isBlockDisplayed (allActions: (DropdownAction<T> | DropdownAction<T>[])[], action: DropdownAction<T>, entry: T) {
// Do not display only the header
if (action.isHeader && !this.areActionsDisplayed(allActions, entry)) return false
return action.isDisplayed === undefined || action.isDisplayed(entry) === true
}
}