xxxxxxxxxx
1
<div class="row d-block d-lg-none d-xl-none d-xxl-none">
2
<div class="d-flex align-items-center">
3
<mdb-form-control class="flex-grow-1">
4
<input
5
mdbInput
6
placeholder="Search..."
7
[ngModel]="searchText | async"
8
(ngModelChange)="searchText.next($event)"
9
[mdbAutocomplete]="autocomplete"
10
type="text"
11
id="autocomplete"
12
class="form-control border"
13
(keydown)="onKeyDown($event)"
14
(keyup)="onChange($event)"
15
aria-describedby="basic-addon2"
16
/>
17
</mdb-form-control>
18
19
<span
20
class="input-group-text"
21
id="basic-addon2"
22
style="background: #032B49; color: #FFF; border-color: #032B49; height: 35px; width: 40px; border-top-left-radius: 0px; border-bottom-left-radius: 0px"
23
(click)="searchTerm()"
24
>
25
<a role="button" style="color: #FFF">
26
<i *ngIf="!loading" class="fas fa-search"></i>
27
<div
28
*ngIf="loading"
29
class="autocomplete-loader spinner-border"
30
style="height: 16px; width: 16px; position: inherit;"
31
role="status"
32
></div>
33
</a>
34
</span>
35
</div>
36
37
<mdb-autocomplete
38
#autocomplete="mdbAutocomplete"
39
[displayValue]="displayValue"
40
(opened)="onOpen()"
41
(selected)="goToProductOption($event)"
42
>
43
<ng-container *ngIf="results | async as options">
44
<ng-container *ngFor="let option of options">
45
<mdb-option *ngIf="option && option.images && option.images.length > 0" [value]="option.brand + ' - ' + option.style" (click)="goToProduct(option)">
46
<img [src]="getVendorImageUrl(option)" alt="Photo Image" class="option-icon rounded-circle me-1">
47
{{ option.brand + ' ' + option.style + ' ' + option.title | ellipsis: 48 }}
48
</mdb-option>
49
</ng-container>
50
</ng-container>
51
52
53
<div *ngIf="notFound" class="autocomplete-no-results">No results found</div>
54
</mdb-autocomplete>
55
</div>
1
1
xxxxxxxxxx
1
import { HttpClient } from '@angular/common/http';
2
import { Component, Input, OnInit, Output, EventEmitter, ViewChild, ElementRef, ChangeDetectorRef } from '@angular/core';
3
import { UntypedFormControl } from '@angular/forms';
4
import { Router } from '@angular/router';
5
import { BehaviorSubject, debounceTime, delay, map, Observable, Subject, switchMap, take, tap } from 'rxjs';
6
import { MongoDbService } from 'src/app/core/services/api/mongodb.service';
7
import { environment } from 'src/environments/environment';
8
9
10
@Component({
11
selector: 'app-nav-menu',
12
templateUrl: './nav-menu.component.html',
13
styleUrls: ['./nav-menu.component.scss'],
14
})
15
export class NavMenuComponent implements OnInit {
16
17
public search: string;
18
private isText = false;
19
private timeout: any = null
20
21
results: Observable<any>
22
notFound = false;
23
loading = false;
24
dataLoaded = false;
25
navigatingDropdown: boolean = false;
26
initialSearchText: string | null = null;
27
28
searchText: BehaviorSubject<string> = new BehaviorSubject<string>('');
29
searchTermValue: string;
30
selectedOption: any; // Property to hold the selected option
31
32
@Input() searchParam = '';
33
@Input() brands;
34
@Input() categories;
35
@Input() viewport
36
@Input() decorations;
37
38
@Output() menuOption = new EventEmitter();
39
40
readonly inputSearch = new UntypedFormControl('')
41
searchValue: string;
42
43
constructor(
44
private router: Router,
45
private http: HttpClient,
46
private _mongo: MongoDbService,
47
private cdRef: ChangeDetectorRef
48
) {
49
this.searchText.pipe(take(1)).subscribe(value => {
50
this.searchTermValue = value
51
});
52
53
this.results = this.searchText.pipe(
54
debounceTime(250),
55
switchMap((value: any) => {
56
const name = typeof value === 'string' ? value : '';
57
return this.filter(name)
58
}),
59
tap((results: any) =>
60
results && results.length > 0 ? (this.notFound = false) : (this.notFound = true)
61
)
62
);
63
}
64
65
ngOnInit(): void {
66
this.search = this.searchParam
67
if(this.categories){
68
let data = Array.of(this.categories)
69
}
70
}
71
72
onSearchInputChange(value: string) {
73
this.searchText.next(value);
74
}
75
76
searchTerm() {
77
this.searchText.pipe(take(1)).subscribe(value => {
78
this.filterNavbar('search', value)
79
80
});
81
}
82
83
filter(value: string): any {
84
const filterValue = value.toLowerCase();
85
if(!this.navigatingDropdown) {
86
const url = `${environment.apiMicroservice}products/search?search=${encodeURI(filterValue)}&page=1&pageSize=10`;
87
88
this.searchValue = value;
89
this.loading = true;
90
91
return this.http.get(url).pipe(
92
map((data: any) => {
93
return data.products
94
}),
95
delay(300),
96
tap(() => (this.loading = false))
97
);
98
}
99
}
100
101
displayValue(value: any): string {
102
return value ? value : '';
103
}
104
105
onOpen(): void {
106
if (!this.dataLoaded) {
107
this.searchText.next('');
108
this.selectedOption = null; // Clear the selected option when autocomplete is opened
109
this.dataLoaded = true;
110
}
111
}
112
113
// Function to handle image URLs based on the vendor
114
getVendorImageUrl(option: any): string {
115
let baseUrl = 'https://img.stokkup.com/app/no-image.png'; // Default URL for images
116
117
if (option && option.vendor) {
118
if (option.vendor.includes('ssactivewear')) {
119
baseUrl = 'https://cdn.ssactivewear.com/';
120
} else if (option.vendor.includes('alphabroder')) {
121
baseUrl = 'https://www.alphabroder.com/prodimg/large/';
122
} else if(option.vendor.includes('sanmar') || option.vendor.includes('AS Colour') || option.customProduct && option.customProduct === true ) {
123
baseUrl = ''
124
}
125
// Add more conditions for other vendors if needed
126
}
127
128
return baseUrl + (option?.images[0] || ''); // Append the image name to the base URL
129
}
130
131
goToProduct(option: any) {
132
const url = `/catalog/${encodeURIComponent(option.brand)}/${option.style}`;
133
134
location.href = url;
135
136
}
137
goToProductOption(option: any) {
138
this.selectedOption = option; // Capture the selected option
139
140
const [brand, title] = option.option.value.split('-');
141
if(this.navigatingDropdown) {
142
const url = `/catalog/${encodeURIComponent(brand)}/${title}`;
143
location.href = url;
144
}
145
146
147
}
148
149
150
onChange(event){
151
if(event.target.value){
152
this.isText = true
153
}else {
154
this.isText = false
155
}
156
157
let value = event.target.value
158
this.search = value
159
160
clearTimeout(this.timeout)
161
this.timeout = setTimeout(() => {
162
if(this.navigatingDropdown && event.keyCode == 13) {
163
let searchQueryObj = {
164
search : this.search
165
}
166
this.goToProduct(searchQueryObj)
167
}
168
else if(event.keyCode == 13){
169
this.filterNavbar('search', this.search) // event emitter to hit api
170
}
171
}, 1500)
172
}
173
174
filterNavbar(key: string, item: string){
175
this.search = !key ? '' : this.search
176
let option = {key, item}
177
if(!key && !item) {
178
this.router.navigate(['catalog/'])
179
} else {
180
this.menuOption.emit(option);
181
}
182
}
183
184
onKeyDown(event: KeyboardEvent) {
185
// Detect arrow keys (Up: 38, Down: 40)
186
if (event.key === "ArrowUp" || event.key === "ArrowDown") {
187
this.navigatingDropdown = true;
188
}
189
}
190
191
192
clear(){
193
this.isText = false
194
this.search = '';
195
this.filterNavbar('', this.search)
196
}
197
198
}
199
Console errors: 0