@salesforcedevs/dx-components 1.31.0 → 1.32.0-alpha.1

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.
@@ -1,526 +1,205 @@
1
1
  import { LightningElement, api, track } from "lwc";
2
- import type * as CoveoSDK from "coveo-search-ui";
3
- import { DateTime } from "luxon";
2
+ import debounce from "debounce";
4
3
  import { track as trackGTM } from "dxUtils/analytics";
5
4
  import {
6
- CONTENT_TYPE_LABELS,
7
- CONTENT_TYPE_ICONS
8
- } from "dxConstants/contentTypes";
9
- import { getContentTypeColorVariables } from "dxUtils/contentTypes";
10
- import { pollUntil } from "dxUtils/async";
11
- import resultsTemplate from "./resultsTemplate";
12
-
13
- // Max height for breadcrumb without wrapping
14
- const MAX_BREADCRUMB_HEIGHT = 16;
15
-
16
- interface CoveoSearch {
17
- state: typeof CoveoSDK.state;
18
- get: typeof CoveoSDK.get;
19
- $$: typeof CoveoSDK.$$;
20
- InitializationEvents: typeof CoveoSDK.InitializationEvents;
21
- QueryEvents: typeof CoveoSDK.QueryEvents;
22
- SearchEndpoint: typeof CoveoSDK.SearchEndpoint;
23
- TemplateCache: typeof CoveoSDK.TemplateCache;
24
- UnderscoreTemplate: typeof CoveoSDK.UnderscoreTemplate;
25
- init: typeof CoveoSDK.init;
26
- IQuerySuccessEventArgs: CoveoSDK.IQuerySuccessEventArgs;
27
- TemplateHelpers: any;
28
- }
29
-
30
- declare const Coveo: CoveoSearch;
31
-
32
- function getPaginationState(event: CoveoSDK.IQuerySuccessEventArgs): {
33
- numberOfPages: number;
34
- currentPage: number;
35
- } {
36
- const pageSize = event.query.numberOfResults!;
37
- const totalResults = event.results.totalCount!;
38
- const numberOfPages = Math.ceil(totalResults / pageSize);
39
-
40
- const currentPage = event.query.firstResult! / pageSize + 1;
41
-
42
- return { numberOfPages, currentPage };
5
+ type Data360SearchResultItem,
6
+ fetchSearch,
7
+ getQueryFromUrl
8
+ } from "dxUtils/data360Search";
9
+
10
+ const SEARCH_DEBOUNCE_DELAY = 400;
11
+
12
+ /** Normalized result for template (with href, uniqueId, openInNewTab) */
13
+ interface SearchResultDisplay {
14
+ title: string;
15
+ href: string;
16
+ matchedText: string;
17
+ uniqueId: string;
18
+ openInNewTab: string | undefined;
19
+ rel: string | undefined;
43
20
  }
44
21
 
