File

src/components/sq-select-multi-tags/sq-select-multi-tags.component.ts

Description

Represents a multi-tag select component.

Example :
<sq-select-multi-tags
  [name]="'tags'"
  [value]="selectedTags"
  [options]="tagOptions"
  (valueChange)="handleTagSelection($event)"
>
</sq-select-multi-tags>

Implements

OnChanges

Metadata

Index

Properties
Methods
Inputs
Outputs

Constructor

constructor(element: ElementRef, translate: TranslateService, changeDetector: ChangeDetectorRef)

Constructs a new SqSelectMultiTagsComponent.

Parameters :
Name Type Optional Description
element ElementRef No
  • The element reference.
translate TranslateService No
  • The optional TranslateService for internationalization.
changeDetector ChangeDetectorRef No
  • Base class that provides change detection functionality.

Inputs

backgroundColor
Type : string
Default value : ''

Background color for the multi-tag select input.

borderColor
Type : string
Default value : ''

Border color for the multi-tag select input.

customClass
Type : string
Default value : ''

Custom CSS class for styling the component.

disabled
Type : boolean
Default value : false

Indicates whether the multi-tag select input is disabled.

errorSpan
Type : boolean
Default value : true

Indicates whether to display an error span.

externalError
Type : string
Default value : ''

External error message for the multi-tag select input.

externalIcon
Type : string
Default value : ''

External icon for the multi-tag select input.

hideSearch
Type : boolean
Default value : false

Indicates whether to hide the search input.

id
Type : string

The id attribute for the multi-tag select input.

label
Type : string
Default value : ''

The label for the multi-tag select input.

labelColor
Type : string
Default value : ''

Color for the label of the multi-tag select input.

loading
Type : boolean
Default value : false

Indicates whether the multi-tag select input is in a loading state.

maxHeight
Type : string
Default value : '100%'

Maximum height for the multi-tag values.

maxTags
Type : number

Maximum number of tags that can be chosen.

minCharactersToSearch
Type : number
Default value : 0

Minimum number of characters to perform the searchChange.

minTags
Type : number

Minimum number of tags that can be chosen.

name
Type : string
Default value : `random-name-${(1 + Date.now() + Math.random()).toString().replace('.', '')}`

The name attribute for the multi-tag select input.

options
Type : Array<OptionMulti>
Default value : []

Options available for selection.

placeholder
Type : string
Default value : ''

Placeholder text for the input field.

placeholderSearch
Type : string
Default value : ''

Placeholder text for the search input field.

readonly
Type : boolean
Default value : false

Indicates whether the multi-tag select input is readonly.

required
Type : boolean
Default value : false

Indicates whether the multi-tag select input is required.

showInside
Type : boolean
Default value : true

Indicates whether to show selected tags inside the input.

timeToChange
Type : number
Default value : 800

The time interval for input timeout in ms.

tooltipColor
Type : string
Default value : 'inherit'

Tooltip color for the multi-tag select input.

tooltipIcon
Type : string
Default value : ''

Tooltip icon for the multi-tag select input.

tooltipMessage
Type : string
Default value : ''

Tooltip message for the multi-tag select input.

tooltipPlacement
Type : "center top" | "center bottom" | "left center" | "right center"
Default value : 'right center'

Tooltip placement for the multi-tag select input.

useFormErrors
Type : boolean
Default value : true

Indicates whether to use form errors for validation.

value
Type : OptionMulti[]
Default value : []

The selected values for the multi-tag select input.

Outputs

closeChange
Type : EventEmitter<boolean>

Event emitted when the multi-tag select dropdown is closed.

removeTag
Type : EventEmitter<OptionMulti>

Event emitted when a tag is removed.

searchChange
Type : EventEmitter<string>

Event emitted when the search input value changes.

valid
Type : EventEmitter<boolean>

Event emitted when the multi-tag select input becomes valid or invalid.

valueChange
Type : EventEmitter<Array<OptionMulti>>

Event emitted when the selected values change.

