@sarasanalytics-com/design-system 0.0.149 → 0.0.151

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -113,9 +113,11 @@ export class CategoryDropdownComponent extends FieldType {
113
113
  this.cdr.detectChanges();
114
114
  if (this.categoryContainer && this.dropdownPanel) {
115
115
  const containerWidth = this.categoryContainer.nativeElement.offsetWidth;
116
- const rootFontSize = parseFloat(getComputedStyle(document.documentElement).fontSize);
117
- const minWidthInRem = (containerWidth + this.props?.['panelWidthOffset'] || 160) / rootFontSize;
118
- this.renderer.setStyle(this.dropdownPanel.nativeElement, 'min-width', `${minWidthInRem}rem`);
116
+ const offset = this.props?.['panelWidthOffset'] ?? 160; // fallback if undefined
117
+ // Use explicit pixel width to avoid reflows caused by fit-content
118
+ const panelWidthPx = containerWidth + offset;
119
+ this.renderer.setStyle(this.dropdownPanel.nativeElement, 'min-width', `${panelWidthPx}px`);
120
+ this.renderer.setStyle(this.dropdownPanel.nativeElement, 'width', `${panelWidthPx}px`);
119
121
  }
120
122
  // 🔑 Sync selectedAttributes from formControl on every open
121
123
  if (this.props?.['multiple']) {
@@ -236,4 +238,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.2.4", ngImpor
236
238
  type: HostListener,
237
239
  args: ['document:click', ['$event']]
238
240
  }] } });