45
- const isInternalDomain = (domain: string) =>
46
- domain === "developer.salesforce.com" ||
47
- domain === "developer-website-s.herokuapp.com";
48
-
49
- const isTrailheadDomain = (domain: string) =>
50
- domain === "trailhead.salesforce.com" ||
51
- domain === "dev.trailhead.salesforce.com";
52
-
53
- const buildTemplateHelperBadge = (value: keyof typeof CONTENT_TYPE_LABELS) => {
54
- const style = getContentTypeColorVariables(value);
55
- const label = CONTENT_TYPE_LABELS[value];
56
- const { iconSprite, iconSymbol } = CONTENT_TYPE_ICONS[value];
57
- return `
58
- <div style="${style}" class="dx-badge">
59
- <svg
60
- aria-hidden="true"
61
- style="display: inline; margin-bottom: 1px; fill: var(--color); height: var(--dx-g-icon-size-xs); width: var(--dx-g-icon-size-xs);"
62
- xmlns="http://www.w3.org/2000/svg"
63
- part="svg"
64
- >
65
- <use xlink:href="/assets/icons/${iconSprite}-sprite/svg/symbols.svg#${iconSymbol}"></use>
66
- </svg>
67
- <span>${label}</span>
68
- </div>
69
- `;
70
- };
71
-
72
- const processParts = (parts: string[], internalFlag = false) => {
73
- // filter /docs/ breadcrumb item from internal domains
74
- const filterFn = internalFlag
75
- ? (part: string) => part !== "docs"
76
- : (part: string) => part;
77
-
78
- return parts.filter(filterFn).map((part) => {
79
- // Remove special characters & .htm/.xml extension
80
- part = part
81
- .replace(/_/g, "")
82
- .replace(/-/g, "")
83
- .replace(/.html*/g, "")
84
- .replace(/.xml/g, "")
85
- .replace(/b2c/g, "B2C");
86
-
87
- // Capitalize first letter of each word
88
- part = part.replace(/\w\S*/g, (w) => {
89
- return w.replace(/^\w/, (c) => c.toUpperCase());
90
- });
91
-
92
- return `<span class="breadcrumb-item">${decodeURI(part)}</span>`;
93
- });
94
- };
95
-
96
- const buildTemplateHelperUriBreadcrumbs = (value: string) => {
97
- const url = new URL(value);
98
-
99
- // exclude youtube links from breadcrumbs
100
- const hostnamePattern = /^((www\.)?(youtube\.com|youtu\.be))$/;
101
-
102
- // we don't want to show atlas docs because the url structure is mad ugly
103
- if (hostnamePattern.test(url.hostname) || url.pathname.includes("atlas.")) {
104
- return "";
105
- }
106
-
107
- let parts = url.pathname.split("/").filter((part) => part !== "");
108
-
109
- // Remove language prefix from trailhead URLs
110
- if (isTrailheadDomain(url.hostname)) {
111
- parts = parts
112
- .slice(1)
113
- .filter((part) => part !== "content" && part !== "learn");
114
- }
115
-
116
- const breadcrumbs = processParts(parts, isInternalDomain(url.hostname));
117
-
118
- if (!isInternalDomain(url.hostname)) {
119
- // Remove the first breadcrumb item if it's an internal domain (i.e drop developer.salesforce.com from developer.salesforce.com / B2C Commerce / Open Commerce API / Filtering)
120
- breadcrumbs.unshift(
121
- `<span class="breadcrumb-item">${url.hostname}</span>`
122
- );
123
-
124
- return `
125
- <span class="breadcrumb">
126
- ${breadcrumbs.join(" / ")}
127
- </span>
128
- `;
129
- } else if (breadcrumbs.length === 1) {
130
- // Hide breadcrumbs if there is only one breadcrumb item
131
- return "";
132
- }
133
-
134
- // remove the last breadcrumb item (the search result title makes it redundant)
135
- breadcrumbs.pop();
136
-
137
- return `<span class="breadcrumb">/ ${breadcrumbs.join(" / ")} /</span>`;
138
- };
139
-
140
- const buildTemplateHelperCommunityBreadcrumbs = () => {
141
- return `<span class="breadcrumb"><span class="breadcrumb-item">trailhead.salesforce.com</span> / <span class="breadcrumb-item">trailblazer-community</span> / <span class="breadcrumb-item">feed</span> / </span>`;
142
- };
143
-
144
- const buildTemplateHelperBreadcrumbs = (value: string) => {
145
- const parts = value.split("/").filter((part) => part !== "");
146
-
147
- // Don't show breadcrumbs if there's only one part
148
- if (parts.length === 1) {
149
- return "";
150
- }
151
-
152
- const breadcrumbs = processParts(parts);
153
-
154
- // remove last breadcrumb item
155
- breadcrumbs.pop();
156
-
157
- return `
158
- <span class="breadcrumb">/ ${breadcrumbs.join(" / ")} /</span>
159
- `;
160
- };
161
-
162
- const buildTemplateHelperMetaBreadcrumbs = (value: string) => {
163
- const parts = value.split("/").filter((part) => part !== "");
164
-
165
- // Don't show breadcrumbs if there's only one part
166
- if (parts.length === 1) {
167
- return "";
168
- }
169
-
170
- const breadcrumbs = processParts(parts);
171
-
172
- return `
173
- <span class="breadcrumb">/ ${breadcrumbs.join(" / ")}</span>
174
- `;
175
- };
176
-
177
- const buildTemplateHelperPostedDate = (value: string) => {
178
- const time = DateTime.fromMillis(Number(value)).toLocaleString(
179
- DateTime.DATE_MED
180
- );
181
-
182
- return `Posted on: ${time} - `;
183
- };
184
-
185
- const buildTemplateHelperReplies = (value: string) => {
186
- const number = Number(value);
187
-
188
- return `<strong>${value} ${number > 1 ? "Replies" : "Reply"}</strong>`;
189
- };
190
-
191
- const buildTemplateHelperLikes = (value: string) => {
192
- const number = Number(value);
193
-
194
- return `<strong>${value} ${number > 1 ? "Likes" : "Like"}</strong>`;
195
- };
196
-
197
- // @ts-ignore Dark Magic (TM) we are overriding the 'title' field with a custom getter. We should really stop doing this.
198
22
  export default class SearchResults extends LightningElement {
199
- @api coveoOrganizationId!: string;
200
- @api coveoPublicAccessToken!: string;
201
- @api coveoSearchPipeline!: string;
202
-
203
- private root: HTMLElement | null = null;
204
- private isInitialized = false;
205
-
206
- @track _query: string = "";
23
+ @track private _query = "";
24
+ @track private results: SearchResultDisplay[] = [];
25
+ @track private isLoading = false;
26
+ @track private hasSearched = false;
27
+ @track private didTrackThisSearch = false;
207
28
 
208
29
  @api
209
30
  get searchQuery() {
210
31
  return this._query;
211
32
  }
212
33
  set searchQuery(q: string) {
213
- this._query = q;
214
- if (this.isInitialized && this._query !== "") {
215
- this.updateSearchQuery();
34
+ const next = q ?? "";
35
+ if (next !== this._query) {
36
+ this._query = next;
37
+ this.syncUrlToQuery();
38
+ if (this._query.trim()) {
39
+ this.runSearch();
40
+ } else {
41
+ this.results = [];
42
+ this.hasSearched = false;
43
+ }
216
44
  }
217
45
  }
218
46
 
219
- private totalResults: number | null = null;
47
+ private get query(): string {
48
+ return this._query;
49
+ }
220
50
 
221
- private get title() {
222
- return this.totalResults ? this.totalResults.toLocaleString() : "0";
51
+ private get totalResults(): number {
52
+ return this.results?.length ?? 0;
223
53
  }
224
54
 
225
- private query: string = "";
226
- private hasFilters: boolean = false;
55
+ private get resultCountLabel(): string {
56
+ return this.totalResults.toLocaleString();
57
+ }
227
58
 
228
59
  private get hasQuery(): boolean {
229
60
  return this.query !== "";
230
61
  }
231
62
 
232
- private get coveoAnalyticsEndpoint() {
233
- return `https://${this.coveoOrganizationId}.analytics.org.coveo.com/rest/ua`;
234
- }
235
-
236
- private updateSearchQuery() {
237
- Coveo.state(this.root!, "q", this._query);
63
+ private get hasResults(): boolean {
64
+ return this.results.length > 0;
238
65
  }
239
66
 
240
- private clearFilters() {
241
- const BreadcrumbManager = Coveo.get(
242
- this.root!.querySelector(".CoveoBreadcrumb") as HTMLElement
243
- ) as any;
244
- BreadcrumbManager.clearBreadcrumbs();
67
+ private get showNoResults(): boolean {
68
+ return this.hasSearched && !this.isLoading && !this.hasResults;
245
69
  }
246
70
 
247
- private currentPage: number = 25;
248
- private totalPages: number = 1;
249
-
250
- private originalBreadcrumbs: string[] = [];
251
- private initialWindowWidth = window.innerWidth;
252
-
253
- private goToPage(e: CustomEvent) {
254
- const page = e.detail;
255
- const Pager = Coveo.get(
256
- this.root!.querySelector(".CoveoPager") as HTMLElement
257
- ) as any;
258
- Pager.setPage(page);
259
- this.currentPage = page;
71
+ private syncUrlToQuery() {
72
+ const url = new URL(window.location.href);
73
+ if (this._query.trim()) {
74
+ url.searchParams.set("q", this._query.trim());
75
+ } else {
76
+ url.searchParams.delete("q");
77
+ }
78
+ const newHref = url.pathname + url.search + url.hash;
79
+ if (
80
+ window.location.pathname +
81
+ window.location.search +
82
+ window.location.hash !==
83
+ newHref
84
+ ) {
85
+ window.history.replaceState(null, "", newHref);
86
+ }
260
87
  }
261
88
 
262
- private dismissFiltersOverlay = () => {
263
- const overlay = this.root!.querySelector(
264
- ".coveo-dropdown-background-active"
265
- ) as HTMLElement | null;
266
- overlay?.click();
267
- };
268
-
269
- private attachListeners(root: HTMLElement) {
270
- Coveo.$$(root).on(
271
- Coveo.InitializationEvents.afterInitialization,
272
- () => {
273
- const url = new URL(window.location.href);
274
- const searchParams = url.searchParams;
275
- const keywordsParam = searchParams.get("keywords");
276
-
277
- // Accomodate for the global nav using 'keywords' param instead of 'q'
278
- if (keywordsParam) {
279
- this._query = searchParams.get("keywords")!;
280
- searchParams.delete("keywords");
281
- window.history.replaceState(null, "", url.href);
282
- }
283
-
284
- if (this._query !== "") {
285
- Coveo.state(this.root!, "q", this.searchQuery);
286
- }
287
- if (Coveo.state(this.root!, "q") === "") {
288
- Coveo.state(this.root!, "sort", "date descending");
289
- }
290
-
291
- this.isInitialized = true;
292
- }
293
- );
294
-
295
- Coveo.$$(root).on(Coveo.QueryEvents.querySuccess, (event: any) => {
296
- const { currentPage, numberOfPages } = getPaginationState(
297
- event.detail
89
+ private async doFetch() {
90
+ const query = this._query.trim();
91
+ if (!query) {
92
+ this.results = [];
93
+ this.isLoading = false;
94
+ return;
95
+ }
96
+ this.isLoading = true;
97
+ this.hasSearched = true;
98
+ this.didTrackThisSearch = false;
99
+ try {
100
+ const raw = await fetchSearch(query);
101
+ this.results = raw.map(
102
+ (item: Data360SearchResultItem, index: number) =>
103
+ this.normalizeResult(item, index)
298
104
  );
299
- this.currentPage = currentPage;
300
- this.totalPages = numberOfPages;
301
- this.totalResults = event.detail.results.totalCount;
105
+ this.trackSearchResultsOnce(query, this.results.length);
106
+ } catch (err) {
107
+ console.error("Data 360 Search request failed", err);
108
+ this.results = [];
109
+ } finally {
110
+ this.isLoading = false;
111
+ }
112
+ }
302
113
 
303
- this.query = event.detail.query.q ?? "";
304
- this.hasFilters = event.detail.query?.facets?.some((f: any) => {
305
- return f.currentValues.some((cv: any) => {
306
- return cv.state === "selected";
307
- });
114
+ private runSearch = debounce(() => {
115
+ this.doFetch();
116
+ }, SEARCH_DEBOUNCE_DELAY);
117
+
118
+ private normalizeResult(
119
+ item: Data360SearchResultItem,
120
+ index: number
121
+ ): SearchResultDisplay {
122
+ const href = item.url ?? "";
123
+ const isExternal =
124
+ href.startsWith("http") &&
125
+ !href.includes("developer.salesforce.com") &&
126
+ !href.includes("developer-website-s.herokuapp.com");
127
+ return {
128
+ title: item.title ?? "",
129
+ href,
130
+ matchedText: item.matchedText ?? "",
131
+ uniqueId: href || `result-${index}`,
132
+ openInNewTab: isExternal ? "_blank" : undefined,
133
+ rel: isExternal ? "noopener noreferrer" : undefined
134
+ };
135
+ }
136
+
137
+ private trackSearchResultsOnce(term: string, resultCount: number) {
138
+ if (this.didTrackThisSearch) {
139
+ return;
140
+ }
141
+ this.didTrackThisSearch = true;
142
+ if (document.readyState === "complete") {
143
+ trackGTM(this.template.host, "custEv_search", {
144
+ search_term: term,
145
+ search_category: "",
146
+ search_type: "site search",
147
+ search_result_count: resultCount
308
148
  });
309
- // Note that this logic means that if someone clicks a search result before
310
- // onetrust has loaded, and thus navigates away from the page, we will lose the tracking
311
- if (document.readyState === "complete") {
312
- this.trackSearchResults(event, this.query, this.totalResults);
313
- } else {
314
- const query = this.query;
315
- const totalResults = this.totalResults;
316
- window.addEventListener("load", () => {
317
- this.trackSearchResults(event, query, totalResults);
318
- });
319
- }
320
- });
321
-
322
- Coveo.$$(root).on(Coveo.QueryEvents.deferredQuerySuccess, async () => {
323
- // wait specified time to ensure breadcrumbs are rendered before processing them
324
- await pollUntil(
149
+ } else {
150
+ window.addEventListener(
151
+ "load",
325
152
  () => {
326
- const coveoResults =
327
- this.root!.querySelector(".CoveoResult");
328
-
329
- return Boolean(coveoResults);
153
+ trackGTM(this.template.host, "custEv_search", {
154
+ search_term: term,
155
+ search_category: "",
156
+ search_type: "site search",
157
+ search_result_count: resultCount
158
+ });
330
159
  },
331
- 20,
332
- 1000
333
- );
334
-
335
- this.processBreadcrumbs(root);
336
-
337
- window.onresize = () => this.processBreadcrumbs(root);
338
- });
339
- }
340
-
341
- // check if the breadcrumb is overflowing or not based on the height of a single character in the text element
342
- private isTextWrapping = (element: HTMLElement) => {
343
- return element.offsetHeight > MAX_BREADCRUMB_HEIGHT;
344
- };
345
-
346
- private truncateBreadcrumbText = (breadcrumbItems: HTMLElement[]) => {
347
- breadcrumbItems.forEach((breadcrumbItem: HTMLElement) => {
348
- const breadcrumbItemText = breadcrumbItem.textContent!;
349
- if (breadcrumbItemText.length > 30) {
350
- breadcrumbItem.textContent = `${breadcrumbItemText.substring(
351
- 0,
352
- 30
353
- )}...`;
354
- }
355
- });
356
- };
357
-
358
- private addBreadcrumbEllipsis = (
359
- breadcrumbItems: HTMLElement[],
360
- breadcrumb: HTMLElement
361
- ) => {
362
- for (let i = 1; i < breadcrumbItems.length; i++) {
363
- if (this.isTextWrapping(breadcrumb)) {
364
- // if the previous element is an ellipsis, make it empty (in order to avoid multiple grouped ellipsis)
365
- if (breadcrumbItems[i - 1]?.textContent === "...") {
366
- breadcrumbItems[i].innerHTML = "";
367
- } else {
368
- breadcrumbItems[i].textContent = "...";
369
- }
370
- } else {
371
- break; // Exit the loop if the breadcrumb is no longer overflowing
372
- }
373
- }
374
-
375
- // remove any empty breadcrumb items
376
- breadcrumb.innerHTML = breadcrumb.innerHTML
377
- .replace(
378
- / ?\/ +<span class="breadcrumb-item"><\/span> +\/ ?/g,
379
- " / "
380
- )
381
- // when first loading the page on mobile, the breadcrumb items are not grouped correctly
382
- .replace(
383
- `<span class="breadcrumb-item">...</span> / <span class="breadcrumb-item">...</span>`,
384
- `<span class="breadcrumb-item">...</span>`
385
- );
386
- };
387
-
388
- private formatBreadcrumbs = (breadcrumbs: HTMLElement[]) => {
389
- breadcrumbs?.forEach((breadcrumb: HTMLElement) => {
390
- // Get all breadcrumb items that are separated by '/'
391
- const breadcrumbItems: any =
392
- breadcrumb.querySelectorAll(".breadcrumb-item");
393
-
394
- // Check if the breadcrumb is overflowing by comparing it's height to the height of the first breadcrumb item
395
- if (this.isTextWrapping(breadcrumb)) {
396
- // it is overflowing, so we need to truncate long titles to 30 characters
397
- this.truncateBreadcrumbText(breadcrumbItems);
398
-
399
- // Iteratively check if the breadcrumb is still overflowing and replace text with '...' starting from the second breadcrumb item
400
- this.addBreadcrumbEllipsis(breadcrumbItems, breadcrumb);
401
-
402
- // After processing all breadcrumb items, if it's still overflowing, hide the breadcrumb element
403
- if (this.isTextWrapping(breadcrumb)) {
404
- breadcrumb.style.display = "none";
405
- }
406
- }
407
- });
408
- };
409
-
410
- private restoreBreadcrumbs = (breadcrumbs: HTMLElement[]) => {
411
- breadcrumbs.forEach((breadcrumb: HTMLElement, index: number) => {
412
- // eslint-disable-next-line @lwc/lwc/no-inner-html
413
- breadcrumb.innerHTML = this.originalBreadcrumbs[index];
414
- });
415
- };
416
-
417
- private windowSizeIncreased = () =>
418
- window.innerWidth > this.initialWindowWidth;
419
-
420
- private processBreadcrumbs(root: HTMLElement) {
421
- // Get all breadcrumbs from search results
422
- const breadcrumbs = Array.from(
423
- root.querySelectorAll(".breadcrumb")
424
- ) as HTMLElement[];
425
-
426
- if (this.originalBreadcrumbs.length === 0) {
427
- this.originalBreadcrumbs = breadcrumbs.map(
428
- (breadcrumb) => breadcrumb.innerHTML
160
+ { once: true }
429
161
  );
430
162
  }
163
+ }
431
164
 
432
- if (this.windowSizeIncreased()) {
433
- /*
434
- Reset the breadcrumbs to their original state and process them again.
435
- The additional space means we can replace ellipsis with full text.
436
- */
437
- this.restoreBreadcrumbs(breadcrumbs);
165
+ private onSearchInputChange(e: CustomEvent) {
166
+ const value = (e.detail as string) ?? "";
167
+ this._query = value;
168
+ this.syncUrlToQuery();
169
+ if (value.trim()) {
170
+ this.runSearch();
171
+ } else {
172
+ this.results = [];
173
+ this.hasSearched = false;
438
174
  }
439
-
440
- this.formatBreadcrumbs(breadcrumbs);
441
175
  }
442
176
 
443
- private initializeCoveo() {
444
- this.root = this.template.querySelector(".CoveoSearchInterface")!;
177
+ private hasRunInitialSearch = false;
445
178
 
446
- const resultsList = this.template.querySelector(".CoveoResultList");
447
-
448
- if (resultsList) {
449
- // eslint-disable-next-line @lwc/lwc/no-inner-html
450
- resultsList.innerHTML = resultsTemplate;
179
+ connectedCallback() {
180
+ const q = getQueryFromUrl();
181
+ if (q) {
182
+ this._query = q;
183
+ this.hasRunInitialSearch = true;
184
+ this.doFetch();
451
185
  }
452
-
453
- Coveo.SearchEndpoint.configureCloudV2Endpoint(
454
- this.coveoOrganizationId,
455
- this.coveoPublicAccessToken,
456
- `https://${this.coveoOrganizationId}.org.coveo.com/rest/search`
457
- );
458
-
459
- this.attachListeners(this.root);
460
-
461
- Coveo.TemplateHelpers.registerTemplateHelper(
462
- "badge",
463
- buildTemplateHelperBadge
464
- );
465
-
466
- Coveo.TemplateHelpers.registerTemplateHelper(
467
- "breadcrumbs",
468
- buildTemplateHelperBreadcrumbs
469
- );
470
-
471
- Coveo.TemplateHelpers.registerTemplateHelper(
472
- "metabreadcrumbs",
473
- buildTemplateHelperMetaBreadcrumbs
474
- );
475
-
476
- Coveo.TemplateHelpers.registerTemplateHelper(
477
- "uriBreadcrumbs",
478
- buildTemplateHelperUriBreadcrumbs
479
- );
480
-
481
- Coveo.TemplateHelpers.registerTemplateHelper(
482
- "communityBreadcrumbs",
483
- buildTemplateHelperCommunityBreadcrumbs
484
- );
485
-
486
- Coveo.TemplateHelpers.registerTemplateHelper(
487
- "postedDate",
488
- buildTemplateHelperPostedDate
489
- );
490
-
491
- Coveo.TemplateHelpers.registerTemplateHelper(
492
- "replies",
493
- buildTemplateHelperReplies
494
- );
495
-
496
- Coveo.TemplateHelpers.registerTemplateHelper(
497
- "likes",
498
- buildTemplateHelperLikes
499
- );
500
-
501
- Coveo.init(this.root);
502
186
  }
503
187
 
504
188
  renderedCallback() {
505
- if (!this.isInitialized) {
506
- if (Object.prototype.hasOwnProperty.call(window, "Coveo")) {
507
- this.initializeCoveo();
508
- } else {
509
- // eslint-disable-next-line @lwc/lwc/no-document-query
510
- const script = document.querySelector("script.coveo-script");
511
- script?.addEventListener("load", () => {
512
- this.initializeCoveo();
513
- });
514
- }
189
+ // Run search on first render if URL has q and we haven't yet (handles #q=, SPA nav, or late URL)
190
+ if (this.hasRunInitialSearch) {
191
+ return;
192
+ }
193
+ const q = getQueryFromUrl();
194
+ if (!q) {
195
+ return;
196
+ }
197
+ this.hasRunInitialSearch = true;
198
+ if (this._query !== q) {
199
+ this._query = q;
200
+ }
201
+ if (!this.hasSearched && !this.isLoading) {
202
+ this.doFetch();
515
203
  }
516
- }
517
-
518
- private trackSearchResults(event: Event, term: string, resultCount: any) {
519
- trackGTM(event.target!, "custEv_search", {
520
- search_term: term,
521
- search_category: "",
522
- search_type: "site search",
523
- search_result_count: resultCount
524
- });
525
204
  }
526
205
  }