Methods

closeDropdown
closeDropdown()

Closes the multi-tag select dropdown.

Returns : void
Async doDropDownAction
doDropDownAction()

Do action to open or close thw dropdown list

Returns : any
emit
emit(object: OptionMulti, checked: boolean)

Emits a change event with the specified object and checked state.

Parameters :
Name Type Optional Description
object OptionMulti No
  • The object to emit.
checked boolean No
  • The checked state.
Returns : void
handleCollapse
handleCollapse(item: OptionMulti)

Handles the collapse of an item and set the cdkItemSize if the item is open.

Parameters :
Name Type Optional Description
item OptionMulti No
  • The item to collapse.
Returns : void
Async modelChange
modelChange(event: any)

Change searchtext with timeout and detect detectChanges

Parameters :
Name Type Optional
event any No
Returns : any
Async ngOnChanges
ngOnChanges(changes: SimpleChanges)

Lifecycle hook called when any input properties change.

Parameters :
Name Type Optional Description
changes SimpleChanges No
  • The changes detected in the component's input properties.
Returns : any
removeItem
removeItem(item: OptionMulti, event: any)

Removes an item from the selected values.

Parameters :
Name Type Optional Description
item OptionMulti No
  • The item to remove.
event any No
Returns : void
Async setError
setError(key: string, interpolateParams: Object)

Sets an error message.

Parameters :
Name Type Optional Default value Description
key string No
  • The translation key for the error message.
interpolateParams Object No {}
Returns : any
validate
validate()

Validates the multi-tag select input and sets the error state.

Returns : void
verifyNewOptions
verifyNewOptions()

Verify new options and set the cdkVirtualScrollViewportHeight

Returns : void

Properties

_options
Type : Array<OptionMulti>
Default value : []

Control options to render

cdkItemSize
Type : string | null
Default value : '32'

The size for the cdk-virtual-scroll-viewport (default 32px).

cdkVirtualScrollViewportHeight
Type : string
Default value : '305px'

The height for the cdk-virtual-scroll-viewport (default 305px).

Public element
Type : ElementRef
- The element reference.
error
Type : boolean | string
Default value : ''

Error message associated with the multi-tag select input.

findItemInValue
Default value : useMemo((item: OptionMulti, value?: Array<OptionMulti>) => { return !!value?.find((value) => value.value === item.value) })

Determines if an item exists in the selected values.

Parameters :
Name Description
item
  • The item to search for.
isMaxTags
Default value : false

Control the readonly on reach the maxTags

labelTemplate
Type : TemplateRef<HTMLElement> | null
Default value : null
Decorators :
@ContentChild('labelTemplate')

The label template for the search-based select input.

nativeElement
Type : ElementRef

Native element reference.

open
Default value : false

Indicates whether the multi-tag select dropdown is open.

renderOptionsList
Default value : false

Indicates when is the time to render the multi-tag select dropdown.

searchText
Type : string
Default value : ''

Text entered in the search input field.

selectEmptyTemplate
Type : TemplateRef<HTMLElement> | null
Default value : null
Decorators :
@ContentChild('selectEmptyTemplate')

The select empty template for the search-based select input.

timeouted
Default value : false

Indicates whether a timeout has occurred for input changes.

timeoutInput
Type : ReturnType<>

Timeout for input changes.

trackByOptValue
Type : TrackByFunction<any>
Default value : useMemo((index, opt) => opt.value)

Return trackBy for ngFor

valueChanged
Default value : false

Indicates whether the value has changed.

verifyIfHasChildrenInValue
Default value : useMemo((item: OptionMulti, value?: Array<OptionMulti>) => { if (item.children?.length) { const hasAllChildren = item.children.every((child) => this.findItemInValue(child, value)) if (hasAllChildren && !this.findItemInValue(item, value) && !this.timeouted) { this.timeouted = true setTimeout(() => { this.emit(item, true) this.timeouted = false }, 0) } return !!item.children.find((child) => this.findItemInValue(child, value)) } return false })