239
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"category-dropdown.component.js","sourceRoot":"","sources":["../../../../../../projects/component-library/src/lib/dropdown/category-dropdown/category-dropdown.component.ts","../../../../../../projects/component-library/src/lib/dropdown/category-dropdown/category-dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectorRef, Component, computed, effect, ElementRef, HostListener, inject, OnDestroy, OnInit, Renderer2, signal, ViewChild } from '@angular/core';\nimport { isObservable, Subject, Subscription } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\nimport { CommonModule } from '@angular/common';\nimport { ReactiveFormsModule, FormsModule } from '@angular/forms';\nimport { FormlyModule } from '@ngx-formly/core';\nimport { FieldType, FieldTypeConfig } from '@ngx-formly/core';\nimport { IconComponent } from '../../icon/icon.component';\nimport { IconService } from '../../icon/icon.service';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatInputModule } from '@angular/material/input';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { MatButtonToggleModule } from '@angular/material/button-toggle';\nimport { MatDatepickerModule } from '@angular/material/datepicker';\nimport { MatNativeDateModule } from '@angular/material/core';\nimport { MatCheckboxModule } from '@angular/material/checkbox';\nimport { MatSelectModule } from '@angular/material/select';\nimport {first as _first} from 'lodash';\n\n@Component({\n  selector: 'sa-category-dropdown',\n  standalone: true,\n  imports: [CommonModule,\n    ReactiveFormsModule,\n    FormsModule,\n    FormlyModule,\n    IconComponent,\n    MatFormFieldModule,\n    MatInputModule,\n    MatIconModule,\n    MatButtonModule,\n    MatTooltipModule,\n    MatButtonToggleModule,\n    MatDatepickerModule,\n    MatNativeDateModule,\n    MatCheckboxModule,\n    MatSelectModule],\n  providers: [IconService],\n  templateUrl: './category-dropdown.component.html',\n  styleUrl: './category-dropdown.component.css'\n})\n\nexport class CategoryDropdownComponent extends FieldType<FieldTypeConfig> implements OnInit, OnDestroy {\n  get bindLabel(): string { return this.props['bindLabel'] || 'label'; }\n  get bindValue(): string { return this.props['bindValue'] || 'value'; }\n  get bindAttributes(): string { return this.props['bindAttributes'] || 'attributes'; }\n  _categories = signal<any[]>([]);\n  destroy$ = new Subject<void>();\n  subscription = new Subscription();\n\n  selectedCategory = signal<any>(undefined);\n  selectedAttributes = signal<any[]>([]);\n  searchTerm = signal<string>('');\n  isOpen = signal(false);\n  userSelectedCategory = signal<any>(undefined); // To hold the user's explicit selection\n\n  private el: ElementRef = inject(ElementRef);\n  @ViewChild('searchInput') searchInput: ElementRef;\n  @ViewChild('categoryContainer') categoryContainer: ElementRef;\n  @ViewChild('dropdownPanel') dropdownPanel: ElementRef;\n\n  constructor(private cdr: ChangeDetectorRef, private renderer: Renderer2) {\n    super();\n    effect(() => {\n      const filtered = this.filteredCategories();\n      const term = this.searchTerm();\n    \n      if (term && filtered.length > 0) {\n        this.selectedCategory.set(_first(filtered));\n        return;\n      }\n    \n      // If attributes exist, stick with their category\n      const attrCategory = this.getSelectedAttributesCategory();\n      if (attrCategory) {\n        this.selectedCategory.set(attrCategory);\n        return;\n      }\n    \n      // Else fallback to user-selected or first\n      if (this.userSelectedCategory()) {\n        this.selectedCategory.set(this.userSelectedCategory());\n      } else if (this._categories().length > 0) {\n        this.selectedCategory.set(_first(this._categories()));\n      }\n    }, { allowSignalWrites: true });\n  }\n\n  ngOnInit(): void {\n    if (this.props?.['multiple']) {\n      const initialValue = this.formControl.value;\n      this.selectedAttributes.set(Array.isArray(initialValue) ? initialValue : []);\n    }\n\n    if (isObservable(this.props.options)) {\n      this.subscription.add(this.props.options.pipe(takeUntil(this.destroy$)).subscribe(options => {\n        this._categories.set(Array.isArray(options) ? options : []);\n        if (this._categories().length > 0 && !this.userSelectedCategory()) {\n          this.selectedCategory.set(this._categories()[0]);\n        }\n      }));\n    } else {\n      this._categories.set(Array.isArray(this.props.options) ? this.props.options : []);\n      if (this._categories().length > 0 && !this.userSelectedCategory()) {\n        this.selectedCategory.set(this._categories()[0]);\n      }\n    }\n  }\n\n  @HostListener('document:click', ['$event'])\n  onClick(event: MouseEvent) {\n    if (!this.el.nativeElement.contains(event.target)) {\n      this.isOpen.set(false);\n    }\n  }\n\n  toggleDropdown() {\n    const isOpening = !this.isOpen();\n    this.isOpen.set(isOpening);\n  \n    if (isOpening) {\n      this.searchTerm.set('');\n      if (this.searchInput) {\n        this.searchInput.nativeElement.value = '';\n      }\n      this.cdr.detectChanges();\n  \n      if (this.categoryContainer && this.dropdownPanel) {\n        const containerWidth = this.categoryContainer.nativeElement.offsetWidth;\n        const rootFontSize = parseFloat(getComputedStyle(document.documentElement).fontSize);\n        const minWidthInRem = (containerWidth + this.props?.['panelWidthOffset'] || 160) / rootFontSize;\n        this.renderer.setStyle(this.dropdownPanel.nativeElement, 'min-width', `${minWidthInRem}rem`);\n      }\n  \n      // 🔑 Sync selectedAttributes from formControl on every open\n      if (this.props?.['multiple']) {\n        const currentValue = this.formControl.value || [];\n        this.selectedAttributes.set(Array.isArray(currentValue) ? currentValue : []);\n      } else if (this.formControl.value) {\n        this.selectedAttributes.set([this.formControl.value]);\n      } else {\n        this.selectedAttributes.set([]);\n      }\n  \n      // 👇 Now this works, because selectedAttributes isn’t empty anymore\n      const attrCategory = this.getSelectedAttributesCategory();\n      if (attrCategory) {\n        this.selectedCategory.set(attrCategory);\n      } else if (this.userSelectedCategory()) {\n        this.selectedCategory.set(this.userSelectedCategory());\n      } else if (this._categories().length > 0) {\n        this.selectedCategory.set(this._categories()[0]);\n      }\n    }\n  }\n\n  selectCategory(category: any) {\n    this.selectedCategory.set(category);\n    // Only update userSelectedCategory if there are NO attributes selected\n    if (this.selectedAttributes().length === 0) {\n      this.userSelectedCategory.set(category);\n    }\n  }\n\n  toggleValue(val: any) {\n    this.selectedAttributes.update(attributes => {\n      const index = attributes.findIndex(item => item[this.bindValue] === val[this.bindValue]);\n      if (index > -1) {\n        attributes.splice(index, 1);\n      } else {\n        const category = this.selectedCategory();\n        const newValue = { ...val, categoryId: category[this.bindValue], categoryLabel: category[this.bindLabel] };\n        attributes.push(newValue);\n      }\n      this.formControl.setValue([...attributes]);\n      return [...attributes];\n    });\n  }\n\n  private getSelectedAttributesCategory(): any {\n    const attrs = this.selectedAttributes();\n    if (attrs.length > 0) {\n      // always prefer category of last selected attribute\n      const lastAttr = attrs[attrs.length - 1];\n      return this._categories().find(c => c[this.bindValue] === lastAttr.categoryId);\n    }\n    return null;\n  }\n\n  isItemSelected(val: any): boolean {\n    return this.selectedAttributes().some(item => item[this.bindValue] === val[this.bindValue]);\n  }\n\n  removeValue(val: any) {\n    this.selectedAttributes.update(attributes => {\n      const newAttributes = attributes.filter(item => item[this.bindValue] !== val[this.bindValue]);\n      this.formControl.setValue(newAttributes);\n      return newAttributes;\n    });\n  }\n\n  selectSingleValue(val: any) {\n    const category = this.selectedCategory();\n    // const newValue = { ...val, categoryId: category[this.bindValue], categoryLabel: category[this.bindLabel] };\n    // this.formControl.setValue(newValue);\n    this.formControl.setValue(val);\n    this.isOpen.set(false);\n  }\n\n  onSearch(event: Event) {\n    const value = (event.target as HTMLInputElement).value;\n    this.searchTerm.set(value);\n  }\n\n  clearSearch(event: Event) {\n    this.searchTerm.set('');\n    this.searchInput.nativeElement.value = '';\n    event.stopPropagation();\n    \n  }\n\n  ngOnDestroy(): void {\n    this.subscription.unsubscribe();\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  filteredAttributes = computed(() => {\n    const category = this.selectedCategory();\n    const term = this.searchTerm().toLowerCase();\n\n    if (!category) {\n      return [];\n    }\n\n    return (category[this.bindAttributes] || []).filter((val: any) => (val[this.bindLabel] || '').toLowerCase().includes(term));\n  });\n\n  filteredCategories = computed(() => {\n    const term = this.searchTerm().toLowerCase();\n    const categories = this._categories();\n    if (!term) {\n      return categories;\n    }\n    return categories.filter(category =>\n      (category[this.bindAttributes] || []).some((val: any) => (val[this.bindLabel] || '').toLowerCase().includes(term))\n    );\n  });\n}","<div class=\"custom-category-dropdown relative\" (click)=\"toggleDropdown()\">\n    <!-- Dropdown Trigger / Display -->\n    <div #categoryContainer class=\"custom-category-container\" [class.open]=\"isOpen()\">\n        <div class=\"custom-category-attribute-container\">\n            @if(props?.['multiple']){\n            @if (selectedAttributes().length > 0) {\n            @for (item of selectedAttributes(); track item[bindValue]) {\n            <div class=\"custom-category-attribute custom-category-custom-chip\" (click)=\"$event.stopPropagation()\">\n                <span class=\"custom-category-attribute-label\">{{ item[bindLabel] }}</span>\n                <span class=\"custom-category-attribute-icon right\">\n                    <sa-icon icon=\"closeOutlined\" (click)=\"removeValue(item)\" size=\"11\"></sa-icon>\n                </span>\n            </div>\n            }\n            } @else {\n            <span class=\"placeholder\">Select items</span>\n            }\n            } @else {\n            @if(formControl.value) {\n            <span class=\"custom-category-attribute-label\">{{ formControl.value[bindLabel] }}</span>\n            } @else {\n            <span class=\"placeholder\">Select item</span>\n            }\n            }\n        </div>\n        <div class=\"custom-category-arrow-wrapper\">\n            <sa-icon icon=\"upDownChevronOutlined\" size=\"11\"></sa-icon>\n        </div>\n    </div>\n\n    <!-- Dropdown Panel -->\n    @if (isOpen()) {\n    <div #dropdownPanel class=\"dropdown-panel\" (click)=\"$event.stopPropagation()\">\n        <div class=\"search-input-container mat-form-field-density-5\">\n            <mat-form-field [appearance]=\"props['appearance'] || 'outline'\" class=\"w-100\">\n                <sa-icon icon=\"searchIcon\" size=\"20\" matPrefix></sa-icon>\n                <input #searchInput matInput type=\"text\" name=\"menu-search-bar\" class=\"search-input\"\n                    (keyup)=\"onSearch($event)\" [placeholder]=\"props?.['searchPlaceholder'] || 'Search'\" />\n                <sa-icon class=\"pointer\" icon=\"closeOutlined\" size=\"11\" matSuffix (click)=\"clearSearch($event)\"></sa-icon>\n            </mat-form-field>\n        </div>\n\n        @if (filteredCategories().length > 0) {\n        <div class=\"two-panel-dropdown\">\n            <!-- Left Panel: Categories -->\n            <div class=\"category-panel\">\n                @for (cat of filteredCategories(); track cat[bindValue]) {\n                <div class=\"category-item\" [class.active]=\"cat[bindValue] === selectedCategory()?.[bindValue]\"\n                    (click)=\"selectCategory(cat); $event.stopPropagation()\">\n                    <span class=\"category-label\">{{ cat[bindLabel] }}\n                        @if(props?.['showAttributesCount']){\n                        <span class=\"attributes-count\">({{ cat?.[bindAttributes]?.length }})</span>\n                        }\n                    </span>\n                </div>\n                }\n            </div>\n            <div class=\"dropdown-divider\"></div>\n\n            <!-- Right Panel: Attributes -->\n            <div class=\"attributes-panel\">\n                @if(filteredAttributes().length > 0) {\n                @for (val of filteredAttributes(); track val[bindValue]) {\n                <div class=\"attribute-item pointer\"\n                    [class.active]=\"props?.['multiple'] ? isItemSelected(val) : formControl.value?.[bindValue] === val[bindValue]\" (click)=\"props?.['multiple'] ? toggleValue(val) : selectSingleValue(val); $event.stopPropagation()\">\n                    @if(props?.['multiple']){\n                    <label class=\"custom-checkbox-container\" (click)=\"$event.stopPropagation()\">\n                        <input type=\"checkbox\" [checked]=\"isItemSelected(val)\" (change)=\"toggleValue(val)\">\n                        <span class=\"checkmark\"></span>\n                        <span class=\"attribute-label\">{{ val[bindLabel] }}</span>\n                    </label>\n                    }\n                    @else{\n                    <label class=\"custom-checkbox-container\">\n                        <span class=\"attribute-label\">{{ val[bindLabel] }}</span>\n                        @if(val?.tooltip || val?.description){\n                        <span class=\"tooltip-container\">\n                            <sa-icon [icon]=\"props?.['tooltipIcon'] || 'infoCircleOutlined'\" customClass=\"info-icon\"\n                                [matTooltip]=\"val?.tooltip || val?.description\" matTooltipClass=\"custom-tooltip\"\n                                class=\"tooltip-icon\"></sa-icon>\n                        </span>\n                        }\n                    </label>\n                    }\n                </div>\n                }\n                }@else {\n                <div class=\"p-4 text-center text-gray-500 w-full\">\n                    No items found.\n                </div>\n                }\n            </div>\n        </div>\n        } @else {\n        <div class=\"p-4 text-center text-gray-500 w-full\">\n            No items found.\n        </div>\n        }\n    </div>\n    }\n</div>"]}
241
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"category-dropdown.component.js","sourceRoot":"","sources":["../../../../../../projects/component-library/src/lib/dropdown/category-dropdown/category-dropdown.component.ts","../../../../../../projects/component-library/src/lib/dropdown/category-dropdown/category-dropdown.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectorRef, Component, computed, effect, ElementRef, HostListener, inject, OnDestroy, OnInit, Renderer2, signal, ViewChild } from '@angular/core';\nimport { isObservable, Subject, Subscription } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\nimport { CommonModule } from '@angular/common';\nimport { ReactiveFormsModule, FormsModule } from '@angular/forms';\nimport { FormlyModule } from '@ngx-formly/core';\nimport { FieldType, FieldTypeConfig } from '@ngx-formly/core';\nimport { IconComponent } from '../../icon/icon.component';\nimport { IconService } from '../../icon/icon.service';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatInputModule } from '@angular/material/input';\nimport { MatIconModule } from '@angular/material/icon';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { MatButtonToggleModule } from '@angular/material/button-toggle';\nimport { MatDatepickerModule } from '@angular/material/datepicker';\nimport { MatNativeDateModule } from '@angular/material/core';\nimport { MatCheckboxModule } from '@angular/material/checkbox';\nimport { MatSelectModule } from '@angular/material/select';\nimport {first as _first} from 'lodash';\n\n@Component({\n  selector: 'sa-category-dropdown',\n  standalone: true,\n  imports: [CommonModule,\n    ReactiveFormsModule,\n    FormsModule,\n    FormlyModule,\n    IconComponent,\n    MatFormFieldModule,\n    MatInputModule,\n    MatIconModule,\n    MatButtonModule,\n    MatTooltipModule,\n    MatButtonToggleModule,\n    MatDatepickerModule,\n    MatNativeDateModule,\n    MatCheckboxModule,\n    MatSelectModule],\n  providers: [IconService],\n  templateUrl: './category-dropdown.component.html',\n  styleUrl: './category-dropdown.component.css'\n})\n\nexport class CategoryDropdownComponent extends FieldType<FieldTypeConfig> implements OnInit, OnDestroy {\n  get bindLabel(): string { return this.props['bindLabel'] || 'label'; }\n  get bindValue(): string { return this.props['bindValue'] || 'value'; }\n  get bindAttributes(): string { return this.props['bindAttributes'] || 'attributes'; }\n  _categories = signal<any[]>([]);\n  destroy$ = new Subject<void>();\n  subscription = new Subscription();\n\n  selectedCategory = signal<any>(undefined);\n  selectedAttributes = signal<any[]>([]);\n  searchTerm = signal<string>('');\n  isOpen = signal(false);\n  userSelectedCategory = signal<any>(undefined); // To hold the user's explicit selection\n\n  private el: ElementRef = inject(ElementRef);\n  @ViewChild('searchInput') searchInput: ElementRef;\n  @ViewChild('categoryContainer') categoryContainer: ElementRef;\n  @ViewChild('dropdownPanel') dropdownPanel: ElementRef;\n\n  constructor(private cdr: ChangeDetectorRef, private renderer: Renderer2) {\n    super();\n    effect(() => {\n      const filtered = this.filteredCategories();\n      const term = this.searchTerm();\n    \n      if (term && filtered.length > 0) {\n        this.selectedCategory.set(_first(filtered));\n        return;\n      }\n    \n      // If attributes exist, stick with their category\n      const attrCategory = this.getSelectedAttributesCategory();\n      if (attrCategory) {\n        this.selectedCategory.set(attrCategory);\n        return;\n      }\n    \n      // Else fallback to user-selected or first\n      if (this.userSelectedCategory()) {\n        this.selectedCategory.set(this.userSelectedCategory());\n      } else if (this._categories().length > 0) {\n        this.selectedCategory.set(_first(this._categories()));\n      }\n    }, { allowSignalWrites: true });\n  }\n\n  ngOnInit(): void {\n    if (this.props?.['multiple']) {\n      const initialValue = this.formControl.value;\n      this.selectedAttributes.set(Array.isArray(initialValue) ? initialValue : []);\n    }\n\n    if (isObservable(this.props.options)) {\n      this.subscription.add(this.props.options.pipe(takeUntil(this.destroy$)).subscribe(options => {\n        this._categories.set(Array.isArray(options) ? options : []);\n        if (this._categories().length > 0 && !this.userSelectedCategory()) {\n          this.selectedCategory.set(this._categories()[0]);\n        }\n      }));\n    } else {\n      this._categories.set(Array.isArray(this.props.options) ? this.props.options : []);\n      if (this._categories().length > 0 && !this.userSelectedCategory()) {\n        this.selectedCategory.set(this._categories()[0]);\n      }\n    }\n  }\n\n  @HostListener('document:click', ['$event'])\n  onClick(event: MouseEvent) {\n    if (!this.el.nativeElement.contains(event.target)) {\n      this.isOpen.set(false);\n    }\n  }\n\n  toggleDropdown() {\n    const isOpening = !this.isOpen();\n    this.isOpen.set(isOpening);\n  \n    if (isOpening) {\n      this.searchTerm.set('');\n      if (this.searchInput) {\n        this.searchInput.nativeElement.value = '';\n      }\n      this.cdr.detectChanges();\n  \n      if (this.categoryContainer && this.dropdownPanel) {\n        const containerWidth = this.categoryContainer.nativeElement.offsetWidth;\n        const offset = this.props?.['panelWidthOffset'] ?? 160; // fallback if undefined\n\n        // Use explicit pixel width to avoid reflows caused by fit-content\n        const panelWidthPx = containerWidth + offset;\n        this.renderer.setStyle(this.dropdownPanel.nativeElement, 'min-width', `${panelWidthPx}px`);\n        this.renderer.setStyle(this.dropdownPanel.nativeElement, 'width', `${panelWidthPx}px`);\n      }\n  \n      // 🔑 Sync selectedAttributes from formControl on every open\n      if (this.props?.['multiple']) {\n        const currentValue = this.formControl.value || [];\n        this.selectedAttributes.set(Array.isArray(currentValue) ? currentValue : []);\n      } else if (this.formControl.value) {\n        this.selectedAttributes.set([this.formControl.value]);\n      } else {\n        this.selectedAttributes.set([]);\n      }\n  \n      // 👇 Now this works, because selectedAttributes isn’t empty anymore\n      const attrCategory = this.getSelectedAttributesCategory();\n      if (attrCategory) {\n        this.selectedCategory.set(attrCategory);\n      } else if (this.userSelectedCategory()) {\n        this.selectedCategory.set(this.userSelectedCategory());\n      } else if (this._categories().length > 0) {\n        this.selectedCategory.set(this._categories()[0]);\n      }\n    }\n  }\n\n  selectCategory(category: any) {\n    this.selectedCategory.set(category);\n    // Only update userSelectedCategory if there are NO attributes selected\n    if (this.selectedAttributes().length === 0) {\n      this.userSelectedCategory.set(category);\n    }\n  }\n\n  toggleValue(val: any) {\n    this.selectedAttributes.update(attributes => {\n      const index = attributes.findIndex(item => item[this.bindValue] === val[this.bindValue]);\n      if (index > -1) {\n        attributes.splice(index, 1);\n      } else {\n        const category = this.selectedCategory();\n        const newValue = { ...val, categoryId: category[this.bindValue], categoryLabel: category[this.bindLabel] };\n        attributes.push(newValue);\n      }\n      this.formControl.setValue([...attributes]);\n      return [...attributes];\n    });\n  }\n\n  private getSelectedAttributesCategory(): any {\n    const attrs = this.selectedAttributes();\n    if (attrs.length > 0) {\n      // always prefer category of last selected attribute\n      const lastAttr = attrs[attrs.length - 1];\n      return this._categories().find(c => c[this.bindValue] === lastAttr.categoryId);\n    }\n    return null;\n  }\n\n  isItemSelected(val: any): boolean {\n    return this.selectedAttributes().some(item => item[this.bindValue] === val[this.bindValue]);\n  }\n\n  removeValue(val: any) {\n    this.selectedAttributes.update(attributes => {\n      const newAttributes = attributes.filter(item => item[this.bindValue] !== val[this.bindValue]);\n      this.formControl.setValue(newAttributes);\n      return newAttributes;\n    });\n  }\n\n  selectSingleValue(val: any) {\n    const category = this.selectedCategory();\n    // const newValue = { ...val, categoryId: category[this.bindValue], categoryLabel: category[this.bindLabel] };\n    // this.formControl.setValue(newValue);\n    this.formControl.setValue(val);\n    this.isOpen.set(false);\n  }\n\n  onSearch(event: Event) {\n    const value = (event.target as HTMLInputElement).value;\n    this.searchTerm.set(value);\n  }\n\n  clearSearch(event: Event) {\n    this.searchTerm.set('');\n    this.searchInput.nativeElement.value = '';\n    event.stopPropagation();\n    \n  }\n\n  ngOnDestroy(): void {\n    this.subscription.unsubscribe();\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  filteredAttributes = computed(() => {\n    const category = this.selectedCategory();\n    const term = this.searchTerm().toLowerCase();\n\n    if (!category) {\n      return [];\n    }\n\n    return (category[this.bindAttributes] || []).filter((val: any) => (val[this.bindLabel] || '').toLowerCase().includes(term));\n  });\n\n  filteredCategories = computed(() => {\n    const term = this.searchTerm().toLowerCase();\n    const categories = this._categories();\n    if (!term) {\n      return categories;\n    }\n    return categories.filter(category =>\n      (category[this.bindAttributes] || []).some((val: any) => (val[this.bindLabel] || '').toLowerCase().includes(term))\n    );\n  });\n}","<div class=\"custom-category-dropdown relative\" (click)=\"toggleDropdown()\">\n    <!-- Dropdown Trigger / Display -->\n    <div #categoryContainer class=\"custom-category-container\" [class.open]=\"isOpen()\">\n        <div class=\"custom-category-attribute-container\">\n            @if(props?.['multiple']){\n            @if (selectedAttributes().length > 0) {\n            @for (item of selectedAttributes(); track item[bindValue]) {\n            <div class=\"custom-category-attribute custom-category-custom-chip\" (click)=\"$event.stopPropagation()\">\n                <span class=\"custom-category-attribute-label\">{{ item[bindLabel] }}</span>\n                <span class=\"custom-category-attribute-icon right\">\n                    <sa-icon icon=\"closeOutlined\" (click)=\"removeValue(item)\" size=\"11\"></sa-icon>\n                </span>\n            </div>\n            }\n            } @else {\n            <span class=\"placeholder\">Select items</span>\n            }\n            } @else {\n            @if(formControl.value) {\n            <span class=\"custom-category-attribute-label\">{{ formControl.value[bindLabel] }}</span>\n            } @else {\n            <span class=\"placeholder\">Select item</span>\n            }\n            }\n        </div>\n        <div class=\"custom-category-arrow-wrapper\">\n            <sa-icon icon=\"upDownChevronOutlined\" size=\"11\"></sa-icon>\n        </div>\n    </div>\n\n    <!-- Dropdown Panel -->\n    @if (isOpen()) {\n    <div #dropdownPanel class=\"dropdown-panel\" (click)=\"$event.stopPropagation()\">\n        <div class=\"search-input-container mat-form-field-density-5\">\n            <mat-form-field [appearance]=\"props['appearance'] || 'outline'\" class=\"w-100\">\n                <sa-icon icon=\"searchIcon\" size=\"20\" matPrefix></sa-icon>\n                <input #searchInput matInput type=\"text\" name=\"menu-search-bar\" class=\"search-input\"\n                    (keyup)=\"onSearch($event)\" [placeholder]=\"props?.['searchPlaceholder'] || 'Search'\" />\n                <sa-icon class=\"pointer\" icon=\"closeOutlined\" size=\"11\" matSuffix (click)=\"clearSearch($event)\"></sa-icon>\n            </mat-form-field>\n        </div>\n\n        @if (filteredCategories().length > 0) {\n        <div class=\"two-panel-dropdown\">\n            <!-- Left Panel: Categories -->\n            <div class=\"category-panel\">\n                @for (cat of filteredCategories(); track cat[bindValue]) {\n                <div class=\"category-item\" [class.active]=\"cat[bindValue] === selectedCategory()?.[bindValue]\"\n                    (click)=\"selectCategory(cat); $event.stopPropagation()\">\n                    <span class=\"category-label\">{{ cat[bindLabel] }}\n                        @if(props?.['showAttributesCount']){\n                        <span class=\"attributes-count\">({{ cat?.[bindAttributes]?.length }})</span>\n                        }\n                    </span>\n                </div>\n                }\n            </div>\n            <div class=\"dropdown-divider\"></div>\n\n            <!-- Right Panel: Attributes -->\n            <div class=\"attributes-panel\">\n                @if(filteredAttributes().length > 0) {\n                @for (val of filteredAttributes(); track val[bindValue]) {\n                <div class=\"attribute-item pointer\"\n                    [class.active]=\"props?.['multiple'] ? isItemSelected(val) : formControl.value?.[bindValue] === val[bindValue]\" (click)=\"props?.['multiple'] ? toggleValue(val) : selectSingleValue(val); $event.stopPropagation()\">\n                    @if(props?.['multiple']){\n                    <label class=\"custom-checkbox-container\" (click)=\"$event.stopPropagation()\">\n                        <input type=\"checkbox\" [checked]=\"isItemSelected(val)\" (change)=\"toggleValue(val)\">\n                        <span class=\"checkmark\"></span>\n                        <span class=\"attribute-label\">{{ val[bindLabel] }}</span>\n                    </label>\n                    }\n                    @else{\n                    <label class=\"custom-checkbox-container\">\n                        <span class=\"attribute-label\">{{ val[bindLabel] }}</span>\n                        @if(val?.tooltip || val?.description){\n                        <span class=\"tooltip-container\">\n                            <sa-icon [icon]=\"props?.['tooltipIcon'] || 'infoCircleOutlined'\" customClass=\"info-icon\"\n                                [matTooltip]=\"val?.tooltip || val?.description\" matTooltipClass=\"custom-tooltip\"\n                                class=\"tooltip-icon\"></sa-icon>\n                        </span>\n                        }\n                    </label>\n                    }\n                </div>\n                }\n                }@else {\n                <div class=\"p-4 text-center text-gray-500 w-full\">\n                    No items found.\n                </div>\n                }\n            </div>\n        </div>\n        } @else {\n        <div class=\"p-4 text-center text-gray-500 w-full\">\n            No items found.\n        </div>\n        }\n    </div>\n    }\n</div>"]}
@@ -3277,9 +3277,11 @@ class CategoryDropdownComponent extends FieldType {
3277
3277
  this.cdr.detectChanges();
3278
3278
  if (this.categoryContainer && this.dropdownPanel) {
3279
3279
  const containerWidth = this.categoryContainer.nativeElement.offsetWidth;
3280
- const rootFontSize = parseFloat(getComputedStyle(document.documentElement).fontSize);
3281
- const minWidthInRem = (containerWidth + this.props?.['panelWidthOffset'] || 160) / rootFontSize;
3282
- this.renderer.setStyle(this.dropdownPanel.nativeElement, 'min-width', `${minWidthInRem}rem`);
3280
+ const offset = this.props?.['panelWidthOffset'] ?? 160; // fallback if undefined
3281
+ // Use explicit pixel width to avoid reflows caused by fit-content
3282
+ const panelWidthPx = containerWidth + offset;
3283
+ this.renderer.setStyle(this.dropdownPanel.nativeElement, 'min-width', `${panelWidthPx}px`);
3284
+ this.renderer.setStyle(this.dropdownPanel.nativeElement, 'width', `${panelWidthPx}px`);
3283
3285
  }
3284
3286
  // 🔑 Sync selectedAttributes from formControl on every open
3285
3287
  if (this.props?.['multiple']) {