Verifies if an item has children that are selected.

Parameters :
Name Description
item
  • The item to check.
verifyIfOptionsHasChildren
Default value : useMemo((options: OptionMulti[]) => { return options.some((item) => item.children?.length) })

Verifies if any options have children.

Parameters :
Name Description
options
  • The options to check.
import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ContentChild, ElementRef, EventEmitter, Input, OnChanges, Optional, Output, SimpleChanges, TemplateRef, TrackByFunction } from '@angular/core'
import { TranslateService } from '@ngx-translate/core'
import { useMemo } from '../../helpers/memo.helper'
import { OptionMulti } from '../../interfaces/option.interface'

/**
 * Represents a multi-tag select component.
 *
 * @example
 * <sq-select-multi-tags
 *   [name]="'tags'"
 *   [value]="selectedTags"
 *   [options]="tagOptions"
 *   (valueChange)="handleTagSelection($event)"
 * >
 * </sq-select-multi-tags>
 * 
 * @implements {OnChanges}
 */
@Component({
  selector: 'sq-select-multi-tags',
  templateUrl: './sq-select-multi-tags.component.html',
  changeDetection: ChangeDetectionStrategy.OnPush,
  styleUrls: ['./sq-select-multi-tags.component.scss'],
  providers: [],
})
export class SqSelectMultiTagsComponent implements OnChanges {
  /**
   * The name attribute for the multi-tag select input.
   * 
   * @default 'random-name-[hash-random-code]'
   */
  @Input() name = `random-name-${(1 + Date.now() + Math.random()).toString().replace('.', '')}`

  /**
   * The selected values for the multi-tag select input.
   */
  @Input() value?: OptionMulti[] = []

  /**
   * The id attribute for the multi-tag select input.
   */
  @Input() id?: string

  /**
   * The label for the multi-tag select input.
   */
  @Input() label = ''

  /**
   * Custom CSS class for styling the component.
   */
  @Input() customClass = ''

  /**
   * Placeholder text for the input field.
   */
  @Input() placeholder = ''

  /**
   * External error message for the multi-tag select input.
   */
  @Input() externalError = ''

  /**
   * External icon for the multi-tag select input.
   */
  @Input() externalIcon = ''

  /**
   * Placeholder text for the search input field.
   */
  @Input() placeholderSearch = ''

  /**
   * Indicates whether the multi-tag select input is disabled.
   */
  @Input() disabled = false

  /**
   * Indicates whether the multi-tag select input is readonly.
   */
  @Input() readonly = false

  /**
   * Indicates whether the multi-tag select input is required.
   */
  @Input() required = false

  /**
   * Indicates whether the multi-tag select input is in a loading state.
   */
  @Input() loading = false

  /**
   * Indicates whether to use form errors for validation.
   */
  @Input() useFormErrors = true

  /**
   * Indicates whether to display an error span.
   */
  @Input() errorSpan = true

  /**
   * Background color for the multi-tag select input.
   */
  @Input() backgroundColor = ''

  /**
   * Border color for the multi-tag select input.
   */
  @Input() borderColor = ''

  /**
   * Color for the label of the multi-tag select input.
   */
  @Input() labelColor = ''

  /**
   * Maximum height for the multi-tag values.
   */
  @Input() maxHeight = '100%'

  /**
   * Minimum number of characters to perform the searchChange.
   */
  @Input() minCharactersToSearch = 0

  /**
   * The time interval for input timeout in ms.
   */
  @Input() timeToChange = 800

  /**
   * Options available for selection.
   */
  @Input() options: Array<OptionMulti> = []

  /**
   * Maximum number of tags that can be chosen.
   */
  @Input() maxTags?: number

  /**
   * Minimum number of tags that can be chosen.
   */
  @Input() minTags?: number

  /**
   * Indicates whether to show selected tags inside the input.
   */
  @Input() showInside = true

  /**
   * Indicates whether to hide the search input.
   */
  @Input() hideSearch = false

  /**
   * Tooltip message for the multi-tag select input.
   */
  @Input() tooltipMessage = ''

  /**
   * Tooltip placement for the multi-tag select input.
   */
  @Input() tooltipPlacement: 'center top' | 'center bottom' | 'left center' | 'right center' = 'right center'

  /**
   * Tooltip color for the multi-tag select input.
   */
  @Input() tooltipColor = 'inherit'

  /**
   * Tooltip icon for the multi-tag select input.
   */
  @Input() tooltipIcon = ''

  /**
   * Event emitted when the selected values change.
   */
  @Output() valueChange: EventEmitter<Array<OptionMulti>> = new EventEmitter()

  /**
   * Event emitted when the search input value changes.
   */
  @Output() searchChange: EventEmitter<string> = new EventEmitter()

  /**
   * Event emitted when the multi-tag select dropdown is closed.
   */
  @Output() closeChange: EventEmitter<boolean> = new EventEmitter()

  /**
   * Event emitted when a tag is removed.
   */
  @Output() removeTag: EventEmitter<OptionMulti> = new EventEmitter()

  /**
   * Event emitted when the multi-tag select input becomes valid or invalid.
   */
  @Output() valid: EventEmitter<boolean> = new EventEmitter()

  /**
   * The label template for the search-based select input.
   */
  @ContentChild('labelTemplate')
  labelTemplate: TemplateRef<HTMLElement> | null = null

  /**
   * The select empty template for the search-based select input.
   */
  @ContentChild('selectEmptyTemplate')
  selectEmptyTemplate: TemplateRef<HTMLElement> | null = null


  /**
   * Indicates when is the time to render the multi-tag select dropdown.
   */
  renderOptionsList = false

  /**
   * Indicates whether the multi-tag select dropdown is open.
   */
  open = false

  /**
   * Text entered in the search input field.
   */
  searchText = ''

  /**
   * Indicates whether the value has changed.
   */
  valueChanged = false

  /**
   * Indicates whether a timeout has occurred for input changes.
   */
  timeouted = false

  /**
   * Error message associated with the multi-tag select input.
   */
  error: boolean | string = ''

  /**
   * Native element reference.
   */
  nativeElement: ElementRef

  /**
   * Control options to render
   */
  _options: Array<OptionMulti> = []

  /**
   * Control the readonly on reach the maxTags
   */
  isMaxTags = false

  /**
   * Timeout for input changes.
   */
  timeoutInput!: ReturnType<typeof setTimeout>

  /**
   * The height for the cdk-virtual-scroll-viewport (default 305px).
   */
  cdkVirtualScrollViewportHeight = '305px'

  /**
   * The size for the cdk-virtual-scroll-viewport (default 32px).
   */
  cdkItemSize: string | null = '32'

  /**
   * Constructs a new SqSelectMultiTagsComponent.
   *
   * @param {ElementRef} element - The element reference.
   * @param {TranslateService} translate - The optional TranslateService for internationalization.
   * @param {ChangeDetectorRef} changeDetector - Base class that provides change detection functionality.
   */
  constructor(public element: ElementRef, @Optional() private translate: TranslateService, private changeDetector: ChangeDetectorRef) {
    this.nativeElement = element.nativeElement
  }

  /**
   * Lifecycle hook called when any input properties change.
   *
   * @param changes - The changes detected in the component's input properties.
   */
  async ngOnChanges(changes: SimpleChanges) {
    if (this.open && changes.hasOwnProperty('options')) {
      this.verifyNewOptions()
    }
    if (changes.hasOwnProperty('value') || changes.hasOwnProperty('minTags') || changes.hasOwnProperty('maxTags')) {
      this.validate()
    }
  }

  /**
   * Determines if an item exists in the selected values.
   *
   * @param {OptionMulti} item - The item to search for.
   * @returns {boolean} True if the item exists in the selected values; otherwise, false.
   */
  findItemInValue = useMemo((item: OptionMulti, value?: Array<OptionMulti>) => {
    return !!value?.find((value) => value.value === item.value)
  })

  /**
   * Verifies if any options have children.
   *
   * @param {OptionMulti[]} options - The options to check.
   * @returns {boolean} True if any option has children; otherwise, false.
   */
  verifyIfOptionsHasChildren = useMemo((options: OptionMulti[]) => {
    return options.some((item) => item.children?.length)
  })

  /**
   * Verifies if an item has children that are selected.
   *
   * @param {OptionMulti} item - The item to check.
   * @returns {boolean} True if the item has selected children; otherwise, false.
   */
  verifyIfHasChildrenInValue = useMemo((item: OptionMulti, value?: Array<OptionMulti>) => {
    if (item.children?.length) {
      const hasAllChildren = item.children.every((child) => this.findItemInValue(child, value))
      if (hasAllChildren && !this.findItemInValue(item, value) && !this.timeouted) {
        this.timeouted = true
        setTimeout(() => {
          this.emit(item, true)
          this.timeouted = false
        }, 0)
      }
      return !!item.children.find((child) => this.findItemInValue(child, value))
    }
    return false
  })

  /**
   * Removes an item from the selected values.
   *
   * @param {OptionMulti} item - The item to remove.
   */
  removeItem(item: OptionMulti, event: any) {
    event?.stopPropagation()
    if (!this.readonly && !this.disabled) {
      if (item.children?.length) {
        item.children.forEach((child) => {
          this.value = this.value?.filter((value) => value.value !== child.value)
        })
      }
      this.value = this.value?.filter((value) => value.value !== item.value)

      this.valueChange.emit(this.value)
      this.removeTag.emit(item)
      this.validate()
    }
  }

  /**
   * Emits a change event with the specified object and checked state.
   *
   * @param {OptionMulti} object - The object to emit.
   * @param {boolean} checked - The checked state.
   */
  emit(object: OptionMulti, checked: boolean) {
    if (checked) {
      this.value?.push(object)
      // This code adds all children of a parent to value. Commented out for now as it is not the desired behavior.
      // if (object.children?.length) {
      //   object.children.forEach((item) => {
      //     if (!this.value.find((child) => child.value === item.value)) {
      //       this.value.push(item)
      //     }
      //   })
      // }
    } else {
      this.value = this.value?.filter((item) => item.value !== object.value)
      if (object.children?.length) {
        object.children.forEach((item) => {
          this.value = this.value?.filter((child) => child.value !== item.value)
        })
      }
    }
    this.valueChanged = true
    this.valueChange.emit(this.value)
    this.validate()
  }

  /**
   * Do action to open or close thw dropdown list
   */
  async doDropDownAction() {
    if (this.open) {
      this.closeDropdown()
      this.renderOptionsList = await new Promise<boolean>(resolve => setTimeout(() => {
        resolve(false)
      }, 300))
      this.changeDetector.detectChanges()
    } else {
      this.verifyNewOptions()
      this.renderOptionsList = true
      this.open = await new Promise<boolean>(resolve => setTimeout(() => {
        resolve(true)
      }, 100))
      this.changeDetector.detectChanges()
    }
  }

  /**
   * Closes the multi-tag select dropdown.
   */
  closeDropdown() {
    this.open = false
    this._options = []
    this.searchText = ''
    this.closeChange.emit(this.valueChanged)
    this.valueChanged = false
  }

  /**
   * Handles the collapse of an item and set the cdkItemSize if the item is open.
   *
   * @param {OptionMulti} item - The item to collapse.
   */
  handleCollapse(item: OptionMulti) {
    item.open = !item.open
    if (item.children) {
      if (this.options.find((option) => option.open) || item.open) {
        this.cdkItemSize = null
      } else {
        this.cdkItemSize = '32'
      }
    }
  }

  /**
   * Sets an error message.
   *
   * @param {string} key - The translation key for the error message.
   */
  async setError(key: string, interpolateParams: Object = {}) {
    if (this.useFormErrors && this.translate) {
      this.error = await this.translate.instant(key, interpolateParams)
    }
  }

  /**
   * Validates the multi-tag select input and sets the error state.
   */
  validate() {
    if (this.externalError) {
      this.error = false
    } else if (this.required && !this.value?.length) {
      this.setError('forms.required')
      this.valid.emit(false)
    } else if (this.minTags && this.value && this.value?.length < this.minTags) {
      this.setError('forms.minimumRequired', { minTags: this.minTags })
      this.valid.emit(false)
    } else if (this.maxTags && this.value && this.value?.length === this.maxTags) {
      this.renderOptionsList = false
      this.isMaxTags = true
      this.error = ''
      this.valid.emit(true)
    } else {
      this.isMaxTags = false
      this.error = ''
      this.valid.emit(true)
    }
  }

  /**
   * Return trackBy for ngFor
   */
  trackByOptValue: TrackByFunction<any> = useMemo((index, opt) => opt.value)

  /**
   * Change searchtext with timeout and detect detectChanges
   */
  async modelChange(event: any) {
    if (!this.minCharactersToSearch || !event.length || event.length >= this.minCharactersToSearch) {
      clearTimeout(this.timeoutInput)
      this.searchText = await new Promise<string>(resolve => this.timeoutInput = setTimeout(() => {
        resolve(event)
      }, this.timeToChange)) || ''
      this.searchChange.emit(event)
      this.changeDetector.detectChanges()
    }
  }

  /**
   * Verify new options and set the cdkVirtualScrollViewportHeight
   */
  verifyNewOptions() {
    this._options = this.options
    if (!this._options.length) {
      this.cdkVirtualScrollViewportHeight = '16px'
    } else if (this._options.length < 15) {
      this.cdkVirtualScrollViewportHeight = this._options.length * 32 + 'px'
    } else {
      this.cdkVirtualScrollViewportHeight = '305px'
    }
  }

}
<div class="wrapper-all-inside-input {{ customClass }}">
  <label
    class="display-flex align-items-center"
    *ngIf="label?.length || labelTemplate || tooltipMessage"
    [ngClass]="{
      readonly: readonly || isMaxTags
    }"
    [for]="id"
  >
    <div *ngIf="label && !labelTemplate" [ngStyle]="{ 'color': labelColor }" [innerHtml]="label | universalSafe"></div>
    <span *ngIf="labelTemplate">
      <ng-container *ngTemplateOutlet="labelTemplate"></ng-container>
    </span>
    <sq-tooltip
      *ngIf="tooltipMessage"
      class="ml-1"
      [message]="tooltipMessage"
      [placement]="tooltipPlacement"
      [color]="tooltipColor"
      [icon]="tooltipIcon"
    ></sq-tooltip>
  </label>
  <div
    class="wrapper-select-multi"
    [ngClass]="{
      error: (externalError && externalError !== '') || (error && error !== ''),
      disabled: disabled,
      readonly: readonly || isMaxTags,
      loading: loading
    }"
  >
    <div
      [class]="'input-fake col  border-' + borderColor"
      style="min-height: auto"
      [ngStyle]="{ 'border-color': borderColor }"
      [ngClass]="{
        'no-label': !(label && label.length > 0),
        'has-icon': error || externalError,
        disabled: disabled,
        readonly: readonly || isMaxTags
      }"
      [ngStyle]="{
        'background-color': backgroundColor,
        'border-color': borderColor
      }"
      [clickOutsideEnabled]="open"
      (clickOutside)="closeDropdown()"
    >
      <div
        class="input-fake-content"
        [ngClass]="{
          disabled: disabled,
          readonly: readonly
        }"
        [style.maxHeight]="maxHeight"
        (click)="doDropDownAction()"
      >
        <div class="loading-wrapper" *ngIf="loading">
          <sq-loader></sq-loader>
        </div>
        <span *ngIf="!value?.length || !showInside">{{ placeholder }}</span>
        <div class="input-fake-content-text" *ngIf="showInside && value?.length">
          <span class="tag" *ngFor="let opt of value; let i = index" (click)="removeItem(opt, $event)">
            {{ opt?.label }} <i *ngIf="!readonly" class="fas fa-times" [style.minWidth]="'auto'"></i>
          </span>
        </div>
        <span *ngIf="value?.length" class="badge">{{ value!.length }}</span>
        <i *ngIf="!loading" class="icon-down fas fa-chevron-down"></i>
      </div>
      <div
        *ngIf="!loading && !disabled && !readonly && !isMaxTags && renderOptionsList"
        id="sq-select-multi-tags-scroll"
        class="input-window scrollbar"
        [ngClass]="{
          open: !loading && !disabled && !isMaxTags && renderOptionsList && open
        }"
      >
        <div class="input-search">
          <div class="wrapper-all-inside-input">
            <div class="p-0 wrapper-input wrapper-input-squid text-ellipsisarea">
              <input
                [name]="name"
                [id]="id"
                [placeholder]="placeholderSearch || ('forms.search' | translateInternal | async) || ''"
                class="col input"
                [ngModel]="searchText"
                (ngModelChange)="modelChange($event)"
                *ngIf="!hideSearch"
              />
            </div>
            <span class="icon icon-external textarea-icon"><i class='fas fa-search'></i></span>
          </div>
        </div>
        <cdk-virtual-scroll-viewport [itemSize]="cdkItemSize" [ngStyle]="{ 'height': cdkVirtualScrollViewportHeight }" class="list scrollbar">
          <ng-container *cdkVirtualFor="let opt of _options | search:searchText; i as index; trackBy: trackByOptValue">
            <ng-template *ngTemplateOutlet="option; context: { opt: opt, i: index }"></ng-template>
          </ng-container>
        </cdk-virtual-scroll-viewport>
        <ng-container *ngIf="!_options?.length">
          <p class="mb-0 mt-3" *ngIf="!selectEmptyTemplate">
            {{ 'forms.searchSelectEmpty' | translateInternal | async }}
          </p>
          <span *ngIf="selectEmptyTemplate">
            <ng-container *ngTemplateOutlet="selectEmptyTemplate"></ng-container>
          </span>
        </ng-container>
      </div>
    </div>
  </div>
  <div
    class="box-validation box-invalid show"
    *ngIf="errorSpan"
    [ngClass]="{
      'visibility-hidden-force': ((!externalError || externalError === '') && (!error || error === '')) || disabled || readonly
    }"
  >
    <i [ngClass]="{
      'visibility-hidden-force': !error && !externalError
    }" class="fa-solid fa-triangle-exclamation"></i>
    {{ externalError ? externalError : '' }}
    {{ error && !externalError ? error : '' }}
  </div>
  <ng-template #option let-opt="opt" let-i="i">
    <li>
      <div class="label">
        <i
          class="icon-collapse fas fa-chevron-down"
          [ngClass]="{ 'fa-rotate-by': !opt.open }"
          [ngStyle]="{ color: !opt?.children?.length || opt?.disabled ? 'transparent' : '' }"
          (click)="handleCollapse(opt)"
          style="--fa-rotate-angle: -90deg"
          *ngIf="verifyIfOptionsHasChildren(options)"
        ></i>
        <sq-selector
          [style.minWidth]="'auto'"
          [id]="(id || name) + '-checkbox-' + opt?.value + '-' + i"
          [name]="name + '-checkbox'"
          [value]="opt?.value"
          [disabled]="opt?.disabled"
          [checked]="findItemInValue(opt, value)"
          [indeterminate]="verifyIfHasChildrenInValue(opt, value)"
          (valueChange)="emit(opt, $event.checked)"
        ></sq-selector>
        <span class="text m-0 display-inline-block" [ngClass]="{ 'cursor-pointer': opt?.children?.length }" (click)="handleCollapse(opt)">{{
          opt?.label
        }}</span>
      </div>
      <ul class="children" *ngIf="opt?.children?.length" [ngClass]="{open: !opt?.disabled && opt?.open}">
        <ng-container *ngFor="let child of opt?.children | searchValidValues:searchText; let j = index; trackBy: trackByOptValue">
          <ng-template *ngTemplateOutlet="option; context: { opt: child, i: j }"></ng-template>
        </ng-container>
      </ul>
    </li>
  </ng-template>
</div>

./sq-select-multi-tags.component.scss

.wrapper-select-multi {
  margin: 0;
  padding: 0;
  text-align: left;
  display: flex;
  flex-direction: column;
  position: relative;

  &.loading {
    cursor: wait;
  }

  .input-fake-content-text {
    display: flex;
    justify-content: flex-start;
    align-items: center;
    flex-wrap: wrap;
    gap: 5px;
    max-width: calc(100% - 40px);
    min-height: 51px;

    .tag {
      margin: 0;
      border: 1px solid var(--border_color);
      border-radius: 5px;
      padding: 0.2rem 0.35rem;
      display: flex;
      align-items: center;
      gap: 5px;
      cursor: pointer;
      transition: var(--transition);
      background: var(--background_secondary);

      &:hover {
        background: var(--border_color);
      }
    }
  }

  .input-fake-content {
    min-height: 44px;
    border: 1px solid var(--border_color);
    background: var(--background);
    padding: 0.75rem 1rem;
    transition: var(--transition);
    color: var(--text_color);
    border-radius: 5px;
    overflow-x: auto;

    .icon-down {
      right: 20px;
      bottom: calc(50% - 6px);
      position: absolute;
      font-size: 0.86rem;
    }
    &.readonly {
      cursor: not-allowed;
      pointer-events: none;
      background-color: var(--color_border_input);
      border-color: var(--color_border_input_disabled);
    }
  }

  .loading-wrapper {
    position: absolute;
    right: 5px;
    bottom: 10px;
  }

  .badge {
    position: absolute;
    bottom: 15px;
    right: 20px;
    margin: 0;
    border-radius: 50%;
    background-color: var(--border_color);
    color: var(--black);
    font-size: 0.86rem;
    font-weight: 700;
    padding: 0 0.35rem;
    border-radius: 50%;
    min-width: 20px;
    height: 20px;
    line-height: 20px;
    text-align: center;
    transition: var(--transition);
  }

  .input-search {
    display: inline-block;
    position: sticky;
    top: 0;
    z-index: 1;
    width: 100%;
    padding-top: 1.1rem;
    margin-bottom: 0.7rem;
    background-color: var(--background_secondary);

    .wrapper-all-inside-input {
      .icon {
        margin: 0;
        font-size: 1rem;
        font-weight: 700;
        position: absolute;
        right: 11px;
        top: 27px;
      }

      .icon-external {
        color: inherit !important;
      }
    }
  }

  .input-window {
    width: 100%;
    position: absolute;
    top: 100%;
    max-height: 0;
    height: 0;
    transition: var(--transition);
    overflow: hidden;
    z-index: 1;

    &.open {
      height: auto;
      max-height: 400px;
      background: var(--background_secondary);
      padding: 0 0.7rem 1.1rem;
      overflow-y: auto;
      box-shadow: var(--box_shadow);
    }
  }

  .icon-collapse {
    transition: var(--transition);
    cursor: pointer;
    padding: 0.35rem;
    position: relative;
  }

  .list,
  .children {
    padding: 0;
    list-style: none;
    margin: 0;
    width: 100%;

    li {
      display: grid;
      .label {
        margin: 0 0 0.5rem;
        display: flex;
        flex-wrap: nowrap;
        align-items: baseline;
        justify-content: flex-start;
        transition: var(--transition);
      }
    }
  }

  .children {
    max-height: 0;
    overflow: hidden;
    transition: var(--transition);
    padding-left: 1.4rem;

    &.open {
      max-height: 9999px;
    }
  }
}
Legend
Html element
Component
Html element with directive

results matching ""

    No results matching ""