@internetarchive/collection-browser 4.1.0 → 4.2.0-alpha-webdev8164.0

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.
Files changed (62) hide show
  1. package/.editorconfig +29 -29
  2. package/.github/workflows/ci.yml +27 -27
  3. package/.github/workflows/gh-pages-main.yml +39 -39
  4. package/.github/workflows/npm-publish.yml +39 -39
  5. package/.github/workflows/pr-preview.yml +38 -38
  6. package/.husky/pre-commit +1 -1
  7. package/.prettierignore +1 -1
  8. package/LICENSE +661 -661
  9. package/README.md +83 -83
  10. package/dist/src/collection-browser.js +761 -761
  11. package/dist/src/collection-browser.js.map +1 -1
  12. package/dist/src/collection-facets/facets-template.js +5 -0
  13. package/dist/src/collection-facets/facets-template.js.map +1 -1
  14. package/dist/src/collection-facets/more-facets-content.d.ts +92 -8
  15. package/dist/src/collection-facets/more-facets-content.js +526 -84
  16. package/dist/src/collection-facets/more-facets-content.js.map +1 -1
  17. package/dist/src/collection-facets/more-facets-pagination.d.ts +12 -3
  18. package/dist/src/collection-facets/more-facets-pagination.js +69 -8
  19. package/dist/src/collection-facets/more-facets-pagination.js.map +1 -1
  20. package/dist/src/collection-facets/toggle-switch.js +1 -0
  21. package/dist/src/collection-facets/toggle-switch.js.map +1 -1
  22. package/dist/src/data-source/collection-browser-data-source.js.map +1 -1
  23. package/dist/src/data-source/collection-browser-query-state.js.map +1 -1
  24. package/dist/src/sort-filter-bar/sort-filter-bar.js +280 -280
  25. package/dist/src/sort-filter-bar/sort-filter-bar.js.map +1 -1
  26. package/dist/test/collection-browser.test.js +189 -189
  27. package/dist/test/collection-browser.test.js.map +1 -1
  28. package/dist/test/collection-facets/more-facets-content.test.js +162 -3
  29. package/dist/test/collection-facets/more-facets-content.test.js.map +1 -1
  30. package/dist/test/collection-facets/more-facets-pagination.test.js +63 -3
  31. package/dist/test/collection-facets/more-facets-pagination.test.js.map +1 -1
  32. package/dist/test/mocks/mock-search-responses.d.ts +5 -0
  33. package/dist/test/mocks/mock-search-responses.js +44 -0
  34. package/dist/test/mocks/mock-search-responses.js.map +1 -1
  35. package/dist/test/mocks/mock-search-service.js +2 -1
  36. package/dist/test/mocks/mock-search-service.js.map +1 -1
  37. package/dist/test/sort-filter-bar/sort-filter-bar.test.js +22 -22
  38. package/dist/test/sort-filter-bar/sort-filter-bar.test.js.map +1 -1
  39. package/eslint.config.mjs +53 -53
  40. package/index.html +24 -24
  41. package/local.archive.org.cert +86 -86
  42. package/local.archive.org.key +27 -27
  43. package/package.json +121 -120
  44. package/renovate.json +6 -6
  45. package/src/collection-browser.ts +3070 -3070
  46. package/src/collection-facets/facets-template.ts +5 -0
  47. package/src/collection-facets/more-facets-content.ts +566 -96
  48. package/src/collection-facets/more-facets-pagination.ts +80 -9
  49. package/src/collection-facets/toggle-switch.ts +1 -0
  50. package/src/data-source/collection-browser-data-source.ts +1444 -1444
  51. package/src/data-source/collection-browser-query-state.ts +60 -60
  52. package/src/sort-filter-bar/sort-filter-bar.ts +733 -733
  53. package/test/collection-browser.test.ts +2402 -2402
  54. package/test/collection-facets/more-facets-content.test.ts +251 -4
  55. package/test/collection-facets/more-facets-pagination.test.ts +87 -3
  56. package/test/mocks/mock-search-responses.ts +48 -0
  57. package/test/mocks/mock-search-service.ts +2 -0
  58. package/test/sort-filter-bar/sort-filter-bar.test.ts +443 -443
  59. package/tsconfig.json +25 -25
  60. package/web-dev-server.config.mjs +30 -30
  61. package/web-test-runner.config.mjs +52 -52
  62. package/.claude/settings.local.json +0 -8
@@ -1,2402 +1,2402 @@
1
- import { aTimeout, expect, fixture } from '@open-wc/testing';
2
- import { html } from 'lit';
3
- import sinon from 'sinon';
4
- import type { InfiniteScroller } from '@internetarchive/infinite-scroller';
5
- import { FilterConstraint, SearchType } from '@internetarchive/search-service';
6
- import type { HistogramDateRange } from '@internetarchive/histogram-date-range';
7
- import type { CollectionBrowser } from '../src/collection-browser';
8
- import '../src/collection-browser';
9
- import {
10
- getDefaultSelectedFacets,
11
- FacetBucket,
12
- SelectedFacets,
13
- SortField,
14
- } from '../src/models';
15
- import { MockSearchService } from './mocks/mock-search-service';
16
- import { MockAnalyticsHandler } from './mocks/mock-analytics-handler';
17
- import {
18
- analyticsActions,
19
- analyticsCategories,
20
- } from '../src/utils/analytics-events';
21
- import type { TileDispatcher } from '../src/tiles/tile-dispatcher';
22
- import type { CollectionFacets } from '../src/collection-facets';
23
- import type { EmptyPlaceholder } from '../src/empty-placeholder';
24
- import type { SortFilterBar } from '../src/sort-filter-bar/sort-filter-bar';
25
-
26
- /**
27
- * Wait for the next tick of the event loop.
28
- *
29
- * This is necessary in some of the tests because certain collection browser
30
- * updates take more than one tick to render (e.g., date picker & query changes).
31
- * These delays are non-ideal and should eventually be investigated and fixed,
32
- * but they are minor enough that waiting for the next tick is a reasonable
33
- * testing solution for now.
34
- */
35
- const nextTick = () => aTimeout(0);
36
-
37
- describe('Collection Browser', () => {
38
- beforeEach(async () => {
39
- // Apparently query params set by one test can bleed into other tests.
40
- // Since collection browser restores its state from certain query params, we need
41
- // to clear these before each test to ensure they run in isolation from one another.
42
- const url = new URL(window.location.href);
43
- const { searchParams } = url;
44
- searchParams.delete('sin');
45
- searchParams.delete('sort');
46
- searchParams.delete('query');
47
- searchParams.delete('page');
48
- searchParams.delete('and[]');
49
- searchParams.delete('not[]');
50
- window.history.replaceState({}, '', url);
51
- });
52
-
53
- it('clears selected facets when requested', async () => {
54
- const selectedFacets = getDefaultSelectedFacets();
55
- selectedFacets.creator.foo = { count: 1, key: 'foo', state: 'selected' };
56
- const el = await fixture<CollectionBrowser>(
57
- html`<collection-browser></collection-browser>`,
58
- );
59
-
60
- el.selectedFacets = selectedFacets;
61
- await el.updateComplete;
62
- el.clearFilters(); // By default, sort is not cleared
63
-
64
- expect(el.selectedFacets).to.deep.equal(getDefaultSelectedFacets());
65
- });
66
-
67
- it('clears existing filters but not sort by default', async () => {
68
- const el = await fixture<CollectionBrowser>(
69
- html`<collection-browser></collection-browser>`,
70
- );
71
-
72
- el.selectedSort = 'title' as SortField;
73
- el.sortDirection = 'asc';
74
- await el.updateComplete;
75
- el.clearFilters(); // By default, sort is not cleared
76
-
77
- expect(el.selectedFacets).to.deep.equal(getDefaultSelectedFacets());
78
- expect(el.selectedSort).to.equal('title');
79
- expect(el.sortDirection).to.equal('asc');
80
- expect(el.sortParam).to.deep.equal({
81
- field: 'titleSorter',
82
- direction: 'asc',
83
- });
84
- expect(el.selectedCreatorFilter).to.be.null;
85
- expect(el.selectedTitleFilter).to.be.null;
86
- });
87
-
88
- it('clears existing filters for facets & sort via option', async () => {
89
- const el = await fixture<CollectionBrowser>(
90
- html`<collection-browser></collection-browser>`,
91
- );
92
-
93
- el.selectedSort = 'title' as SortField;
94
- await el.updateComplete;
95
- el.clearFilters({ sort: true }); // Sort is reset too due to the option
96
-
97
- expect(el.selectedFacets).to.deep.equal(getDefaultSelectedFacets());
98
- expect(el.selectedSort).to.equal(SortField.default);
99
- expect(el.sortDirection).to.be.null;
100
- expect(el.sortParam).to.be.null;
101
- expect(el.selectedCreatorFilter).to.be.null;
102
- expect(el.selectedTitleFilter).to.be.null;
103
- });
104
-
105
- it('filterBy creator with analytics', async () => {
106
- const mockAnalyticsHandler = new MockAnalyticsHandler();
107
- const el = await fixture<CollectionBrowser>(
108
- html`<collection-browser .analyticsHandler=${mockAnalyticsHandler}>
109
- </collection-browser>`,
110
- );
111
-
112
- el.searchContext = 'betaSearchService';
113
- el.selectedSort = 'creator' as SortField;
114
- el.sortDirection = 'asc';
115
- el.selectedCreatorFilter = 'A';
116
- await el.updateComplete;
117
-
118
- expect(mockAnalyticsHandler.callCategory).to.equal('betaSearchService');
119
- expect(mockAnalyticsHandler.callAction).to.equal('filterByCreator');
120
- expect(mockAnalyticsHandler.callLabel).to.equal('start-A');
121
-
122
- el.clearFilters();
123
- await el.updateComplete;
124
-
125
- expect(el.selectedTitleFilter).to.be.null;
126
- expect(mockAnalyticsHandler.callCategory).to.equal('betaSearchService');
127
- expect(mockAnalyticsHandler.callAction).to.equal('filterByCreator');
128
- expect(mockAnalyticsHandler.callLabel).to.equal('clear-A');
129
- });
130
-
131
- it('filterBy title with analytics', async () => {
132
- const mockAnalyticsHandler = new MockAnalyticsHandler();
133
- const el = await fixture<CollectionBrowser>(
134
- html`<collection-browser .analyticsHandler=${mockAnalyticsHandler}>
135
- </collection-browser>`,
136
- );
137
-
138
- el.searchContext = 'beta-search-service';
139
- el.selectedSort = 'title' as SortField;
140
- el.sortDirection = 'asc';
141
- el.selectedTitleFilter = 'A';
142
- await el.updateComplete;
143
-
144
- expect(mockAnalyticsHandler.callCategory).to.equal('beta-search-service');
145
- expect(mockAnalyticsHandler.callAction).to.equal('filterByTitle');
146
- expect(mockAnalyticsHandler.callLabel).to.equal('start-A');
147
-
148
- el.clearFilters();
149
- await el.updateComplete;
150
-
151
- expect(el.selectedTitleFilter).to.be.null;
152
- expect(mockAnalyticsHandler.callCategory).to.equal('beta-search-service');
153
- expect(mockAnalyticsHandler.callAction).to.equal('filterByTitle');
154
- expect(mockAnalyticsHandler.callLabel).to.equal('clear-A');
155
- });
156
-
157
- it('selected facets with analytics - not negative facets', async () => {
158
- const mockAnalyticsHandler = new MockAnalyticsHandler();
159
- const mediaTypeBucket = { count: 123, state: 'selected' } as FacetBucket;
160
- const mockedSelectedFacets: SelectedFacets = {
161
- subject: {},
162
- lending: {},
163
- mediatype: { data: mediaTypeBucket },
164
- language: {},
165
- creator: {},
166
- collection: {},
167
- year: {},
168
- };
169
-
170
- const el = await fixture<CollectionBrowser>(
171
- html`<collection-browser .analyticsHandler=${mockAnalyticsHandler}>
172
- </collection-browser>`,
173
- );
174
-
175
- el.searchContext = 'search-service';
176
- el.selectedFacets = mockedSelectedFacets;
177
- await el.updateComplete;
178
-
179
- el.facetClickHandler(
180
- new CustomEvent('facetClick', {
181
- detail: {
182
- facetType: 'mediatype',
183
- bucket: {
184
- key: '',
185
- state: 'selected',
186
- count: 123,
187
- },
188
- negative: false,
189
- },
190
- }),
191
- );
192
- expect(mockAnalyticsHandler.callCategory).to.equal('search-service');
193
- expect(mockAnalyticsHandler.callAction).to.equal('facetSelected');
194
- expect(mockAnalyticsHandler.callLabel).to.equal('mediatype');
195
-
196
- el.facetClickHandler(
197
- new CustomEvent('facetClick', {
198
- detail: {
199
- facetType: 'mediatype',
200
- bucket: {
201
- key: '',
202
- state: 'none',
203
- count: 123,
204
- },
205
- negative: false,
206
- },
207
- }),
208
- );
209
- expect(el.selectedFacets).to.equal(mockedSelectedFacets);
210
- expect(mockAnalyticsHandler.callCategory).to.equal('search-service');
211
- expect(mockAnalyticsHandler.callAction).to.equal('facetDeselected');
212
- expect(mockAnalyticsHandler.callLabel).to.equal('mediatype');
213
- });
214
-
215
- it('selected facets with analytics - negative facets', async () => {
216
- const mockAnalyticsHandler = new MockAnalyticsHandler();
217
- const mediaTypeBucket = { count: 123, state: 'selected' } as FacetBucket;
218
- const mockedSelectedFacets: SelectedFacets = {
219
- subject: {},
220
- lending: {},
221
- mediatype: { data: mediaTypeBucket },
222
- language: {},
223
- creator: {},
224
- collection: {},
225
- year: {},
226
- };
227
-
228
- const el = await fixture<CollectionBrowser>(
229
- html`<collection-browser .analyticsHandler=${mockAnalyticsHandler}>
230
- </collection-browser>`,
231
- );
232
-
233
- el.searchContext = 'beta-search-service';
234
- el.selectedFacets = mockedSelectedFacets;
235
- await el.updateComplete;
236
-
237
- el.facetClickHandler(
238
- new CustomEvent('facetClick', {
239
- detail: {
240
- facetType: 'mediatype',
241
- bucket: {
242
- key: '',
243
- state: 'hidden',
244
- count: 123,
245
- },
246
- negative: true,
247
- },
248
- }),
249
- );
250
- expect(mockAnalyticsHandler.callCategory).to.equal('beta-search-service');
251
- expect(mockAnalyticsHandler.callAction).to.equal('facetNegativeSelected');
252
- expect(mockAnalyticsHandler.callLabel).to.equal('mediatype');
253
-
254
- el.facetClickHandler(
255
- new CustomEvent('facetClick', {
256
- detail: {
257
- facetType: 'mediatype',
258
- bucket: {
259
- key: '',
260
- state: 'none',
261
- count: 123,
262
- },
263
- negative: true,
264
- },
265
- }),
266
- );
267
- expect(el.selectedFacets).to.equal(mockedSelectedFacets);
268
- expect(mockAnalyticsHandler.callCategory).to.equal('beta-search-service');
269
- expect(mockAnalyticsHandler.callAction).to.equal('facetNegativeDeselected');
270
- expect(mockAnalyticsHandler.callLabel).to.equal('mediatype');
271
- });
272
-
273
- it('should render with a sort bar, facets, and infinite scroller', async () => {
274
- const searchService = new MockSearchService();
275
-
276
- const el = await fixture<CollectionBrowser>(
277
- html`<collection-browser .searchService=${searchService}>
278
- </collection-browser>`,
279
- );
280
-
281
- el.baseQuery = 'hello';
282
- await el.updateComplete;
283
- await nextTick();
284
-
285
- const facets = el.shadowRoot?.querySelector('collection-facets');
286
- const sortBar = el.shadowRoot?.querySelector('sort-filter-bar');
287
- const infiniteScroller = el.shadowRoot?.querySelector('infinite-scroller');
288
- expect(facets, 'facets').to.exist;
289
- expect(sortBar, 'sort bar').to.exist;
290
- expect(infiniteScroller, 'infinite scroller').to.exist;
291
- });
292
-
293
- it('queries the search service when given a base query', async () => {
294
- const searchService = new MockSearchService();
295
-
296
- const el = await fixture<CollectionBrowser>(
297
- html`<collection-browser .searchService=${searchService}>
298
- </collection-browser>`,
299
- );
300
-
301
- el.baseQuery = 'collection:foo';
302
- await el.updateComplete;
303
- await el.initialSearchComplete;
304
-
305
- expect(searchService.searchParams?.query).to.equal('collection:foo');
306
- expect(
307
- el.shadowRoot?.querySelector('#big-results-label')?.textContent,
308
- ).to.contains('Results');
309
- });
310
-
311
- it('queries the search service when given a list of identifiers and no query', async () => {
312
- const searchService = new MockSearchService();
313
-
314
- const el = await fixture<CollectionBrowser>(
315
- html`<collection-browser .searchService=${searchService}>
316
- </collection-browser>`,
317
- );
318
-
319
- el.identifiers = ['foo', 'bar'];
320
- await el.updateComplete;
321
- await el.initialSearchComplete;
322
-
323
- expect(searchService.searchParams?.identifiers).to.deep.equal([
324
- 'foo',
325
- 'bar',
326
- ]);
327
- expect(
328
- el.shadowRoot?.querySelector('#big-results-label')?.textContent,
329
- ).to.contains('Results');
330
- });
331
-
332
- it('queries the search service when given a list of identifiers with a query', async () => {
333
- const searchService = new MockSearchService();
334
-
335
- const el = await fixture<CollectionBrowser>(
336
- html`<collection-browser .searchService=${searchService}>
337
- </collection-browser>`,
338
- );
339
-
340
- el.baseQuery = 'collection:foo';
341
- el.identifiers = ['foo', 'bar'];
342
- await el.updateComplete;
343
- await el.initialSearchComplete;
344
-
345
- expect(searchService.searchParams?.query).to.equal('collection:foo');
346
- expect(searchService.searchParams?.identifiers).to.deep.equal([
347
- 'foo',
348
- 'bar',
349
- ]);
350
- expect(
351
- el.shadowRoot?.querySelector('#big-results-label')?.textContent,
352
- ).to.contains('Results');
353
- });
354
-
355
- it('queries the search service with a metadata search', async () => {
356
- const searchService = new MockSearchService();
357
-
358
- const el = await fixture<CollectionBrowser>(
359
- html` <collection-browser .searchService=${searchService}>
360
- </collection-browser>`,
361
- );
362
-
363
- el.searchType = SearchType.METADATA;
364
- await el.updateComplete;
365
-
366
- el.baseQuery = 'collection:foo';
367
- await el.updateComplete;
368
- await el.initialSearchComplete;
369
-
370
- expect(searchService.searchParams?.query).to.equal('collection:foo');
371
- expect(searchService.searchType).to.equal(SearchType.METADATA);
372
- expect(
373
- el.shadowRoot?.querySelector('#big-results-label')?.textContent,
374
- ).to.contains('Results');
375
- });
376
-
377
- it('can change search type', async () => {
378
- const searchService = new MockSearchService();
379
- const el = await fixture<CollectionBrowser>(
380
- html`<collection-browser .searchService=${searchService}>
381
- </collection-browser>`,
382
- );
383
-
384
- el.baseQuery = 'collection:foo';
385
- await el.updateComplete;
386
-
387
- el.searchType = SearchType.FULLTEXT;
388
- await el.updateComplete;
389
- await el.initialSearchComplete;
390
-
391
- expect(searchService.searchType).to.equal(SearchType.FULLTEXT);
392
- });
393
-
394
- it('queries the search service with a fulltext search', async () => {
395
- const searchService = new MockSearchService();
396
-
397
- const el = await fixture<CollectionBrowser>(
398
- html` <collection-browser .searchService=${searchService}>
399
- </collection-browser>`,
400
- );
401
-
402
- el.searchType = SearchType.FULLTEXT;
403
- await el.updateComplete;
404
-
405
- el.baseQuery = 'collection:foo';
406
- await el.updateComplete;
407
- await el.initialSearchComplete;
408
-
409
- expect(searchService.searchParams?.query).to.equal('collection:foo');
410
- expect(searchService.searchType).to.equal(SearchType.FULLTEXT);
411
- expect(
412
- el.shadowRoot?.querySelector('#big-results-label')?.textContent,
413
- ).to.contains('Results');
414
- });
415
-
416
- it('queries the search service with a radio search', async () => {
417
- const searchService = new MockSearchService();
418
-
419
- const el = await fixture<CollectionBrowser>(
420
- html` <collection-browser .searchService=${searchService}>
421
- </collection-browser>`,
422
- );
423
-
424
- el.searchType = SearchType.RADIO;
425
- await el.updateComplete;
426
-
427
- el.baseQuery = 'collection:foo';
428
- await el.updateComplete;
429
- await el.initialSearchComplete;
430
-
431
- expect(searchService.searchParams?.query).to.equal('collection:foo');
432
- expect(searchService.searchType).to.equal(SearchType.RADIO);
433
- expect(
434
- el.shadowRoot?.querySelector('#big-results-label')?.textContent,
435
- ).to.contains('Results');
436
- });
437
-
438
- it('queries the search service with a TV search', async () => {
439
- const searchService = new MockSearchService();
440
-
441
- const el = await fixture<CollectionBrowser>(
442
- html` <collection-browser .searchService=${searchService}>
443
- </collection-browser>`,
444
- );
445
-
446
- el.searchType = SearchType.TV;
447
- await el.updateComplete;
448
-
449
- el.baseQuery = 'collection:foo';
450
- await el.updateComplete;
451
- await el.initialSearchComplete;
452
-
453
- expect(searchService.searchParams?.query).to.equal('collection:foo');
454
- expect(searchService.searchType).to.equal(SearchType.TV);
455
- expect(
456
- el.shadowRoot?.querySelector('#big-results-label')?.textContent,
457
- ).to.contains('Results');
458
- });
459
-
460
- it('queries the search service with facets selected/negated', async () => {
461
- const searchService = new MockSearchService();
462
- const selectedFacets: SelectedFacets = {
463
- subject: {
464
- foo: {
465
- key: 'foo',
466
- count: 1,
467
- state: 'selected',
468
- },
469
- bar: {
470
- key: 'bar',
471
- count: 2,
472
- state: 'hidden',
473
- },
474
- },
475
- lending: {},
476
- mediatype: {},
477
- language: {
478
- en: {
479
- key: 'en',
480
- count: 1,
481
- state: 'selected',
482
- },
483
- },
484
- creator: {},
485
- collection: {},
486
- year: {},
487
- };
488
-
489
- const el = await fixture<CollectionBrowser>(
490
- html`<collection-browser .searchService=${searchService}>
491
- </collection-browser>`,
492
- );
493
-
494
- el.baseQuery = 'collection:foo';
495
- el.selectedFacets = selectedFacets;
496
- await el.updateComplete;
497
- await el.initialSearchComplete;
498
-
499
- expect(searchService.searchParams?.query).to.equal('collection:foo');
500
- expect(searchService.searchParams?.filters).to.deep.equal({
501
- subject: {
502
- foo: 'inc',
503
- bar: 'exc',
504
- },
505
- language: {
506
- en: 'inc',
507
- },
508
- });
509
- });
510
-
511
- it('fails gracefully if no search service provided', async () => {
512
- const el = await fixture<CollectionBrowser>(
513
- html`<collection-browser></collection-browser>`,
514
- );
515
-
516
- el.baseQuery = 'collection:foo';
517
- await el.updateComplete;
518
-
519
- // This shouldn't throw an error
520
- expect(el.dataSource.fetchPage(3)).to.exist;
521
-
522
- // Should continue showing the empty placeholder
523
- expect(el.shadowRoot?.querySelector('empty-placeholder')).to.exist;
524
- });
525
-
526
- it('restores search type from URL param', async () => {
527
- // Add a sin=TXT param to the URL
528
- const url = new URL(window.location.href);
529
- url.searchParams.append('sin', 'TXT');
530
- window.history.replaceState({}, '', url);
531
-
532
- const searchService = new MockSearchService();
533
-
534
- const el = await fixture<CollectionBrowser>(
535
- html`<collection-browser .searchService=${searchService}>
536
- </collection-browser>`,
537
- );
538
-
539
- expect(el.searchType).to.equal(SearchType.FULLTEXT);
540
- });
541
-
542
- it('does not persist or restore search type from URL param if suppressed', async () => {
543
- // Add a sin=TXT param to the URL
544
- let url = new URL(window.location.href);
545
- url.searchParams.append('sin', 'TXT');
546
- window.history.replaceState({}, '', url);
547
-
548
- const searchService = new MockSearchService();
549
-
550
- const el = await fixture<CollectionBrowser>(
551
- html`<collection-browser
552
- .searchService=${searchService}
553
- suppressURLSinParam
554
- >
555
- </collection-browser>`,
556
- );
557
-
558
- url = new URL(window.location.href);
559
- expect(el.searchType).to.equal(SearchType.DEFAULT);
560
- expect(url.searchParams.has('sin')).to.be.false; // Removes existing sin param
561
-
562
- el.searchType = SearchType.RADIO;
563
- await el.updateComplete;
564
-
565
- url = new URL(window.location.href);
566
- expect(url.searchParams.has('sin')).to.be.false; // Doesn't add sin param
567
- });
568
-
569
- it('can construct tile models with many fields present', async () => {
570
- const searchService = new MockSearchService();
571
-
572
- const el = await fixture<CollectionBrowser>(
573
- html`<collection-browser .searchService=${searchService}>
574
- </collection-browser>`,
575
- );
576
-
577
- el.baseQuery = 'many-fields';
578
- await el.updateComplete;
579
- await el.initialSearchComplete;
580
-
581
- const cellTemplate = el.cellForIndex(0);
582
- expect(cellTemplate).to.exist;
583
-
584
- const cell = await fixture<TileDispatcher>(cellTemplate!);
585
- expect(cell).to.exist;
586
- });
587
-
588
- it('emits empty results event when search fetches no results', async () => {
589
- const searchService = new MockSearchService();
590
- const emptyResultsSpy = sinon.spy();
591
-
592
- const el = await fixture<CollectionBrowser>(
593
- html`<collection-browser
594
- .searchService=${searchService}
595
- @emptyResults=${emptyResultsSpy}
596
- >
597
- </collection-browser>`,
598
- );
599
-
600
- el.baseQuery = 'no-results';
601
- await el.updateComplete;
602
- await el.initialSearchComplete;
603
-
604
- expect(emptyResultsSpy.callCount).to.equal(1);
605
- });
606
-
607
- it('emits searchError event when search results in an error', async () => {
608
- const searchService = new MockSearchService();
609
- const searchErrorSpy = sinon.spy();
610
-
611
- const el = await fixture<CollectionBrowser>(
612
- html`<collection-browser
613
- .searchService=${searchService}
614
- @searchError=${searchErrorSpy}
615
- >
616
- </collection-browser>`,
617
- );
618
-
619
- el.baseQuery = 'error';
620
- await el.updateComplete;
621
- await el.initialSearchComplete;
622
-
623
- expect(searchErrorSpy.callCount).to.equal(1);
624
- });
625
-
626
- it('applies loggedin flag to tile models if needed', async () => {
627
- const searchService = new MockSearchService();
628
-
629
- const el = await fixture<CollectionBrowser>(
630
- html`<collection-browser .searchService=${searchService}>
631
- </collection-browser>`,
632
- );
633
-
634
- el.baseQuery = 'loggedin';
635
- await el.updateComplete;
636
- await el.initialSearchComplete;
637
-
638
- const cellTemplate = el.cellForIndex(0);
639
- expect(cellTemplate).to.exist;
640
-
641
- const cell = await fixture<TileDispatcher>(cellTemplate!);
642
- expect(cell).to.exist;
643
-
644
- expect(cell.model?.loginRequired).to.be.true;
645
- });
646
-
647
- it('applies no-preview flag to tile models if needed', async () => {
648
- const searchService = new MockSearchService();
649
-
650
- const el = await fixture<CollectionBrowser>(
651
- html`<collection-browser .searchService=${searchService}>
652
- </collection-browser>`,
653
- );
654
-
655
- el.baseQuery = 'no-preview';
656
- await el.updateComplete;
657
- await el.initialSearchComplete;
658
-
659
- const cellTemplate = el.cellForIndex(0);
660
- expect(cellTemplate).to.exist;
661
-
662
- const cell = await fixture<TileDispatcher>(cellTemplate!);
663
- expect(cell).to.exist;
664
-
665
- expect(cell.model?.contentWarning).to.be.true;
666
- });
667
-
668
- it('both loggedin and no-preview flags can be set simultaneously', async () => {
669
- const searchService = new MockSearchService();
670
-
671
- const el = await fixture<CollectionBrowser>(
672
- html`<collection-browser .searchService=${searchService}>
673
- </collection-browser>`,
674
- );
675
-
676
- el.baseQuery = 'loggedin-no-preview';
677
- await el.updateComplete;
678
- await el.initialSearchComplete;
679
-
680
- const cellTemplate = el.cellForIndex(0);
681
- expect(cellTemplate).to.exist;
682
-
683
- const cell = await fixture<TileDispatcher>(cellTemplate!);
684
- expect(cell).to.exist;
685
-
686
- expect(cell.model?.loginRequired).to.be.true;
687
- expect(cell.model?.contentWarning).to.be.true;
688
- });
689
-
690
- it('joins full description array into a single string with line breaks', async () => {
691
- const searchService = new MockSearchService();
692
-
693
- const el = await fixture<CollectionBrowser>(
694
- html`<collection-browser .searchService=${searchService}>
695
- </collection-browser>`,
696
- );
697
-
698
- // This query receives an array description like ['line1', 'line2']
699
- el.baseQuery = 'multi-line-description';
700
- await el.updateComplete;
701
- await el.initialSearchComplete;
702
-
703
- const cellTemplate = el.cellForIndex(0);
704
- expect(cellTemplate).to.exist;
705
-
706
- const cell = await fixture<TileDispatcher>(cellTemplate!);
707
- expect(cell).to.exist;
708
-
709
- // Actual model description should be joined
710
- expect(cell.model?.description).to.equal('line1\nline2');
711
- });
712
-
713
- it('can change search type', async () => {
714
- const searchService = new MockSearchService();
715
-
716
- const el = await fixture<CollectionBrowser>(
717
- html`<collection-browser
718
- .searchService=${searchService}
719
- .searchType=${SearchType.METADATA}
720
- ></collection-browser>`,
721
- );
722
-
723
- el.baseQuery = 'collection:foo';
724
- el.searchType = SearchType.FULLTEXT;
725
- await el.updateComplete;
726
- await el.initialSearchComplete;
727
-
728
- expect(searchService.searchParams?.query).to.equal('collection:foo');
729
- expect(searchService.searchType).to.equal(SearchType.FULLTEXT);
730
- });
731
-
732
- it('trims queries of leading/trailing whitespace', async () => {
733
- const searchService = new MockSearchService();
734
- const el = await fixture<CollectionBrowser>(
735
- html`<collection-browser
736
- .searchService=${searchService}
737
- ></collection-browser>`,
738
- );
739
-
740
- el.baseQuery = ' collection:foo ';
741
- await el.updateComplete;
742
- await el.initialSearchComplete;
743
-
744
- expect(searchService.searchParams?.query).to.equal('collection:foo');
745
- });
746
-
747
- it('shows error message when error response received', async () => {
748
- const searchService = new MockSearchService();
749
- const el = await fixture<CollectionBrowser>(
750
- html`<collection-browser
751
- .searchService=${searchService}
752
- ></collection-browser>`,
753
- );
754
-
755
- el.baseQuery = 'error';
756
- await el.updateComplete;
757
- await el.initialSearchComplete;
758
-
759
- const errorPlaceholder = el.shadowRoot?.querySelector(
760
- 'empty-placeholder',
761
- ) as EmptyPlaceholder;
762
- const errorDetails = errorPlaceholder?.shadowRoot?.querySelector(
763
- '.error-details',
764
- ) as HTMLParagraphElement;
765
-
766
- expect(errorDetails).to.exist;
767
- expect(errorDetails.textContent).to.contain('foo');
768
- });
769
-
770
- it('shows error message when error response received for a collection', async () => {
771
- const searchService = new MockSearchService();
772
- const el = await fixture<CollectionBrowser>(
773
- html`<collection-browser
774
- .searchService=${searchService}
775
- ></collection-browser>`,
776
- );
777
-
778
- el.withinCollection = 'error';
779
- await el.updateComplete;
780
- await el.initialSearchComplete;
781
-
782
- const errorPlaceholder = el.shadowRoot?.querySelector(
783
- 'empty-placeholder',
784
- ) as EmptyPlaceholder;
785
- const errorDetails = errorPlaceholder?.shadowRoot?.querySelector(
786
- '.error-details',
787
- ) as HTMLParagraphElement;
788
-
789
- expect(errorDetails).to.exist;
790
- expect(errorDetails.textContent).to.contain('foo');
791
- });
792
-
793
- it('reports malformed response errors to Sentry', async () => {
794
- const sentrySpy = sinon.spy();
795
- // eslint-disable-next-line @typescript-eslint/no-explicit-any
796
- (window as any).Sentry = { captureMessage: sentrySpy };
797
- const searchService = new MockSearchService();
798
- const el = await fixture<CollectionBrowser>(
799
- html`<collection-browser
800
- .searchService=${searchService}
801
- ></collection-browser>`,
802
- );
803
-
804
- el.baseQuery = 'malformed';
805
- await el.updateComplete;
806
- await el.initialSearchComplete;
807
-
808
- expect(sentrySpy.callCount).to.be.greaterThanOrEqual(1);
809
- });
810
-
811
- it('adds collection names to cache when present on response', async () => {
812
- const searchService = new MockSearchService();
813
-
814
- const el = await fixture<CollectionBrowser>(
815
- html`<collection-browser .searchService=${searchService}>
816
- </collection-browser>`,
817
- );
818
-
819
- el.baseQuery = 'collection-titles';
820
- await el.updateComplete;
821
- await el.initialSearchComplete;
822
-
823
- expect(el.dataSource.collectionTitles.get('foo')).to.equal(
824
- 'Foo Collection',
825
- );
826
- expect(el.dataSource.collectionTitles.get('bar')).to.equal(
827
- 'Bar Collection',
828
- );
829
- expect(el.dataSource.collectionTitles.get('baz')).to.equal(
830
- 'Baz Collection',
831
- );
832
- expect(el.dataSource.collectionTitles.get('boop')).to.equal(
833
- 'Boop Collection',
834
- );
835
- });
836
-
837
- it('adds tv channel aliases to cache when present on response', async () => {
838
- const searchService = new MockSearchService();
839
-
840
- const el = await fixture<CollectionBrowser>(
841
- html`<collection-browser .searchService=${searchService}>
842
- </collection-browser>`,
843
- );
844
-
845
- el.baseQuery = 'channel-aliases';
846
- await el.updateComplete;
847
- await el.initialSearchComplete;
848
-
849
- expect(el.dataSource.tvChannelAliases.get('foo')).to.equal('Foo Network');
850
- expect(el.dataSource.tvChannelAliases.get('bar')).to.equal('Bar Network');
851
- });
852
-
853
- it('keeps search results from fetch if no change to query or sort param', async () => {
854
- const resultsSpy = sinon.spy();
855
- const searchService = new MockSearchService({
856
- asyncResponse: true,
857
- resultsSpy,
858
- });
859
-
860
- const el = await fixture<CollectionBrowser>(
861
- html`<collection-browser .searchService=${searchService}>
862
- </collection-browser>`,
863
- );
864
-
865
- el.baseQuery = 'with-sort';
866
- el.selectedSort = SortField.date;
867
- el.sortDirection = 'asc';
868
- await el.updateComplete;
869
-
870
- await el.dataSource.fetchPage(3);
871
-
872
- // If there is no change to the query or sort param during the fetch, the results
873
- // should be read.
874
- expect(resultsSpy.callCount).to.be.greaterThanOrEqual(1);
875
- });
876
-
877
- it('discards obsolete search results if sort params changed before arrival', async () => {
878
- const resultsSpy = sinon.spy();
879
- const searchService = new MockSearchService({
880
- asyncResponse: true,
881
- resultsSpy,
882
- });
883
-
884
- const el = await fixture<CollectionBrowser>(
885
- html`<collection-browser .searchService=${searchService}>
886
- </collection-browser>`,
887
- );
888
-
889
- el.baseQuery = 'with-sort';
890
- el.selectedSort = SortField.date;
891
- el.sortDirection = 'asc';
892
- await el.updateComplete;
893
-
894
- // We want to spy exclusively on the first set of results, not the second
895
- searchService.asyncResponse = false;
896
- searchService.resultsSpy = () => {};
897
-
898
- el.sortDirection = 'desc';
899
- await el.updateComplete;
900
- await el.initialSearchComplete;
901
-
902
- // If the different sort param causes the results to be discarded,
903
- // the first results array should never be read.
904
- expect(resultsSpy.callCount).to.equal(0);
905
- });
906
-
907
- it('discards obsolete search results if sort param added before arrival', async () => {
908
- const resultsSpy = sinon.spy();
909
- const searchService = new MockSearchService({
910
- asyncResponse: true,
911
- resultsSpy,
912
- });
913
-
914
- const el = await fixture<CollectionBrowser>(
915
- html`<collection-browser .searchService=${searchService}>
916
- </collection-browser>`,
917
- );
918
-
919
- el.baseQuery = 'single-result';
920
- await el.updateComplete;
921
-
922
- // We want to spy exclusively on the first set of results, not the second
923
- searchService.asyncResponse = false;
924
- searchService.resultsSpy = () => {};
925
-
926
- el.selectedSort = SortField.date;
927
- el.sortDirection = 'asc';
928
- await el.updateComplete;
929
- await el.initialSearchComplete;
930
-
931
- // If the different sort param causes the results to be discarded,
932
- // the first results array should never be read.
933
- expect(resultsSpy.callCount).to.equal(0);
934
- });
935
-
936
- it('discards obsolete search results if sort param cleared before arrival', async () => {
937
- const resultsSpy = sinon.spy();
938
- const searchService = new MockSearchService({
939
- asyncResponse: true,
940
- resultsSpy,
941
- });
942
-
943
- const el = await fixture<CollectionBrowser>(
944
- html`<collection-browser .searchService=${searchService}>
945
- </collection-browser>`,
946
- );
947
-
948
- el.baseQuery = 'with-sort';
949
- el.selectedSort = SortField.date;
950
- el.sortDirection = 'asc';
951
- await el.updateComplete;
952
-
953
- // We want to spy exclusively on the first set of results, not the second
954
- searchService.asyncResponse = false;
955
- searchService.resultsSpy = () => {};
956
-
957
- el.selectedSort = SortField.default;
958
- await el.updateComplete;
959
- await el.initialSearchComplete;
960
-
961
- // If the different sort param causes the results to be discarded,
962
- // the first results array should never be read.
963
- expect(resultsSpy.callCount).to.equal(0);
964
- });
965
-
966
- it('sets sort properties when user changes sort', async () => {
967
- const searchService = new MockSearchService();
968
- const el = await fixture<CollectionBrowser>(
969
- html`<collection-browser .searchService=${searchService}>
970
- </collection-browser>`,
971
- );
972
-
973
- expect(el.selectedSort).to.equal(SortField.default);
974
-
975
- el.baseQuery = 'foo';
976
- await el.updateComplete;
977
- await nextTick();
978
-
979
- const sortBar = el.shadowRoot?.querySelector(
980
- '#content-container sort-filter-bar',
981
- ) as SortFilterBar;
982
- expect(sortBar, 'sort bar').to.exist;
983
-
984
- // Switch to title sort
985
- sortBar.setSelectedSort(SortField.title);
986
- await el.updateComplete;
987
- expect(el.selectedSort).to.equal(SortField.title);
988
- expect(el.sortDirection).to.equal('asc'); // Default for title sort
989
-
990
- // Switch to descending sort order
991
- sortBar.setSortDirection('desc');
992
- await el.updateComplete;
993
- expect(el.selectedSort).to.equal(SortField.title);
994
- expect(el.sortDirection).to.equal('desc');
995
- });
996
-
997
- it('sets sort filter properties when user selects title filter', async () => {
998
- const searchService = new MockSearchService();
999
- const el = await fixture<CollectionBrowser>(
1000
- html`<collection-browser .searchService=${searchService}>
1001
- </collection-browser>`,
1002
- );
1003
-
1004
- el.baseQuery = 'first-title';
1005
- el.selectedSort = 'title' as SortField;
1006
- el.sortDirection = 'asc';
1007
- el.selectedTitleFilter = 'X';
1008
- await el.updateComplete;
1009
- await el.initialSearchComplete;
1010
-
1011
- expect(searchService.searchParams?.query).to.equal('first-title');
1012
- expect(searchService.searchParams?.filters?.firstTitle?.X).to.equal(
1013
- FilterConstraint.INCLUDE,
1014
- );
1015
- });
1016
-
1017
- it('sets sort filter properties when user selects creator filter', async () => {
1018
- const searchService = new MockSearchService();
1019
- const el = await fixture<CollectionBrowser>(
1020
- html`<collection-browser .searchService=${searchService}>
1021
- </collection-browser>`,
1022
- );
1023
-
1024
- el.baseQuery = 'first-creator';
1025
- el.selectedSort = 'creator' as SortField;
1026
- el.sortDirection = 'asc';
1027
- el.selectedCreatorFilter = 'X';
1028
- await el.updateComplete;
1029
- await el.initialSearchComplete;
1030
-
1031
- expect(searchService.searchParams?.query).to.equal('first-creator');
1032
- expect(searchService.searchParams?.filters?.firstCreator?.X).to.equal(
1033
- FilterConstraint.INCLUDE,
1034
- );
1035
- });
1036
-
1037
- it('sets sort filter properties simultaneous with facets and date range', async () => {
1038
- const searchService = new MockSearchService();
1039
- const selectedFacets: SelectedFacets = {
1040
- collection: { foo: { key: 'foo', state: 'selected', count: 1 } },
1041
- creator: {},
1042
- language: {},
1043
- lending: {},
1044
- mediatype: {},
1045
- subject: {},
1046
- year: {},
1047
- };
1048
-
1049
- const el = await fixture<CollectionBrowser>(
1050
- html`<collection-browser .searchService=${searchService}>
1051
- </collection-browser>`,
1052
- );
1053
-
1054
- el.baseQuery = 'first-creator';
1055
- el.selectedSort = 'creator' as SortField;
1056
- el.selectedFacets = selectedFacets;
1057
- el.minSelectedDate = '1950';
1058
- el.maxSelectedDate = '1970';
1059
- el.sortDirection = 'asc';
1060
- el.selectedCreatorFilter = 'X';
1061
- await el.updateComplete;
1062
- await el.initialSearchComplete;
1063
-
1064
- expect(searchService.searchParams?.query).to.equal('first-creator');
1065
- expect(searchService.searchParams?.filters).to.deep.equal({
1066
- collection: {
1067
- foo: 'inc',
1068
- },
1069
- year: {
1070
- '1950': 'gte',
1071
- '1970': 'lte',
1072
- },
1073
- firstCreator: {
1074
- X: 'inc',
1075
- },
1076
- });
1077
- });
1078
-
1079
- it('applies correct TV search filter for commercials', async () => {
1080
- const searchService = new MockSearchService();
1081
- const el = await fixture<CollectionBrowser>(
1082
- html`<collection-browser .searchService=${searchService}>
1083
- </collection-browser>`,
1084
- );
1085
-
1086
- el.baseQuery = 'tv-fields';
1087
- el.searchType = SearchType.TV;
1088
- el.selectedFacets = {
1089
- clip_type: {
1090
- commercial: { key: 'commercial', count: 1, state: 'selected' },
1091
- },
1092
- };
1093
- await el.updateComplete;
1094
- await el.initialSearchComplete;
1095
-
1096
- expect(searchService.searchParams?.filters?.clip_type?.commercial).to.equal(
1097
- FilterConstraint.INCLUDE,
1098
- );
1099
- });
1100
-
1101
- it('applies correct TV search filter for fact checks', async () => {
1102
- const searchService = new MockSearchService();
1103
- const el = await fixture<CollectionBrowser>(
1104
- html`<collection-browser .searchService=${searchService}>
1105
- </collection-browser>`,
1106
- );
1107
-
1108
- el.baseQuery = 'tv-fields';
1109
- el.searchType = SearchType.TV;
1110
- el.selectedFacets = {
1111
- clip_type: {
1112
- 'fact check': { key: 'fact check', count: 1, state: 'selected' },
1113
- },
1114
- };
1115
- await el.updateComplete;
1116
- await el.initialSearchComplete;
1117
-
1118
- expect(
1119
- searchService.searchParams?.filters?.clip_type?.['fact check'],
1120
- ).to.equal(FilterConstraint.INCLUDE);
1121
- });
1122
-
1123
- it('applies correct TV search filter for quotes', async () => {
1124
- const searchService = new MockSearchService();
1125
- const el = await fixture<CollectionBrowser>(
1126
- html`<collection-browser .searchService=${searchService}>
1127
- </collection-browser>`,
1128
- );
1129
-
1130
- el.baseQuery = 'tv-fields';
1131
- el.searchType = SearchType.TV;
1132
- el.selectedFacets = {
1133
- clip_type: { quote: { key: 'quote', count: 1, state: 'selected' } },
1134
- };
1135
- await el.updateComplete;
1136
- await el.initialSearchComplete;
1137
-
1138
- expect(searchService.searchParams?.filters?.clip_type?.quote).to.equal(
1139
- FilterConstraint.INCLUDE,
1140
- );
1141
- });
1142
-
1143
- it('resets letter filters when query changes', async () => {
1144
- const searchService = new MockSearchService();
1145
- const el = await fixture<CollectionBrowser>(
1146
- html`<collection-browser .searchService=${searchService}>
1147
- </collection-browser>`,
1148
- );
1149
-
1150
- el.baseQuery = 'first-creator';
1151
- el.selectedSort = 'creator' as SortField;
1152
- el.sortDirection = 'asc';
1153
- el.selectedCreatorFilter = 'X';
1154
- await el.updateComplete;
1155
- await el.initialSearchComplete;
1156
- await nextTick();
1157
-
1158
- expect(searchService.searchParams?.query).to.equal('first-creator');
1159
- expect(searchService.searchParams?.filters?.firstCreator?.X).to.equal(
1160
- FilterConstraint.INCLUDE,
1161
- );
1162
-
1163
- el.baseQuery = 'collection:foo';
1164
- await el.updateComplete;
1165
- await nextTick();
1166
-
1167
- expect(searchService.searchParams?.query).to.equal('collection:foo');
1168
- expect(searchService.searchParams?.filters?.firstCreator).not.to.exist;
1169
- });
1170
-
1171
- it('sets date range query when date picker selection changed', async () => {
1172
- const searchService = new MockSearchService();
1173
- const mockAnalyticsHandler = new MockAnalyticsHandler();
1174
- const el = await fixture<CollectionBrowser>(
1175
- html`<collection-browser
1176
- .searchService=${searchService}
1177
- .analyticsHandler=${mockAnalyticsHandler}
1178
- .suppressPlaceholders=${true}
1179
- >
1180
- </collection-browser>`,
1181
- );
1182
-
1183
- el.baseQuery = 'years'; // Includes year_histogram aggregation in response
1184
- el.showHistogramDatePicker = true;
1185
- await el.updateComplete;
1186
-
1187
- const facets = el.shadowRoot?.querySelector(
1188
- 'collection-facets',
1189
- ) as CollectionFacets;
1190
- await facets?.updateComplete;
1191
-
1192
- // Wait for the date picker to be rendered (which may take until the next tick)
1193
- await nextTick();
1194
-
1195
- const histogram = facets?.shadowRoot?.querySelector(
1196
- 'histogram-date-range',
1197
- ) as HistogramDateRange;
1198
-
1199
- expect(histogram, 'histogram exists').to.exist;
1200
-
1201
- // Enter a new min date into the date picker
1202
- const minDateInput = histogram.shadowRoot?.querySelector(
1203
- '#date-min',
1204
- ) as HTMLInputElement;
1205
-
1206
- const pressEnterEvent = new KeyboardEvent('keyup', {
1207
- key: 'Enter',
1208
- });
1209
-
1210
- minDateInput.value = '1960';
1211
- minDateInput.dispatchEvent(pressEnterEvent);
1212
-
1213
- // Wait for the histogram's update delay
1214
- await aTimeout(histogram.updateDelay + 50);
1215
-
1216
- // Ensure that the histogram change propagated to the collection browser's
1217
- // date query correctly.
1218
- await el.updateComplete;
1219
- expect(el.minSelectedDate).to.equal('1960');
1220
- expect(el.maxSelectedDate).to.equal('2009');
1221
- });
1222
-
1223
- it('sets date range query when monthly date picker selection changed', async () => {
1224
- const searchService = new MockSearchService();
1225
- const el = await fixture<CollectionBrowser>(
1226
- html`<collection-browser
1227
- .searchService=${searchService}
1228
- .suppressPlaceholders=${true}
1229
- >
1230
- </collection-browser>`,
1231
- );
1232
-
1233
- el.baseQuery = 'months'; // Includes date_histogram aggregation in response
1234
- el.searchType = SearchType.TV;
1235
- el.showHistogramDatePicker = true;
1236
- await el.updateComplete;
1237
-
1238
- const facets = el.shadowRoot?.querySelector(
1239
- 'collection-facets',
1240
- ) as CollectionFacets;
1241
- await facets?.updateComplete;
1242
-
1243
- // Wait for the date picker to be rendered (which may take until the next tick)
1244
- await nextTick();
1245
-
1246
- const histogram = facets?.shadowRoot?.querySelector(
1247
- 'histogram-date-range',
1248
- ) as HistogramDateRange;
1249
-
1250
- expect(histogram, 'histogram exists').to.exist;
1251
-
1252
- // Enter a new min date into the date picker
1253
- const minDateInput = histogram.shadowRoot?.querySelector(
1254
- '#date-min',
1255
- ) as HTMLInputElement;
1256
-
1257
- const pressEnterEvent = new KeyboardEvent('keyup', {
1258
- key: 'Enter',
1259
- });
1260
-
1261
- minDateInput.value = '2001-02';
1262
- minDateInput.dispatchEvent(pressEnterEvent);
1263
-
1264
- // Wait for the histogram's update delay
1265
- await aTimeout(histogram.updateDelay + 50);
1266
-
1267
- // Ensure that the histogram change propagated to the collection browser's
1268
- // date query correctly.
1269
- await el.updateComplete;
1270
- expect(el.minSelectedDate).to.equal('2001-02');
1271
- expect(el.maxSelectedDate).to.equal('2002-12');
1272
- });
1273
-
1274
- it('emits event when results start and end loading', async () => {
1275
- const spy = sinon.spy();
1276
- const searchService = new MockSearchService();
1277
- const el = await fixture<CollectionBrowser>(
1278
- html`<collection-browser
1279
- .searchService=${searchService}
1280
- @searchResultsLoadingChanged=${spy}
1281
- ></collection-browser>`,
1282
- );
1283
- spy.resetHistory();
1284
-
1285
- el.baseQuery = 'collection:foo';
1286
- await el.updateComplete;
1287
- await el.initialSearchComplete;
1288
-
1289
- // Should initially emit loading=true, then later emit loading=false
1290
- expect(spy.callCount).to.equal(2);
1291
- expect(spy.firstCall.firstArg?.detail?.loading).to.equal(true);
1292
- expect(spy.secondCall.firstArg?.detail?.loading).to.equal(false);
1293
- });
1294
-
1295
- it('collapses extra set of quotes around href field', async () => {
1296
- const searchService = new MockSearchService();
1297
- const el = await fixture<CollectionBrowser>(
1298
- html`<collection-browser
1299
- .searchService=${searchService}
1300
- .baseNavigationUrl=${''}
1301
- ></collection-browser>`,
1302
- );
1303
-
1304
- el.baseQuery = 'extra-quoted-href';
1305
- await el.updateComplete;
1306
- await el.initialSearchComplete;
1307
- await el.updateComplete;
1308
- await aTimeout(50);
1309
-
1310
- // Original href q param starts/ends with %22%22, but should be collapsed to %22 before render
1311
- expect(el.dataSource.getTileModelAt(0)?.href).to.equal(
1312
- '/details/foo?q=%22quoted+query%22',
1313
- );
1314
- });
1315
-
1316
- it('sets default sort from collection metadata', async () => {
1317
- const searchService = new MockSearchService();
1318
- const el = await fixture<CollectionBrowser>(
1319
- html`<collection-browser
1320
- .searchService=${searchService}
1321
- .baseNavigationUrl=${''}
1322
- ></collection-browser>`,
1323
- );
1324
-
1325
- el.withinCollection = 'default-sort';
1326
- await el.updateComplete;
1327
- await el.initialSearchComplete;
1328
- await el.updateComplete;
1329
- await aTimeout(50);
1330
-
1331
- const sortBar = el.shadowRoot?.querySelector(
1332
- 'sort-filter-bar',
1333
- ) as SortFilterBar;
1334
- expect(sortBar).to.exist;
1335
- expect(sortBar.defaultSortField).to.equal(SortField.title);
1336
- expect(sortBar.defaultSortDirection).to.equal('asc');
1337
- expect(sortBar.selectedSort).to.equal(SortField.default);
1338
- expect(sortBar.sortDirection).to.be.null;
1339
- });
1340
-
1341
- it('sets default sort from collection metadata in "-field" format', async () => {
1342
- const searchService = new MockSearchService();
1343
- const el = await fixture<CollectionBrowser>(
1344
- html`<collection-browser
1345
- .searchService=${searchService}
1346
- .baseNavigationUrl=${''}
1347
- ></collection-browser>`,
1348
- );
1349
-
1350
- el.withinCollection = 'default-sort-concise';
1351
- await el.updateComplete;
1352
- await el.initialSearchComplete;
1353
- await el.updateComplete;
1354
- await aTimeout(50);
1355
-
1356
- const sortBar = el.shadowRoot?.querySelector(
1357
- 'sort-filter-bar',
1358
- ) as SortFilterBar;
1359
- expect(sortBar).to.exist;
1360
- expect(sortBar.defaultSortField).to.equal(SortField.dateadded);
1361
- expect(sortBar.defaultSortDirection).to.equal('desc');
1362
- expect(sortBar.selectedSort).to.equal(SortField.default);
1363
- expect(sortBar.sortDirection).to.be.null;
1364
- });
1365
-
1366
- it('falls back to weekly views default sorting on profiles when tab not set', async () => {
1367
- const el = await fixture<CollectionBrowser>(
1368
- html`<collection-browser
1369
- .withinProfile=${'@foobar'}
1370
- ></collection-browser>`,
1371
- );
1372
-
1373
- el.applyDefaultProfileSort();
1374
- expect(el.defaultSortParam).to.deep.equal({
1375
- field: 'week',
1376
- direction: 'desc',
1377
- });
1378
- });
1379
-
1380
- it('uses relevance sort as default when a query is set', async () => {
1381
- const searchService = new MockSearchService();
1382
- const el = await fixture<CollectionBrowser>(
1383
- html`<collection-browser
1384
- .searchService=${searchService}
1385
- .baseNavigationUrl=${''}
1386
- ></collection-browser>`,
1387
- );
1388
-
1389
- el.withinCollection = 'default-sort';
1390
- el.baseQuery = 'default-sort';
1391
- await el.updateComplete;
1392
- await el.initialSearchComplete;
1393
- await el.updateComplete;
1394
- await aTimeout(50);
1395
-
1396
- const sortBar = el.shadowRoot?.querySelector(
1397
- 'sort-filter-bar',
1398
- ) as SortFilterBar;
1399
- expect(sortBar).to.exist;
1400
- expect(sortBar.defaultSortField).to.equal(SortField.relevance);
1401
- expect(sortBar.defaultSortDirection).to.be.null;
1402
- expect(sortBar.selectedSort).to.equal(SortField.default);
1403
- expect(sortBar.sortDirection).to.be.null;
1404
- });
1405
-
1406
- it('uses date favorited sort as default when targeting fav- collection', async () => {
1407
- const searchService = new MockSearchService();
1408
- const el = await fixture<CollectionBrowser>(
1409
- html`<collection-browser
1410
- .searchService=${searchService}
1411
- .baseNavigationUrl=${''}
1412
- ></collection-browser>`,
1413
- );
1414
-
1415
- el.withinCollection = 'fav-sort';
1416
- await el.updateComplete;
1417
- await el.initialSearchComplete;
1418
- await el.updateComplete;
1419
- await aTimeout(50);
1420
-
1421
- const sortBar = el.shadowRoot?.querySelector(
1422
- 'sort-filter-bar',
1423
- ) as SortFilterBar;
1424
- expect(sortBar).to.exist;
1425
- expect(sortBar.defaultSortField).to.equal(SortField.datefavorited);
1426
- expect(sortBar.defaultSortDirection).to.equal('desc');
1427
- expect(sortBar.selectedSort).to.equal(SortField.default);
1428
- expect(sortBar.sortDirection).to.be.null;
1429
- });
1430
-
1431
- it('scrolls to page', async () => {
1432
- const searchService = new MockSearchService();
1433
- const el = await fixture<CollectionBrowser>(
1434
- html`<collection-browser .searchService=${searchService}>
1435
- </collection-browser>`,
1436
- );
1437
-
1438
- // Infinite scroller won't exist unless there's a base query.
1439
- // First ensure that we don't throw errors when it doesn't exist.
1440
- await el.goToPage(1);
1441
-
1442
- // And make sure it correctly calls scrollToCell when the
1443
- // infinite scroller does exist.
1444
- el.baseQuery = 'collection:foo';
1445
- await el.updateComplete;
1446
- await nextTick();
1447
-
1448
- const infiniteScroller = el.shadowRoot?.querySelector(
1449
- 'infinite-scroller',
1450
- ) as InfiniteScroller;
1451
- expect(infiniteScroller).to.exist;
1452
-
1453
- const oldScrollToCell = infiniteScroller.scrollToCell;
1454
- const spy = sinon.spy();
1455
- infiniteScroller.scrollToCell = spy;
1456
-
1457
- await el.goToPage(1);
1458
- expect(spy.callCount, 'scroll to page fires once').to.equal(1);
1459
-
1460
- infiniteScroller.scrollToCell = oldScrollToCell;
1461
- });
1462
-
1463
- it('shows mobile facets in mobile view', async () => {
1464
- const searchService = new MockSearchService();
1465
- const el = await fixture<CollectionBrowser>(
1466
- html`<collection-browser
1467
- .searchService=${searchService}
1468
- .mobileBreakpoint=${9999}
1469
- ></collection-browser>`,
1470
- );
1471
-
1472
- el.baseQuery = 'collection:foo';
1473
- await el.updateComplete;
1474
-
1475
- const contentContainer = el.shadowRoot?.querySelector(
1476
- '#content-container',
1477
- ) as HTMLElement;
1478
-
1479
- el.handleResize({
1480
- target: contentContainer,
1481
- contentRect: contentContainer.getBoundingClientRect(),
1482
- borderBoxSize: [],
1483
- contentBoxSize: [],
1484
- devicePixelContentBoxSize: [],
1485
- });
1486
- await el.updateComplete;
1487
-
1488
- const mobileFacets = el.shadowRoot?.querySelector(
1489
- '#mobile-filter-collapse',
1490
- );
1491
- expect(mobileFacets).to.exist;
1492
- });
1493
-
1494
- it('fires analytics when mobile facets toggled', async () => {
1495
- const searchService = new MockSearchService();
1496
- const analyticsHandler = new MockAnalyticsHandler();
1497
- const el = await fixture<CollectionBrowser>(
1498
- html`<collection-browser
1499
- .searchService=${searchService}
1500
- .analyticsHandler=${analyticsHandler}
1501
- .searchContext=${'foobar-context'}
1502
- .mobileBreakpoint=${9999}
1503
- ></collection-browser>`,
1504
- );
1505
-
1506
- el.baseQuery = 'collection:foo';
1507
- await el.updateComplete;
1508
-
1509
- const contentContainer = el.shadowRoot?.querySelector(
1510
- '#content-container',
1511
- ) as HTMLElement;
1512
-
1513
- el.handleResize({
1514
- target: contentContainer,
1515
- contentRect: contentContainer.getBoundingClientRect(),
1516
- borderBoxSize: [],
1517
- contentBoxSize: [],
1518
- devicePixelContentBoxSize: [],
1519
- });
1520
- await el.updateComplete;
1521
-
1522
- const mobileFacets = el.shadowRoot?.querySelector(
1523
- '#mobile-filter-collapse',
1524
- ) as HTMLDetailsElement;
1525
- expect(mobileFacets).to.exist;
1526
-
1527
- // We set up a Promise to wait for the 'toggle' event on the collapser,
1528
- // which is what triggers the analytics.
1529
- let facetsToggled = new Promise(resolve => {
1530
- mobileFacets.addEventListener('toggle', resolve);
1531
- });
1532
-
1533
- // Open the mobile facets accordion & check analytics
1534
- const mobileFacetsHeader = mobileFacets.querySelector('summary');
1535
- expect(mobileFacetsHeader).to.exist;
1536
- mobileFacetsHeader!.click();
1537
- await facetsToggled;
1538
- expect(analyticsHandler.callCategory).to.equal('foobar-context');
1539
- expect(analyticsHandler.callAction).to.equal(
1540
- analyticsActions.mobileFacetsToggled,
1541
- );
1542
- expect(analyticsHandler.callLabel).to.equal('open');
1543
-
1544
- // Close the mobile facets accordion & check analytics
1545
- facetsToggled = new Promise(resolve => {
1546
- mobileFacets.addEventListener('toggle', resolve);
1547
- });
1548
- mobileFacetsHeader!.click();
1549
- await facetsToggled;
1550
- expect(analyticsHandler.callCategory).to.equal('foobar-context');
1551
- expect(analyticsHandler.callAction).to.equal(
1552
- analyticsActions.mobileFacetsToggled,
1553
- );
1554
- expect(analyticsHandler.callLabel).to.equal('closed');
1555
- });
1556
-
1557
- it('sets parent collections to prop when searching a collection', async () => {
1558
- const searchService = new MockSearchService();
1559
- const el = await fixture<CollectionBrowser>(
1560
- html`<collection-browser
1561
- .searchService=${searchService}
1562
- .withinCollection=${'fake'}
1563
- ></collection-browser>`,
1564
- );
1565
-
1566
- el.baseQuery = 'parent-collections';
1567
- await el.updateComplete;
1568
- await el.initialSearchComplete;
1569
- await aTimeout(0);
1570
-
1571
- expect(el.dataSource.parentCollections).to.deep.equal(['foo', 'bar']);
1572
- });
1573
-
1574
- it('recognizes TV collections', async () => {
1575
- const searchService = new MockSearchService();
1576
- const el = await fixture<CollectionBrowser>(
1577
- html`<collection-browser
1578
- .searchService=${searchService}
1579
- .withinCollection=${'TV-FOO'}
1580
- ></collection-browser>`,
1581
- );
1582
-
1583
- el.baseQuery = 'tv-collection';
1584
- await el.updateComplete;
1585
- await el.initialSearchComplete;
1586
- await aTimeout(0);
1587
-
1588
- expect(el.isTVCollection).to.be.true;
1589
- });
1590
-
1591
- it('refreshes when certain properties change - with some analytics event sampling', async () => {
1592
- const mockAnalyticsHandler = new MockAnalyticsHandler();
1593
- const searchService = new MockSearchService();
1594
- const el = await fixture<CollectionBrowser>(
1595
- html`<collection-browser
1596
- .analyticsHandler=${mockAnalyticsHandler}
1597
- .searchService=${searchService}
1598
- ></collection-browser>`,
1599
- );
1600
- const infiniteScrollerRefreshSpy = sinon.spy();
1601
-
1602
- // Infinite scroller won't exist unless there's a base query
1603
- el.baseQuery = 'collection:foo';
1604
- await el.updateComplete;
1605
- await nextTick();
1606
-
1607
- const infiniteScroller = el.shadowRoot?.querySelector('infinite-scroller');
1608
- (infiniteScroller as InfiniteScroller).reload = infiniteScrollerRefreshSpy;
1609
- expect(infiniteScrollerRefreshSpy.called).to.be.false;
1610
- expect(infiniteScrollerRefreshSpy.callCount).to.equal(0);
1611
-
1612
- // testing: `loggedIn`
1613
- el.loggedIn = true;
1614
- await el.updateComplete;
1615
-
1616
- expect(infiniteScrollerRefreshSpy.called, 'Infinite Scroller Refresh').to.be
1617
- .true;
1618
- expect(
1619
- infiniteScrollerRefreshSpy.callCount,
1620
- 'Infinite Scroller Refresh call count',
1621
- ).to.equal(1);
1622
-
1623
- el.loggedIn = false;
1624
- await el.updateComplete;
1625
-
1626
- expect(
1627
- infiniteScrollerRefreshSpy.callCount,
1628
- '2nd Infinite Scroller Refresh',
1629
- ).to.equal(2);
1630
-
1631
- // testing: `displayMode`
1632
- el.displayMode = 'list-compact';
1633
- el.searchContext = 'beta-search';
1634
- await el.updateComplete;
1635
- expect(
1636
- infiniteScrollerRefreshSpy.callCount,
1637
- '3rd Infinite Scroller Refresh',
1638
- ).to.equal(3);
1639
-
1640
- expect(mockAnalyticsHandler.callCategory).to.equal('beta-search');
1641
- expect(mockAnalyticsHandler.callAction).to.equal('displayMode');
1642
- expect(mockAnalyticsHandler.callLabel).to.equal('list-compact');
1643
-
1644
- el.displayMode = 'list-detail';
1645
- await el.updateComplete;
1646
- expect(
1647
- infiniteScrollerRefreshSpy.callCount,
1648
- '4th Infinite Scroller Refresh',
1649
- ).to.equal(4);
1650
-
1651
- expect(mockAnalyticsHandler.callCategory).to.equal('beta-search');
1652
- expect(mockAnalyticsHandler.callAction).to.equal('displayMode');
1653
- expect(mockAnalyticsHandler.callLabel).to.equal('list-detail');
1654
-
1655
- // testing: `baseNavigationUrl`
1656
- el.baseNavigationUrl = 'https://funtestsite.com';
1657
- await el.updateComplete;
1658
- expect(
1659
- infiniteScrollerRefreshSpy.callCount,
1660
- '5th Infinite Scroller Refresh',
1661
- ).to.equal(5);
1662
-
1663
- // testing: `baseImageUrl`
1664
- el.baseImageUrl = 'https://funtestsiteforimages.com';
1665
- await el.updateComplete;
1666
- expect(
1667
- infiniteScrollerRefreshSpy.callCount,
1668
- '6th Infinite Scroller Refresh',
1669
- ).to.equal(6);
1670
- });
1671
-
1672
- it('query the search service for single result', async () => {
1673
- const searchService = new MockSearchService();
1674
-
1675
- const el = await fixture<CollectionBrowser>(
1676
- html`<collection-browser .searchService=${searchService}>
1677
- </collection-browser>`,
1678
- );
1679
-
1680
- el.baseQuery = 'single-result';
1681
- await el.updateComplete;
1682
- await el.initialSearchComplete;
1683
-
1684
- expect(
1685
- el.shadowRoot?.querySelector('#big-results-label')?.textContent,
1686
- ).to.contains('Result');
1687
- });
1688
-
1689
- it('`searchContext` prop helps describe where component is being used', async () => {
1690
- const el = await fixture<CollectionBrowser>(
1691
- html`<collection-browser></collection-browser>`,
1692
- );
1693
-
1694
- expect(el.searchContext).to.equal(analyticsCategories.default);
1695
-
1696
- el.searchContext = 'unicorn-search';
1697
- await el.updateComplete;
1698
-
1699
- expect(el.searchContext).to.equal('unicorn-search');
1700
-
1701
- // property is reflected as attribute
1702
- expect(el.getAttribute('searchcontext')).to.equal('unicorn-search');
1703
- });
1704
-
1705
- it('respects the initial set of URL parameters for a general search', async () => {
1706
- const url = new URL(window.location.href);
1707
- const { searchParams } = url;
1708
- searchParams.set('query', 'foo');
1709
- searchParams.set('sin', 'TXT');
1710
- searchParams.set('sort', 'title');
1711
- searchParams.append('not[]', 'mediatype:"data"');
1712
- searchParams.append('and[]', 'subject:"baz"');
1713
- searchParams.append('and[]', 'firstTitle:X');
1714
- searchParams.append('and[]', 'year:[2000 TO 2010]');
1715
- window.history.replaceState({}, '', url);
1716
-
1717
- const searchService = new MockSearchService();
1718
- const el = await fixture<CollectionBrowser>(
1719
- html`<collection-browser .searchService=${searchService}>
1720
- </collection-browser>`,
1721
- );
1722
-
1723
- await el.initialSearchComplete;
1724
- await el.updateComplete;
1725
-
1726
- expect(el.baseQuery).to.equal('foo');
1727
- expect(el.searchType).to.equal(SearchType.FULLTEXT);
1728
- expect(el.selectedSort).to.equal(SortField.title);
1729
- expect(el.selectedFacets?.mediatype?.data?.state).to.equal('hidden');
1730
- expect(el.selectedFacets?.subject?.baz?.state).to.equal('selected');
1731
- expect(el.selectedTitleFilter).to.equal('X');
1732
- expect(el.minSelectedDate).to.equal('2000');
1733
- expect(el.maxSelectedDate).to.equal('2010');
1734
- });
1735
-
1736
- it('respects the initial set of URL parameters within a collection', async () => {
1737
- const url = new URL(window.location.href);
1738
- const { searchParams } = url;
1739
- searchParams.set('query', 'foo');
1740
- searchParams.set('sin', 'TXT');
1741
- searchParams.set('sort', 'title');
1742
- searchParams.append('not[]', 'mediatype:"data"');
1743
- searchParams.append('and[]', 'subject:"baz"');
1744
- searchParams.append('and[]', 'firstTitle:X');
1745
- searchParams.append('and[]', 'year:[2000 TO 2010]');
1746
- window.history.replaceState({}, '', url);
1747
-
1748
- const searchService = new MockSearchService();
1749
- const el = await fixture<CollectionBrowser>(
1750
- html`<collection-browser
1751
- .searchService=${searchService}
1752
- .withinCollection=${'foobar'}
1753
- >
1754
- </collection-browser>`,
1755
- );
1756
-
1757
- await el.initialSearchComplete;
1758
- await el.updateComplete;
1759
-
1760
- expect(el.withinCollection).to.equal('foobar');
1761
- expect(el.baseQuery).to.equal('foo');
1762
- expect(el.searchType).to.equal(SearchType.FULLTEXT);
1763
- expect(el.selectedSort).to.equal(SortField.title);
1764
- expect(el.selectedFacets?.mediatype?.data?.state).to.equal('hidden');
1765
- expect(el.selectedFacets?.subject?.baz?.state).to.equal('selected');
1766
- expect(el.selectedTitleFilter).to.equal('X');
1767
- expect(el.minSelectedDate).to.equal('2000');
1768
- expect(el.maxSelectedDate).to.equal('2010');
1769
- });
1770
-
1771
- it('respects the initial set of URL parameters within a profile page', async () => {
1772
- const url = new URL(window.location.href);
1773
- const { searchParams } = url;
1774
- searchParams.set('query', 'foo');
1775
- searchParams.append('not[]', 'mediatype:"data"');
1776
- searchParams.append('and[]', 'subject:"baz"');
1777
- searchParams.append('and[]', 'firstTitle:X');
1778
- searchParams.append('and[]', 'year:[2000 TO 2010]');
1779
- window.history.replaceState({}, '', url);
1780
-
1781
- const searchService = new MockSearchService();
1782
- const el = await fixture<CollectionBrowser>(
1783
- html`<collection-browser
1784
- .searchService=${searchService}
1785
- .withinProfile=${'@foobar'}
1786
- .profileElement=${'uploads'}
1787
- >
1788
- </collection-browser>`,
1789
- );
1790
-
1791
- await el.initialSearchComplete;
1792
- await el.updateComplete;
1793
-
1794
- expect(el.withinProfile).to.equal('@foobar');
1795
- expect(el.profileElement).to.equal('uploads');
1796
- expect(el.baseQuery).to.equal('foo');
1797
- expect(el.searchType).to.equal(SearchType.DEFAULT);
1798
- expect(el.selectedFacets?.mediatype?.data?.state).to.equal('hidden');
1799
- expect(el.selectedFacets?.subject?.baz?.state).to.equal('selected');
1800
- expect(el.selectedTitleFilter).to.equal('X');
1801
- expect(el.minSelectedDate).to.equal('2000');
1802
- expect(el.maxSelectedDate).to.equal('2010');
1803
- });
1804
-
1805
- it('clears filters except sort when query changes for a general search', async () => {
1806
- const url = new URL(window.location.href);
1807
- const { searchParams } = url;
1808
- searchParams.set('query', 'foo');
1809
- searchParams.set('sin', 'TXT');
1810
- searchParams.set('sort', 'title');
1811
- searchParams.append('not[]', 'mediatype:"data"');
1812
- searchParams.append('and[]', 'subject:"baz"');
1813
- searchParams.append('and[]', 'firstTitle:X');
1814
- searchParams.append('and[]', 'year:[2000 TO 2010]');
1815
- window.history.replaceState({}, '', url);
1816
-
1817
- const searchService = new MockSearchService();
1818
- const el = await fixture<CollectionBrowser>(
1819
- html`<collection-browser .searchService=${searchService}>
1820
- </collection-browser>`,
1821
- );
1822
-
1823
- await el.initialSearchComplete;
1824
- await el.updateComplete;
1825
-
1826
- el.baseQuery = 'bar';
1827
- await el.updateComplete;
1828
-
1829
- expect(el.baseQuery).to.equal('bar');
1830
- expect(el.searchType).to.equal(SearchType.FULLTEXT);
1831
- expect(el.selectedSort).to.equal(SortField.title);
1832
- expect(el.selectedFacets?.mediatype?.data).not.to.exist;
1833
- expect(el.selectedFacets?.subject?.baz).not.to.exist;
1834
- expect(el.selectedTitleFilter).not.to.exist;
1835
- expect(el.minSelectedDate).not.to.exist;
1836
- expect(el.maxSelectedDate).not.to.exist;
1837
- });
1838
-
1839
- it('clears filters except sort when query changes within a collection', async () => {
1840
- const url = new URL(window.location.href);
1841
- const { searchParams } = url;
1842
- searchParams.set('query', 'foo');
1843
- searchParams.set('sin', 'TXT');
1844
- searchParams.set('sort', 'title');
1845
- searchParams.append('not[]', 'mediatype:"data"');
1846
- searchParams.append('and[]', 'subject:"baz"');
1847
- searchParams.append('and[]', 'firstTitle:X');
1848
- searchParams.append('and[]', 'year:[2000 TO 2010]');
1849
- window.history.replaceState({}, '', url);
1850
-
1851
- const searchService = new MockSearchService();
1852
- const el = await fixture<CollectionBrowser>(
1853
- html`<collection-browser
1854
- .searchService=${searchService}
1855
- .withinCollection=${'foobar'}
1856
- >
1857
- </collection-browser>`,
1858
- );
1859
-
1860
- el.baseQuery = 'bar';
1861
- await el.updateComplete;
1862
-
1863
- expect(el.withinCollection).to.equal('foobar');
1864
- expect(el.baseQuery).to.equal('bar');
1865
- expect(el.searchType).to.equal(SearchType.FULLTEXT);
1866
- expect(el.selectedSort).to.equal(SortField.title);
1867
- expect(el.selectedFacets?.mediatype?.data).not.to.exist;
1868
- expect(el.selectedFacets?.subject?.baz).not.to.exist;
1869
- expect(el.selectedTitleFilter).not.to.exist;
1870
- expect(el.minSelectedDate).not.to.exist;
1871
- expect(el.maxSelectedDate).not.to.exist;
1872
- });
1873
-
1874
- it('clears filters *including* sort when target collection changes', async () => {
1875
- const url = new URL(window.location.href);
1876
- const { searchParams } = url;
1877
- searchParams.set('query', 'foo');
1878
- searchParams.set('sin', 'TXT');
1879
- searchParams.set('sort', 'title');
1880
- searchParams.append('not[]', 'mediatype:"data"');
1881
- searchParams.append('and[]', 'subject:"baz"');
1882
- searchParams.append('and[]', 'firstTitle:X');
1883
- searchParams.append('and[]', 'year:[2000 TO 2010]');
1884
- window.history.replaceState({}, '', url);
1885
-
1886
- const searchService = new MockSearchService();
1887
- const el = await fixture<CollectionBrowser>(
1888
- html`<collection-browser
1889
- .searchService=${searchService}
1890
- .withinCollection=${'foobar'}
1891
- >
1892
- </collection-browser>`,
1893
- );
1894
-
1895
- el.withinCollection = 'bar';
1896
- await el.updateComplete;
1897
-
1898
- expect(el.withinCollection).to.equal('bar');
1899
- expect(el.baseQuery).to.equal('foo');
1900
- expect(el.searchType).to.equal(SearchType.FULLTEXT);
1901
- expect(el.selectedSort).to.equal(SortField.default);
1902
- expect(el.selectedFacets?.mediatype?.data).not.to.exist;
1903
- expect(el.selectedFacets?.subject?.baz).not.to.exist;
1904
- expect(el.selectedTitleFilter).not.to.exist;
1905
- expect(el.minSelectedDate).not.to.exist;
1906
- expect(el.maxSelectedDate).not.to.exist;
1907
- });
1908
-
1909
- it('correctly retrieves web archive hits', async () => {
1910
- const searchService = new MockSearchService();
1911
- const el = await fixture<CollectionBrowser>(
1912
- html`<collection-browser
1913
- .searchService=${searchService}
1914
- .withinProfile=${'@foo'}
1915
- .profileElement=${'web_archives'}
1916
- >
1917
- </collection-browser>`,
1918
- );
1919
-
1920
- el.baseQuery = 'web-archive';
1921
- await el.updateComplete;
1922
- await el.initialSearchComplete;
1923
- await nextTick();
1924
-
1925
- console.log(
1926
- '\n\n*****\n\n*****\n\n',
1927
- el.dataSource.getAllPages(),
1928
- '\n\n*****\n\n*****\n\n',
1929
- );
1930
- expect(el.dataSource.totalResults, 'total results').to.equal(1);
1931
- expect(el.dataSource.getTileModelAt(0)?.title).to.equal(
1932
- 'https://example.com',
1933
- );
1934
- expect(
1935
- el.dataSource.getTileModelAt(0)?.captureDates?.length,
1936
- 'capture dates',
1937
- ).to.equal(1);
1938
- });
1939
-
1940
- it('shows dropdown accordion in facet sidebar when opt-in strategy is specified', async () => {
1941
- const searchService = new MockSearchService();
1942
- const el = await fixture<CollectionBrowser>(
1943
- html`<collection-browser
1944
- .searchService=${searchService}
1945
- facetLoadStrategy=${'opt-in'}
1946
- >
1947
- </collection-browser>`,
1948
- );
1949
-
1950
- el.baseQuery = 'foo';
1951
- await el.updateComplete;
1952
- await el.initialSearchComplete;
1953
-
1954
- const facetsDropdown = el.shadowRoot?.querySelector(
1955
- '.desktop-facets-dropdown',
1956
- );
1957
- expect(facetsDropdown).to.exist;
1958
- });
1959
-
1960
- it('shows temporarily unavailable message when facets suppressed', async () => {
1961
- const searchService = new MockSearchService();
1962
- const el = await fixture<CollectionBrowser>(
1963
- html`<collection-browser
1964
- .searchService=${searchService}
1965
- facetLoadStrategy=${'off'}
1966
- >
1967
- </collection-browser>`,
1968
- );
1969
-
1970
- el.baseQuery = 'foo';
1971
- await el.updateComplete;
1972
- await el.initialSearchComplete;
1973
-
1974
- const facetsMsg = el.shadowRoot?.querySelector('.facets-message');
1975
- expect(facetsMsg).to.exist;
1976
- expect(facetsMsg?.textContent?.trim()).to.equal(
1977
- 'Facets are temporarily unavailable.',
1978
- );
1979
- });
1980
-
1981
- it('shows manage bar interface instead of sort bar when in manage view', async () => {
1982
- const searchService = new MockSearchService();
1983
- const el = await fixture<CollectionBrowser>(
1984
- html`<collection-browser .searchService=${searchService}>
1985
- </collection-browser>`,
1986
- );
1987
-
1988
- el.baseQuery = 'foo';
1989
- await el.updateComplete;
1990
- await el.initialSearchComplete;
1991
-
1992
- el.isManageView = true;
1993
- await el.updateComplete;
1994
-
1995
- expect(el.shadowRoot?.querySelector('manage-bar')).to.exist;
1996
- expect(el.shadowRoot?.querySelector('sort-filter-bar')).not.to.exist;
1997
-
1998
- el.isManageView = false;
1999
- await el.updateComplete;
2000
-
2001
- expect(el.shadowRoot?.querySelector('manage-bar')).not.to.exist;
2002
- expect(el.shadowRoot?.querySelector('sort-filter-bar')).to.exist;
2003
- });
2004
-
2005
- it('switches to grid display mode when manage view activated', async () => {
2006
- const searchService = new MockSearchService();
2007
- const el = await fixture<CollectionBrowser>(
2008
- html`<collection-browser
2009
- .searchService=${searchService}
2010
- .baseQuery=${'foo'}
2011
- .displayMode=${'list-detail'}
2012
- >
2013
- </collection-browser>`,
2014
- );
2015
-
2016
- el.isManageView = true;
2017
- await el.updateComplete;
2018
-
2019
- expect(el.displayMode).to.equal('grid');
2020
- });
2021
-
2022
- it('can remove all checked tiles', async () => {
2023
- const searchService = new MockSearchService();
2024
- const el = await fixture<CollectionBrowser>(
2025
- html`<collection-browser
2026
- .searchService=${searchService}
2027
- .baseNavigationUrl=${''}
2028
- >
2029
- </collection-browser>`,
2030
- );
2031
-
2032
- el.baseQuery = 'foo';
2033
- el.pageSize = 1; // To hit the edge case of a page break while offsetting tiles
2034
- await el.updateComplete;
2035
- await el.initialSearchComplete;
2036
-
2037
- el.isManageView = true;
2038
- await el.updateComplete;
2039
-
2040
- const infiniteScroller = el.shadowRoot?.querySelector('infinite-scroller');
2041
- expect(infiniteScroller).to.exist;
2042
-
2043
- let tiles =
2044
- infiniteScroller!.shadowRoot?.querySelectorAll('tile-dispatcher');
2045
- expect(tiles).to.exist;
2046
- expect(tiles?.length).to.equal(2);
2047
-
2048
- const firstTile = tiles![0] as TileDispatcher;
2049
- const firstTileLink = firstTile.shadowRoot?.querySelector(
2050
- 'a[href]',
2051
- ) as HTMLAnchorElement;
2052
- expect(firstTile.model?.identifier).to.equal('foo');
2053
- expect(firstTileLink).to.exist;
2054
-
2055
- // No effect if no tiles checked
2056
- el.removeCheckedTiles();
2057
- await el.updateComplete;
2058
- tiles = infiniteScroller!.shadowRoot?.querySelectorAll('tile-dispatcher');
2059
- expect(tiles).to.exist;
2060
- expect(tiles?.length).to.equal(2);
2061
-
2062
- // Check the first tile
2063
- firstTileLink!.click();
2064
- expect(firstTile.model?.checked).to.be.true;
2065
-
2066
- // Remove checked tiles and verify that we only kept the second tile
2067
- el.removeCheckedTiles();
2068
- await el.updateComplete;
2069
- expect(el?.dataSource?.size, 'data source count').to.equal(1);
2070
-
2071
- tiles = el.shadowRoot
2072
- ?.querySelector('infinite-scroller')!
2073
- .shadowRoot?.querySelectorAll('tile-dispatcher');
2074
- expect(tiles).to.exist;
2075
- expect(
2076
- tiles!.length,
2077
- 'tile count after `el.removeCheckedTiles()`',
2078
- ).to.equal(1);
2079
- });
2080
-
2081
- it('can check/uncheck all tiles', async () => {
2082
- const searchService = new MockSearchService();
2083
- const el = await fixture<CollectionBrowser>(
2084
- html`<collection-browser
2085
- .searchService=${searchService}
2086
- .baseNavigationUrl=${''}
2087
- >
2088
- </collection-browser>`,
2089
- );
2090
-
2091
- el.baseQuery = 'foo';
2092
- await el.updateComplete;
2093
- await el.initialSearchComplete;
2094
-
2095
- el.isManageView = true;
2096
- await el.updateComplete;
2097
-
2098
- expect(el.dataSource.checkedTileModels.length).to.equal(0);
2099
- expect(el.dataSource.uncheckedTileModels.length).to.equal(2);
2100
-
2101
- el.dataSource.checkAllTiles();
2102
- expect(el.dataSource.checkedTileModels.length).to.equal(2);
2103
- expect(el.dataSource.uncheckedTileModels.length).to.equal(0);
2104
-
2105
- el.dataSource.uncheckAllTiles();
2106
- expect(el.dataSource.checkedTileModels.length).to.equal(0);
2107
- expect(el.dataSource.uncheckedTileModels.length).to.equal(2);
2108
- });
2109
-
2110
- it('emits event when manage view state changes', async () => {
2111
- const spy = sinon.spy();
2112
- const searchService = new MockSearchService();
2113
- const el = await fixture<CollectionBrowser>(
2114
- html`<collection-browser
2115
- .searchService=${searchService}
2116
- .baseNavigationUrl=${''}
2117
- @manageModeChanged=${spy}
2118
- ></collection-browser>`,
2119
- );
2120
-
2121
- el.isManageView = true;
2122
- await el.updateComplete;
2123
-
2124
- expect(spy.callCount).to.equal(1);
2125
- expect(spy.args[0][0]?.detail).to.be.true;
2126
-
2127
- el.isManageView = false;
2128
- await el.updateComplete;
2129
-
2130
- expect(spy.callCount).to.equal(2);
2131
- expect(spy.args[1][0]?.detail).to.be.false;
2132
- });
2133
-
2134
- it('emits event when item removal requested', async () => {
2135
- const spy = sinon.spy();
2136
- const searchService = new MockSearchService();
2137
- const el = await fixture<CollectionBrowser>(
2138
- html`<collection-browser
2139
- .searchService=${searchService}
2140
- .baseNavigationUrl=${''}
2141
- @itemRemovalRequested=${spy}
2142
- >
2143
- </collection-browser>`,
2144
- );
2145
-
2146
- el.baseQuery = 'foo';
2147
- await el.updateComplete;
2148
- await el.initialSearchComplete;
2149
-
2150
- el.isManageView = true;
2151
- await el.updateComplete;
2152
-
2153
- const infiniteScroller = el.shadowRoot?.querySelector('infinite-scroller');
2154
- expect(infiniteScroller).to.exist;
2155
-
2156
- const tiles =
2157
- infiniteScroller!.shadowRoot?.querySelectorAll('tile-dispatcher');
2158
- expect(tiles).to.exist.and.have.length(2);
2159
-
2160
- const firstTile = tiles![0] as TileDispatcher;
2161
- const firstTileLink = firstTile.shadowRoot?.querySelector(
2162
- 'a[href]',
2163
- ) as HTMLAnchorElement;
2164
- expect(firstTile.model?.identifier).to.equal('foo');
2165
- expect(firstTileLink).to.exist;
2166
-
2167
- // Check the first tile
2168
- firstTileLink!.click();
2169
- await el.updateComplete;
2170
-
2171
- const manageBar = el.shadowRoot?.querySelector('manage-bar');
2172
- expect(manageBar).to.exist;
2173
-
2174
- // Emit remove event from manage bar
2175
- manageBar!.dispatchEvent(new CustomEvent('removeItems'));
2176
-
2177
- await el.updateComplete;
2178
- expect(spy.callCount).to.equal(1);
2179
- expect(spy.args[0].length).to.equal(1);
2180
- expect(spy.args[0][0].detail.items[0]).to.equal('foo');
2181
- });
2182
-
2183
- it('disables manage view when manage bar cancelled', async () => {
2184
- const searchService = new MockSearchService();
2185
- const el = await fixture<CollectionBrowser>(
2186
- html`<collection-browser
2187
- .searchService=${searchService}
2188
- .baseNavigationUrl=${''}
2189
- >
2190
- </collection-browser>`,
2191
- );
2192
-
2193
- el.baseQuery = 'foo';
2194
- await el.updateComplete;
2195
- await el.initialSearchComplete;
2196
-
2197
- el.isManageView = true;
2198
- await el.updateComplete;
2199
-
2200
- const manageBar = el.shadowRoot?.querySelector('manage-bar');
2201
- expect(manageBar).to.exist;
2202
-
2203
- // Emit remove event from manage bar
2204
- manageBar!.dispatchEvent(new CustomEvent('cancel'));
2205
-
2206
- await el.updateComplete;
2207
- expect(el.isManageView).to.be.false;
2208
- });
2209
-
2210
- it('enable/disable manage view delete button when you selectAll/unselectAll', async () => {
2211
- const searchService = new MockSearchService();
2212
- const el = await fixture<CollectionBrowser>(
2213
- html`<collection-browser .searchService=${searchService}>
2214
- </collection-browser>`,
2215
- );
2216
-
2217
- el.baseQuery = 'foo';
2218
- await el.updateComplete;
2219
- await el.initialSearchComplete;
2220
-
2221
- el.isManageView = true;
2222
- await el.updateComplete;
2223
-
2224
- const manageBar = el.shadowRoot?.querySelector('manage-bar');
2225
- expect(manageBar).to.exist;
2226
-
2227
- // disable button exists
2228
- expect(manageBar?.shadowRoot?.querySelector('.danger:disabled')).to.be
2229
- .exist;
2230
-
2231
- // Emit remove event from manage bar
2232
- manageBar!.dispatchEvent(new CustomEvent('selectAll'));
2233
- await el.updateComplete;
2234
-
2235
- // disable button does not exists
2236
- expect(manageBar?.shadowRoot?.querySelector('.danger:disabled')).to.be.not
2237
- .exist;
2238
-
2239
- // Emit remove event from manage bar
2240
- manageBar!.dispatchEvent(new CustomEvent('unselectAll'));
2241
- await el.updateComplete;
2242
-
2243
- // disable button exists again
2244
- expect(manageBar?.shadowRoot?.querySelector('.danger:disabled')).to.be
2245
- .exist;
2246
- });
2247
-
2248
- it('shows Blurring checkbox for admin users', async () => {
2249
- const searchService = new MockSearchService();
2250
- const el = await fixture<CollectionBrowser>(
2251
- html`<collection-browser
2252
- .baseNavigationUrl=${''}
2253
- .searchService=${searchService}
2254
- >
2255
- </collection-browser>`,
2256
- );
2257
-
2258
- el.baseQuery = 'archive-org-user-loggedin';
2259
- await el.updateComplete;
2260
- await el.initialSearchComplete;
2261
-
2262
- const blurringCheck = el.shadowRoot?.querySelector(
2263
- '#tile-blur-check',
2264
- ) as HTMLInputElement;
2265
- expect(blurringCheck).to.exist;
2266
- expect(blurringCheck.checked).to.be.true;
2267
- });
2268
-
2269
- it('unchecks Blurring checkbox for admin users with blurring preference off', async () => {
2270
- const searchService = new MockSearchService();
2271
- const el = await fixture<CollectionBrowser>(
2272
- html`<collection-browser
2273
- .baseNavigationUrl=${''}
2274
- .searchService=${searchService}
2275
- >
2276
- </collection-browser>`,
2277
- );
2278
-
2279
- el.baseQuery = 'archive-org-user-loggedin-noblur';
2280
- await el.updateComplete;
2281
- await el.initialSearchComplete;
2282
-
2283
- const blurringCheck = el.shadowRoot?.querySelector(
2284
- '#tile-blur-check',
2285
- ) as HTMLInputElement;
2286
- expect(blurringCheck).to.exist;
2287
- expect(blurringCheck.checked).to.be.false;
2288
- });
2289
-
2290
- it('toggles blur state when Blurring checkbox is toggled', async () => {
2291
- const searchService = new MockSearchService();
2292
- const el = await fixture<CollectionBrowser>(
2293
- html`<collection-browser
2294
- .baseNavigationUrl=${''}
2295
- .searchService=${searchService}
2296
- >
2297
- </collection-browser>`,
2298
- );
2299
-
2300
- el.baseQuery = 'archive-org-user-loggedin';
2301
- await el.updateComplete;
2302
- await el.initialSearchComplete;
2303
-
2304
- const blurringCheck = el.shadowRoot?.querySelector(
2305
- '#tile-blur-check',
2306
- ) as HTMLInputElement;
2307
- expect(blurringCheck).to.exist;
2308
-
2309
- blurringCheck.dispatchEvent(new PointerEvent('click'));
2310
- await el.updateComplete;
2311
-
2312
- const infiniteScroller = el.shadowRoot?.querySelector(
2313
- 'infinite-scroller',
2314
- ) as InfiniteScroller;
2315
- const firstTile = infiniteScroller?.shadowRoot?.querySelector(
2316
- 'tile-dispatcher',
2317
- ) as TileDispatcher;
2318
- expect(firstTile.suppressBlurring).to.be.true;
2319
- });
2320
-
2321
- it('applies loans tab properties to sort bar', async () => {
2322
- const searchService = new MockSearchService();
2323
- const el = await fixture<CollectionBrowser>(
2324
- html`<collection-browser
2325
- .baseNavigationUrl=${''}
2326
- .searchService=${searchService}
2327
- .enableSortOptionsSlot=${true}
2328
- >
2329
- </collection-browser>`,
2330
- );
2331
-
2332
- el.baseQuery = 'collection:foo';
2333
- await el.updateComplete;
2334
- await aTimeout(10);
2335
-
2336
- const sortBar = el.shadowRoot?.querySelector(
2337
- 'sort-filter-bar',
2338
- ) as SortFilterBar;
2339
- expect(sortBar?.enableSortOptionsSlot, 'show loans in sort bar').to.be.true;
2340
- expect(el.enableSortOptionsSlot, 'collection browser is loans tab').to.be
2341
- .true;
2342
-
2343
- const loansTabSlot = sortBar.querySelector('slot[name="sort-options"]');
2344
- expect(loansTabSlot).to.exist;
2345
- });
2346
-
2347
- it('can suppress presence of result count', async () => {
2348
- const searchService = new MockSearchService();
2349
- const el = await fixture<CollectionBrowser>(
2350
- html`<collection-browser
2351
- .searchService=${searchService}
2352
- suppressResultCount
2353
- ></collection-browser>`,
2354
- );
2355
-
2356
- el.baseQuery = 'collection:foo';
2357
- await el.updateComplete;
2358
- await el.initialSearchComplete;
2359
-
2360
- const resultCount = el.shadowRoot?.querySelector('#results-total');
2361
- expect(resultCount).not.to.exist;
2362
- });
2363
-
2364
- it('can suppress presence of result tiles', async () => {
2365
- const searchService = new MockSearchService();
2366
- const el = await fixture<CollectionBrowser>(
2367
- html`<collection-browser
2368
- .searchService=${searchService}
2369
- suppressResultTiles
2370
- ></collection-browser>`,
2371
- );
2372
-
2373
- el.baseQuery = 'collection:foo';
2374
- await el.updateComplete;
2375
-
2376
- const infiniteScroller = el.shadowRoot?.querySelector('infinite-scroller');
2377
- expect(infiniteScroller).not.to.exist;
2378
- });
2379
-
2380
- it('fetch larger result on search page for admin user to manage items', async () => {
2381
- const resultsSpy = sinon.spy();
2382
- const searchService = new MockSearchService({
2383
- asyncResponse: true,
2384
- resultsSpy,
2385
- });
2386
-
2387
- const el = await fixture<CollectionBrowser>(
2388
- html`<collection-browser .searchService=${searchService}>
2389
- </collection-browser>`,
2390
- );
2391
-
2392
- const numberOfPages = 15;
2393
-
2394
- el.baseQuery = 'jack';
2395
- el.isManageView = true;
2396
- await el.dataSource.fetchPage(1, numberOfPages);
2397
- await el.updateComplete;
2398
-
2399
- const initialResults = el.dataSource.getAllPages();
2400
- expect(Object.keys(initialResults).length).to.deep.equal(numberOfPages);
2401
- });
2402
- });
1
+ import { aTimeout, expect, fixture } from '@open-wc/testing';
2
+ import { html } from 'lit';
3
+ import sinon from 'sinon';
4
+ import type { InfiniteScroller } from '@internetarchive/infinite-scroller';
5
+ import { FilterConstraint, SearchType } from '@internetarchive/search-service';
6
+ import type { HistogramDateRange } from '@internetarchive/histogram-date-range';
7
+ import type { CollectionBrowser } from '../src/collection-browser';
8
+ import '../src/collection-browser';
9
+ import {
10
+ getDefaultSelectedFacets,
11
+ FacetBucket,
12
+ SelectedFacets,
13
+ SortField,
14
+ } from '../src/models';
15
+ import { MockSearchService } from './mocks/mock-search-service';
16
+ import { MockAnalyticsHandler } from './mocks/mock-analytics-handler';
17
+ import {
18
+ analyticsActions,
19
+ analyticsCategories,
20
+ } from '../src/utils/analytics-events';
21
+ import type { TileDispatcher } from '../src/tiles/tile-dispatcher';
22
+ import type { CollectionFacets } from '../src/collection-facets';
23
+ import type { EmptyPlaceholder } from '../src/empty-placeholder';
24
+ import type { SortFilterBar } from '../src/sort-filter-bar/sort-filter-bar';
25
+
26
+ /**
27
+ * Wait for the next tick of the event loop.
28
+ *
29
+ * This is necessary in some of the tests because certain collection browser
30
+ * updates take more than one tick to render (e.g., date picker & query changes).
31
+ * These delays are non-ideal and should eventually be investigated and fixed,
32
+ * but they are minor enough that waiting for the next tick is a reasonable
33
+ * testing solution for now.
34
+ */
35
+ const nextTick = () => aTimeout(0);
36
+
37
+ describe('Collection Browser', () => {
38
+ beforeEach(async () => {
39
+ // Apparently query params set by one test can bleed into other tests.
40
+ // Since collection browser restores its state from certain query params, we need
41
+ // to clear these before each test to ensure they run in isolation from one another.
42
+ const url = new URL(window.location.href);
43
+ const { searchParams } = url;
44
+ searchParams.delete('sin');
45
+ searchParams.delete('sort');
46
+ searchParams.delete('query');
47
+ searchParams.delete('page');
48
+ searchParams.delete('and[]');
49
+ searchParams.delete('not[]');
50
+ window.history.replaceState({}, '', url);
51
+ });
52
+
53
+ it('clears selected facets when requested', async () => {
54
+ const selectedFacets = getDefaultSelectedFacets();
55
+ selectedFacets.creator.foo = { count: 1, key: 'foo', state: 'selected' };
56
+ const el = await fixture<CollectionBrowser>(
57
+ html`<collection-browser></collection-browser>`,
58
+ );
59
+
60
+ el.selectedFacets = selectedFacets;
61
+ await el.updateComplete;
62
+ el.clearFilters(); // By default, sort is not cleared
63
+
64
+ expect(el.selectedFacets).to.deep.equal(getDefaultSelectedFacets());
65
+ });
66
+
67
+ it('clears existing filters but not sort by default', async () => {
68
+ const el = await fixture<CollectionBrowser>(
69
+ html`<collection-browser></collection-browser>`,
70
+ );
71
+
72
+ el.selectedSort = 'title' as SortField;
73
+ el.sortDirection = 'asc';
74
+ await el.updateComplete;
75
+ el.clearFilters(); // By default, sort is not cleared
76
+
77
+ expect(el.selectedFacets).to.deep.equal(getDefaultSelectedFacets());
78
+ expect(el.selectedSort).to.equal('title');
79
+ expect(el.sortDirection).to.equal('asc');
80
+ expect(el.sortParam).to.deep.equal({
81
+ field: 'titleSorter',
82
+ direction: 'asc',
83
+ });
84
+ expect(el.selectedCreatorFilter).to.be.null;
85
+ expect(el.selectedTitleFilter).to.be.null;
86
+ });
87
+
88
+ it('clears existing filters for facets & sort via option', async () => {
89
+ const el = await fixture<CollectionBrowser>(
90
+ html`<collection-browser></collection-browser>`,
91
+ );
92
+
93
+ el.selectedSort = 'title' as SortField;
94
+ await el.updateComplete;
95
+ el.clearFilters({ sort: true }); // Sort is reset too due to the option
96
+
97
+ expect(el.selectedFacets).to.deep.equal(getDefaultSelectedFacets());
98
+ expect(el.selectedSort).to.equal(SortField.default);
99
+ expect(el.sortDirection).to.be.null;
100
+ expect(el.sortParam).to.be.null;
101
+ expect(el.selectedCreatorFilter).to.be.null;
102
+ expect(el.selectedTitleFilter).to.be.null;
103
+ });
104
+
105
+ it('filterBy creator with analytics', async () => {
106
+ const mockAnalyticsHandler = new MockAnalyticsHandler();
107
+ const el = await fixture<CollectionBrowser>(
108
+ html`<collection-browser .analyticsHandler=${mockAnalyticsHandler}>
109
+ </collection-browser>`,
110
+ );
111
+
112
+ el.searchContext = 'betaSearchService';
113
+ el.selectedSort = 'creator' as SortField;
114
+ el.sortDirection = 'asc';
115
+ el.selectedCreatorFilter = 'A';
116
+ await el.updateComplete;
117
+
118
+ expect(mockAnalyticsHandler.callCategory).to.equal('betaSearchService');
119
+ expect(mockAnalyticsHandler.callAction).to.equal('filterByCreator');
120
+ expect(mockAnalyticsHandler.callLabel).to.equal('start-A');
121
+
122
+ el.clearFilters();
123
+ await el.updateComplete;
124
+
125
+ expect(el.selectedTitleFilter).to.be.null;
126
+ expect(mockAnalyticsHandler.callCategory).to.equal('betaSearchService');
127
+ expect(mockAnalyticsHandler.callAction).to.equal('filterByCreator');
128
+ expect(mockAnalyticsHandler.callLabel).to.equal('clear-A');
129
+ });
130
+
131
+ it('filterBy title with analytics', async () => {
132
+ const mockAnalyticsHandler = new MockAnalyticsHandler();
133
+ const el = await fixture<CollectionBrowser>(
134
+ html`<collection-browser .analyticsHandler=${mockAnalyticsHandler}>
135
+ </collection-browser>`,
136
+ );
137
+
138
+ el.searchContext = 'beta-search-service';
139
+ el.selectedSort = 'title' as SortField;
140
+ el.sortDirection = 'asc';
141
+ el.selectedTitleFilter = 'A';
142
+ await el.updateComplete;
143
+
144
+ expect(mockAnalyticsHandler.callCategory).to.equal('beta-search-service');
145
+ expect(mockAnalyticsHandler.callAction).to.equal('filterByTitle');
146
+ expect(mockAnalyticsHandler.callLabel).to.equal('start-A');
147
+
148
+ el.clearFilters();
149
+ await el.updateComplete;
150
+
151
+ expect(el.selectedTitleFilter).to.be.null;
152
+ expect(mockAnalyticsHandler.callCategory).to.equal('beta-search-service');
153
+ expect(mockAnalyticsHandler.callAction).to.equal('filterByTitle');
154
+ expect(mockAnalyticsHandler.callLabel).to.equal('clear-A');
155
+ });
156
+
157
+ it('selected facets with analytics - not negative facets', async () => {
158
+ const mockAnalyticsHandler = new MockAnalyticsHandler();
159
+ const mediaTypeBucket = { count: 123, state: 'selected' } as FacetBucket;
160
+ const mockedSelectedFacets: SelectedFacets = {
161
+ subject: {},
162
+ lending: {},
163
+ mediatype: { data: mediaTypeBucket },
164
+ language: {},
165
+ creator: {},
166
+ collection: {},
167
+ year: {},
168
+ };
169
+
170
+ const el = await fixture<CollectionBrowser>(
171
+ html`<collection-browser .analyticsHandler=${mockAnalyticsHandler}>
172
+ </collection-browser>`,
173
+ );
174
+
175
+ el.searchContext = 'search-service';
176
+ el.selectedFacets = mockedSelectedFacets;
177
+ await el.updateComplete;
178
+
179
+ el.facetClickHandler(
180
+ new CustomEvent('facetClick', {
181
+ detail: {
182
+ facetType: 'mediatype',
183
+ bucket: {
184
+ key: '',
185
+ state: 'selected',
186
+ count: 123,
187
+ },
188
+ negative: false,
189
+ },
190
+ }),
191
+ );
192
+ expect(mockAnalyticsHandler.callCategory).to.equal('search-service');
193
+ expect(mockAnalyticsHandler.callAction).to.equal('facetSelected');
194
+ expect(mockAnalyticsHandler.callLabel).to.equal('mediatype');
195
+
196
+ el.facetClickHandler(
197
+ new CustomEvent('facetClick', {
198
+ detail: {
199
+ facetType: 'mediatype',
200
+ bucket: {
201
+ key: '',
202
+ state: 'none',
203
+ count: 123,
204
+ },
205
+ negative: false,
206
+ },
207
+ }),
208
+ );
209
+ expect(el.selectedFacets).to.equal(mockedSelectedFacets);
210
+ expect(mockAnalyticsHandler.callCategory).to.equal('search-service');
211
+ expect(mockAnalyticsHandler.callAction).to.equal('facetDeselected');
212
+ expect(mockAnalyticsHandler.callLabel).to.equal('mediatype');
213
+ });
214
+
215
+ it('selected facets with analytics - negative facets', async () => {
216
+ const mockAnalyticsHandler = new MockAnalyticsHandler();
217
+ const mediaTypeBucket = { count: 123, state: 'selected' } as FacetBucket;
218
+ const mockedSelectedFacets: SelectedFacets = {
219
+ subject: {},
220
+ lending: {},
221
+ mediatype: { data: mediaTypeBucket },
222
+ language: {},
223
+ creator: {},
224
+ collection: {},
225
+ year: {},
226
+ };
227
+
228
+ const el = await fixture<CollectionBrowser>(
229
+ html`<collection-browser .analyticsHandler=${mockAnalyticsHandler}>
230
+ </collection-browser>`,
231
+ );
232
+
233
+ el.searchContext = 'beta-search-service';
234
+ el.selectedFacets = mockedSelectedFacets;
235
+ await el.updateComplete;
236
+
237
+ el.facetClickHandler(
238
+ new CustomEvent('facetClick', {
239
+ detail: {
240
+ facetType: 'mediatype',
241
+ bucket: {
242
+ key: '',
243
+ state: 'hidden',
244
+ count: 123,
245
+ },
246
+ negative: true,
247
+ },
248
+ }),
249
+ );
250
+ expect(mockAnalyticsHandler.callCategory).to.equal('beta-search-service');
251
+ expect(mockAnalyticsHandler.callAction).to.equal('facetNegativeSelected');
252
+ expect(mockAnalyticsHandler.callLabel).to.equal('mediatype');
253
+
254
+ el.facetClickHandler(
255
+ new CustomEvent('facetClick', {
256
+ detail: {
257
+ facetType: 'mediatype',
258
+ bucket: {
259
+ key: '',
260
+ state: 'none',
261
+ count: 123,
262
+ },
263
+ negative: true,
264
+ },
265
+ }),
266
+ );
267
+ expect(el.selectedFacets).to.equal(mockedSelectedFacets);
268
+ expect(mockAnalyticsHandler.callCategory).to.equal('beta-search-service');
269
+ expect(mockAnalyticsHandler.callAction).to.equal('facetNegativeDeselected');
270
+ expect(mockAnalyticsHandler.callLabel).to.equal('mediatype');
271
+ });
272
+
273
+ it('should render with a sort bar, facets, and infinite scroller', async () => {
274
+ const searchService = new MockSearchService();
275
+
276
+ const el = await fixture<CollectionBrowser>(
277
+ html`<collection-browser .searchService=${searchService}>
278
+ </collection-browser>`,
279
+ );
280
+
281
+ el.baseQuery = 'hello';
282
+ await el.updateComplete;
283
+ await nextTick();
284
+
285
+ const facets = el.shadowRoot?.querySelector('collection-facets');
286
+ const sortBar = el.shadowRoot?.querySelector('sort-filter-bar');
287
+ const infiniteScroller = el.shadowRoot?.querySelector('infinite-scroller');
288
+ expect(facets, 'facets').to.exist;
289
+ expect(sortBar, 'sort bar').to.exist;
290
+ expect(infiniteScroller, 'infinite scroller').to.exist;
291
+ });
292
+
293
+ it('queries the search service when given a base query', async () => {
294
+ const searchService = new MockSearchService();
295
+
296
+ const el = await fixture<CollectionBrowser>(
297
+ html`<collection-browser .searchService=${searchService}>
298
+ </collection-browser>`,
299
+ );
300
+
301
+ el.baseQuery = 'collection:foo';
302
+ await el.updateComplete;
303
+ await el.initialSearchComplete;
304
+
305
+ expect(searchService.searchParams?.query).to.equal('collection:foo');
306
+ expect(
307
+ el.shadowRoot?.querySelector('#big-results-label')?.textContent,
308
+ ).to.contains('Results');
309
+ });
310
+
311
+ it('queries the search service when given a list of identifiers and no query', async () => {
312
+ const searchService = new MockSearchService();
313
+
314
+ const el = await fixture<CollectionBrowser>(
315
+ html`<collection-browser .searchService=${searchService}>
316
+ </collection-browser>`,
317
+ );
318
+
319
+ el.identifiers = ['foo', 'bar'];
320
+ await el.updateComplete;
321
+ await el.initialSearchComplete;
322
+
323
+ expect(searchService.searchParams?.identifiers).to.deep.equal([
324
+ 'foo',
325
+ 'bar',
326
+ ]);
327
+ expect(
328
+ el.shadowRoot?.querySelector('#big-results-label')?.textContent,
329
+ ).to.contains('Results');
330
+ });
331
+
332
+ it('queries the search service when given a list of identifiers with a query', async () => {
333
+ const searchService = new MockSearchService();
334
+
335
+ const el = await fixture<CollectionBrowser>(
336
+ html`<collection-browser .searchService=${searchService}>
337
+ </collection-browser>`,
338
+ );
339
+
340
+ el.baseQuery = 'collection:foo';
341
+ el.identifiers = ['foo', 'bar'];
342
+ await el.updateComplete;
343
+ await el.initialSearchComplete;
344
+
345
+ expect(searchService.searchParams?.query).to.equal('collection:foo');
346
+ expect(searchService.searchParams?.identifiers).to.deep.equal([
347
+ 'foo',
348
+ 'bar',
349
+ ]);
350
+ expect(
351
+ el.shadowRoot?.querySelector('#big-results-label')?.textContent,
352
+ ).to.contains('Results');
353
+ });
354
+
355
+ it('queries the search service with a metadata search', async () => {
356
+ const searchService = new MockSearchService();
357
+
358
+ const el = await fixture<CollectionBrowser>(
359
+ html` <collection-browser .searchService=${searchService}>
360
+ </collection-browser>`,
361
+ );
362
+
363
+ el.searchType = SearchType.METADATA;
364
+ await el.updateComplete;
365
+
366
+ el.baseQuery = 'collection:foo';
367
+ await el.updateComplete;
368
+ await el.initialSearchComplete;
369
+
370
+ expect(searchService.searchParams?.query).to.equal('collection:foo');
371
+ expect(searchService.searchType).to.equal(SearchType.METADATA);
372
+ expect(
373
+ el.shadowRoot?.querySelector('#big-results-label')?.textContent,
374
+ ).to.contains('Results');
375
+ });
376
+
377
+ it('can change search type', async () => {
378
+ const searchService = new MockSearchService();
379
+ const el = await fixture<CollectionBrowser>(
380
+ html`<collection-browser .searchService=${searchService}>
381
+ </collection-browser>`,
382
+ );
383
+
384
+ el.baseQuery = 'collection:foo';
385
+ await el.updateComplete;
386
+
387
+ el.searchType = SearchType.FULLTEXT;
388
+ await el.updateComplete;
389
+ await el.initialSearchComplete;
390
+
391
+ expect(searchService.searchType).to.equal(SearchType.FULLTEXT);
392
+ });
393
+
394
+ it('queries the search service with a fulltext search', async () => {
395
+ const searchService = new MockSearchService();
396
+
397
+ const el = await fixture<CollectionBrowser>(
398
+ html` <collection-browser .searchService=${searchService}>
399
+ </collection-browser>`,
400
+ );
401
+
402
+ el.searchType = SearchType.FULLTEXT;
403
+ await el.updateComplete;
404
+
405
+ el.baseQuery = 'collection:foo';
406
+ await el.updateComplete;
407
+ await el.initialSearchComplete;
408
+
409
+ expect(searchService.searchParams?.query).to.equal('collection:foo');
410
+ expect(searchService.searchType).to.equal(SearchType.FULLTEXT);
411
+ expect(
412
+ el.shadowRoot?.querySelector('#big-results-label')?.textContent,
413
+ ).to.contains('Results');
414
+ });
415
+
416
+ it('queries the search service with a radio search', async () => {
417
+ const searchService = new MockSearchService();
418
+
419
+ const el = await fixture<CollectionBrowser>(
420
+ html` <collection-browser .searchService=${searchService}>
421
+ </collection-browser>`,
422
+ );
423
+
424
+ el.searchType = SearchType.RADIO;
425
+ await el.updateComplete;
426
+
427
+ el.baseQuery = 'collection:foo';
428
+ await el.updateComplete;
429
+ await el.initialSearchComplete;
430
+
431
+ expect(searchService.searchParams?.query).to.equal('collection:foo');
432
+ expect(searchService.searchType).to.equal(SearchType.RADIO);
433
+ expect(
434
+ el.shadowRoot?.querySelector('#big-results-label')?.textContent,
435
+ ).to.contains('Results');
436
+ });
437
+
438
+ it('queries the search service with a TV search', async () => {
439
+ const searchService = new MockSearchService();
440
+
441
+ const el = await fixture<CollectionBrowser>(
442
+ html` <collection-browser .searchService=${searchService}>
443
+ </collection-browser>`,
444
+ );
445
+
446
+ el.searchType = SearchType.TV;
447
+ await el.updateComplete;
448
+
449
+ el.baseQuery = 'collection:foo';
450
+ await el.updateComplete;
451
+ await el.initialSearchComplete;
452
+
453
+ expect(searchService.searchParams?.query).to.equal('collection:foo');
454
+ expect(searchService.searchType).to.equal(SearchType.TV);
455
+ expect(
456
+ el.shadowRoot?.querySelector('#big-results-label')?.textContent,
457
+ ).to.contains('Results');
458
+ });
459
+
460
+ it('queries the search service with facets selected/negated', async () => {
461
+ const searchService = new MockSearchService();
462
+ const selectedFacets: SelectedFacets = {
463
+ subject: {
464
+ foo: {
465
+ key: 'foo',
466
+ count: 1,
467
+ state: 'selected',
468
+ },
469
+ bar: {
470
+ key: 'bar',
471
+ count: 2,
472
+ state: 'hidden',
473
+ },
474
+ },
475
+ lending: {},
476
+ mediatype: {},
477
+ language: {
478
+ en: {
479
+ key: 'en',
480
+ count: 1,
481
+ state: 'selected',
482
+ },
483
+ },
484
+ creator: {},
485
+ collection: {},
486
+ year: {},
487
+ };
488
+
489
+ const el = await fixture<CollectionBrowser>(
490
+ html`<collection-browser .searchService=${searchService}>
491
+ </collection-browser>`,
492
+ );
493
+
494
+ el.baseQuery = 'collection:foo';
495
+ el.selectedFacets = selectedFacets;
496
+ await el.updateComplete;
497
+ await el.initialSearchComplete;
498
+
499
+ expect(searchService.searchParams?.query).to.equal('collection:foo');
500
+ expect(searchService.searchParams?.filters).to.deep.equal({
501
+ subject: {
502
+ foo: 'inc',
503
+ bar: 'exc',
504
+ },
505
+ language: {
506
+ en: 'inc',
507
+ },
508
+ });
509
+ });
510
+
511
+ it('fails gracefully if no search service provided', async () => {
512
+ const el = await fixture<CollectionBrowser>(
513
+ html`<collection-browser></collection-browser>`,
514
+ );
515
+
516
+ el.baseQuery = 'collection:foo';
517
+ await el.updateComplete;
518
+
519
+ // This shouldn't throw an error
520
+ expect(el.dataSource.fetchPage(3)).to.exist;
521
+
522
+ // Should continue showing the empty placeholder
523
+ expect(el.shadowRoot?.querySelector('empty-placeholder')).to.exist;
524
+ });
525
+
526
+ it('restores search type from URL param', async () => {
527
+ // Add a sin=TXT param to the URL
528
+ const url = new URL(window.location.href);
529
+ url.searchParams.append('sin', 'TXT');
530
+ window.history.replaceState({}, '', url);
531
+
532
+ const searchService = new MockSearchService();
533
+
534
+ const el = await fixture<CollectionBrowser>(
535
+ html`<collection-browser .searchService=${searchService}>
536
+ </collection-browser>`,
537
+ );
538
+
539
+ expect(el.searchType).to.equal(SearchType.FULLTEXT);
540
+ });
541
+
542
+ it('does not persist or restore search type from URL param if suppressed', async () => {
543
+ // Add a sin=TXT param to the URL
544
+ let url = new URL(window.location.href);
545
+ url.searchParams.append('sin', 'TXT');
546
+ window.history.replaceState({}, '', url);
547
+
548
+ const searchService = new MockSearchService();
549
+
550
+ const el = await fixture<CollectionBrowser>(
551
+ html`<collection-browser
552
+ .searchService=${searchService}
553
+ suppressURLSinParam
554
+ >
555
+ </collection-browser>`,
556
+ );
557
+
558
+ url = new URL(window.location.href);
559
+ expect(el.searchType).to.equal(SearchType.DEFAULT);
560
+ expect(url.searchParams.has('sin')).to.be.false; // Removes existing sin param
561
+
562
+ el.searchType = SearchType.RADIO;
563
+ await el.updateComplete;
564
+
565
+ url = new URL(window.location.href);
566
+ expect(url.searchParams.has('sin')).to.be.false; // Doesn't add sin param
567
+ });
568
+
569
+ it('can construct tile models with many fields present', async () => {
570
+ const searchService = new MockSearchService();
571
+
572
+ const el = await fixture<CollectionBrowser>(
573
+ html`<collection-browser .searchService=${searchService}>
574
+ </collection-browser>`,
575
+ );
576
+
577
+ el.baseQuery = 'many-fields';
578
+ await el.updateComplete;
579
+ await el.initialSearchComplete;
580
+
581
+ const cellTemplate = el.cellForIndex(0);
582
+ expect(cellTemplate).to.exist;
583
+
584
+ const cell = await fixture<TileDispatcher>(cellTemplate!);
585
+ expect(cell).to.exist;
586
+ });
587
+
588
+ it('emits empty results event when search fetches no results', async () => {
589
+ const searchService = new MockSearchService();
590
+ const emptyResultsSpy = sinon.spy();
591
+
592
+ const el = await fixture<CollectionBrowser>(
593
+ html`<collection-browser
594
+ .searchService=${searchService}
595
+ @emptyResults=${emptyResultsSpy}
596
+ >
597
+ </collection-browser>`,
598
+ );
599
+
600
+ el.baseQuery = 'no-results';
601
+ await el.updateComplete;
602
+ await el.initialSearchComplete;
603
+
604
+ expect(emptyResultsSpy.callCount).to.equal(1);
605
+ });
606
+
607
+ it('emits searchError event when search results in an error', async () => {
608
+ const searchService = new MockSearchService();
609
+ const searchErrorSpy = sinon.spy();
610
+
611
+ const el = await fixture<CollectionBrowser>(
612
+ html`<collection-browser
613
+ .searchService=${searchService}
614
+ @searchError=${searchErrorSpy}
615
+ >
616
+ </collection-browser>`,
617
+ );
618
+
619
+ el.baseQuery = 'error';
620
+ await el.updateComplete;
621
+ await el.initialSearchComplete;
622
+
623
+ expect(searchErrorSpy.callCount).to.equal(1);
624
+ });
625
+
626
+ it('applies loggedin flag to tile models if needed', async () => {
627
+ const searchService = new MockSearchService();
628
+
629
+ const el = await fixture<CollectionBrowser>(
630
+ html`<collection-browser .searchService=${searchService}>
631
+ </collection-browser>`,
632
+ );
633
+
634
+ el.baseQuery = 'loggedin';
635
+ await el.updateComplete;
636
+ await el.initialSearchComplete;
637
+
638
+ const cellTemplate = el.cellForIndex(0);
639
+ expect(cellTemplate).to.exist;
640
+
641
+ const cell = await fixture<TileDispatcher>(cellTemplate!);
642
+ expect(cell).to.exist;
643
+
644
+ expect(cell.model?.loginRequired).to.be.true;
645
+ });
646
+
647
+ it('applies no-preview flag to tile models if needed', async () => {
648
+ const searchService = new MockSearchService();
649
+
650
+ const el = await fixture<CollectionBrowser>(
651
+ html`<collection-browser .searchService=${searchService}>
652
+ </collection-browser>`,
653
+ );
654
+
655
+ el.baseQuery = 'no-preview';
656
+ await el.updateComplete;
657
+ await el.initialSearchComplete;
658
+
659
+ const cellTemplate = el.cellForIndex(0);
660
+ expect(cellTemplate).to.exist;
661
+
662
+ const cell = await fixture<TileDispatcher>(cellTemplate!);
663
+ expect(cell).to.exist;
664
+
665
+ expect(cell.model?.contentWarning).to.be.true;
666
+ });
667
+
668
+ it('both loggedin and no-preview flags can be set simultaneously', async () => {
669
+ const searchService = new MockSearchService();
670
+
671
+ const el = await fixture<CollectionBrowser>(
672
+ html`<collection-browser .searchService=${searchService}>
673
+ </collection-browser>`,
674
+ );
675
+
676
+ el.baseQuery = 'loggedin-no-preview';
677
+ await el.updateComplete;
678
+ await el.initialSearchComplete;
679
+
680
+ const cellTemplate = el.cellForIndex(0);
681
+ expect(cellTemplate).to.exist;
682
+
683
+ const cell = await fixture<TileDispatcher>(cellTemplate!);
684
+ expect(cell).to.exist;
685
+
686
+ expect(cell.model?.loginRequired).to.be.true;
687
+ expect(cell.model?.contentWarning).to.be.true;
688
+ });
689
+
690
+ it('joins full description array into a single string with line breaks', async () => {
691
+ const searchService = new MockSearchService();
692
+
693
+ const el = await fixture<CollectionBrowser>(
694
+ html`<collection-browser .searchService=${searchService}>
695
+ </collection-browser>`,
696
+ );
697
+
698
+ // This query receives an array description like ['line1', 'line2']
699
+ el.baseQuery = 'multi-line-description';
700
+ await el.updateComplete;
701
+ await el.initialSearchComplete;
702
+
703
+ const cellTemplate = el.cellForIndex(0);
704
+ expect(cellTemplate).to.exist;
705
+
706
+ const cell = await fixture<TileDispatcher>(cellTemplate!);
707
+ expect(cell).to.exist;
708
+
709
+ // Actual model description should be joined
710
+ expect(cell.model?.description).to.equal('line1\nline2');
711
+ });
712
+
713
+ it('can change search type', async () => {
714
+ const searchService = new MockSearchService();
715
+
716
+ const el = await fixture<CollectionBrowser>(
717
+ html`<collection-browser
718
+ .searchService=${searchService}
719
+ .searchType=${SearchType.METADATA}
720
+ ></collection-browser>`,
721
+ );
722
+
723
+ el.baseQuery = 'collection:foo';
724
+ el.searchType = SearchType.FULLTEXT;
725
+ await el.updateComplete;
726
+ await el.initialSearchComplete;
727
+
728
+ expect(searchService.searchParams?.query).to.equal('collection:foo');
729
+ expect(searchService.searchType).to.equal(SearchType.FULLTEXT);
730
+ });
731
+
732
+ it('trims queries of leading/trailing whitespace', async () => {
733
+ const searchService = new MockSearchService();
734
+ const el = await fixture<CollectionBrowser>(
735
+ html`<collection-browser
736
+ .searchService=${searchService}
737
+ ></collection-browser>`,
738
+ );
739
+
740
+ el.baseQuery = ' collection:foo ';
741
+ await el.updateComplete;
742
+ await el.initialSearchComplete;
743
+
744
+ expect(searchService.searchParams?.query).to.equal('collection:foo');
745
+ });
746
+
747
+ it('shows error message when error response received', async () => {
748
+ const searchService = new MockSearchService();
749
+ const el = await fixture<CollectionBrowser>(
750
+ html`<collection-browser
751
+ .searchService=${searchService}
752
+ ></collection-browser>`,
753
+ );
754
+
755
+ el.baseQuery = 'error';
756
+ await el.updateComplete;
757
+ await el.initialSearchComplete;
758
+
759
+ const errorPlaceholder = el.shadowRoot?.querySelector(
760
+ 'empty-placeholder',
761
+ ) as EmptyPlaceholder;
762
+ const errorDetails = errorPlaceholder?.shadowRoot?.querySelector(
763
+ '.error-details',
764
+ ) as HTMLParagraphElement;
765
+
766
+ expect(errorDetails).to.exist;
767
+ expect(errorDetails.textContent).to.contain('foo');
768
+ });
769
+
770
+ it('shows error message when error response received for a collection', async () => {
771
+ const searchService = new MockSearchService();
772
+ const el = await fixture<CollectionBrowser>(
773
+ html`<collection-browser
774
+ .searchService=${searchService}
775
+ ></collection-browser>`,
776
+ );
777
+
778
+ el.withinCollection = 'error';
779
+ await el.updateComplete;
780
+ await el.initialSearchComplete;
781
+
782
+ const errorPlaceholder = el.shadowRoot?.querySelector(
783
+ 'empty-placeholder',
784
+ ) as EmptyPlaceholder;
785
+ const errorDetails = errorPlaceholder?.shadowRoot?.querySelector(
786
+ '.error-details',
787
+ ) as HTMLParagraphElement;
788
+
789
+ expect(errorDetails).to.exist;
790
+ expect(errorDetails.textContent).to.contain('foo');
791
+ });
792
+
793
+ it('reports malformed response errors to Sentry', async () => {
794
+ const sentrySpy = sinon.spy();
795
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
796
+ (window as any).Sentry = { captureMessage: sentrySpy };
797
+ const searchService = new MockSearchService();
798
+ const el = await fixture<CollectionBrowser>(
799
+ html`<collection-browser
800
+ .searchService=${searchService}
801
+ ></collection-browser>`,
802
+ );
803
+
804
+ el.baseQuery = 'malformed';
805
+ await el.updateComplete;
806
+ await el.initialSearchComplete;
807
+
808
+ expect(sentrySpy.callCount).to.be.greaterThanOrEqual(1);
809
+ });
810
+
811
+ it('adds collection names to cache when present on response', async () => {
812
+ const searchService = new MockSearchService();
813
+
814
+ const el = await fixture<CollectionBrowser>(
815
+ html`<collection-browser .searchService=${searchService}>
816
+ </collection-browser>`,
817
+ );
818
+
819
+ el.baseQuery = 'collection-titles';
820
+ await el.updateComplete;
821
+ await el.initialSearchComplete;
822
+
823
+ expect(el.dataSource.collectionTitles.get('foo')).to.equal(
824
+ 'Foo Collection',
825
+ );
826
+ expect(el.dataSource.collectionTitles.get('bar')).to.equal(
827
+ 'Bar Collection',
828
+ );
829
+ expect(el.dataSource.collectionTitles.get('baz')).to.equal(
830
+ 'Baz Collection',
831
+ );
832
+ expect(el.dataSource.collectionTitles.get('boop')).to.equal(
833
+ 'Boop Collection',
834
+ );
835
+ });
836
+
837
+ it('adds tv channel aliases to cache when present on response', async () => {
838
+ const searchService = new MockSearchService();
839
+
840
+ const el = await fixture<CollectionBrowser>(
841
+ html`<collection-browser .searchService=${searchService}>
842
+ </collection-browser>`,
843
+ );
844
+
845
+ el.baseQuery = 'channel-aliases';
846
+ await el.updateComplete;
847
+ await el.initialSearchComplete;
848
+
849
+ expect(el.dataSource.tvChannelAliases.get('foo')).to.equal('Foo Network');
850
+ expect(el.dataSource.tvChannelAliases.get('bar')).to.equal('Bar Network');
851
+ });
852
+
853
+ it('keeps search results from fetch if no change to query or sort param', async () => {
854
+ const resultsSpy = sinon.spy();
855
+ const searchService = new MockSearchService({
856
+ asyncResponse: true,
857
+ resultsSpy,
858
+ });
859
+
860
+ const el = await fixture<CollectionBrowser>(
861
+ html`<collection-browser .searchService=${searchService}>
862
+ </collection-browser>`,
863
+ );
864
+
865
+ el.baseQuery = 'with-sort';
866
+ el.selectedSort = SortField.date;
867
+ el.sortDirection = 'asc';
868
+ await el.updateComplete;
869
+
870
+ await el.dataSource.fetchPage(3);
871
+
872
+ // If there is no change to the query or sort param during the fetch, the results
873
+ // should be read.
874
+ expect(resultsSpy.callCount).to.be.greaterThanOrEqual(1);
875
+ });
876
+
877
+ it('discards obsolete search results if sort params changed before arrival', async () => {
878
+ const resultsSpy = sinon.spy();
879
+ const searchService = new MockSearchService({
880
+ asyncResponse: true,
881
+ resultsSpy,
882
+ });
883
+
884
+ const el = await fixture<CollectionBrowser>(
885
+ html`<collection-browser .searchService=${searchService}>
886
+ </collection-browser>`,
887
+ );
888
+
889
+ el.baseQuery = 'with-sort';
890
+ el.selectedSort = SortField.date;
891
+ el.sortDirection = 'asc';
892
+ await el.updateComplete;
893
+
894
+ // We want to spy exclusively on the first set of results, not the second
895
+ searchService.asyncResponse = false;
896
+ searchService.resultsSpy = () => {};
897
+
898
+ el.sortDirection = 'desc';
899
+ await el.updateComplete;
900
+ await el.initialSearchComplete;
901
+
902
+ // If the different sort param causes the results to be discarded,
903
+ // the first results array should never be read.
904
+ expect(resultsSpy.callCount).to.equal(0);
905
+ });
906
+
907
+ it('discards obsolete search results if sort param added before arrival', async () => {
908
+ const resultsSpy = sinon.spy();
909
+ const searchService = new MockSearchService({
910
+ asyncResponse: true,
911
+ resultsSpy,
912
+ });
913
+
914
+ const el = await fixture<CollectionBrowser>(
915
+ html`<collection-browser .searchService=${searchService}>
916
+ </collection-browser>`,
917
+ );
918
+
919
+ el.baseQuery = 'single-result';
920
+ await el.updateComplete;
921
+
922
+ // We want to spy exclusively on the first set of results, not the second
923
+ searchService.asyncResponse = false;
924
+ searchService.resultsSpy = () => {};
925
+
926
+ el.selectedSort = SortField.date;
927
+ el.sortDirection = 'asc';
928
+ await el.updateComplete;
929
+ await el.initialSearchComplete;
930
+
931
+ // If the different sort param causes the results to be discarded,
932
+ // the first results array should never be read.
933
+ expect(resultsSpy.callCount).to.equal(0);
934
+ });
935
+
936
+ it('discards obsolete search results if sort param cleared before arrival', async () => {
937
+ const resultsSpy = sinon.spy();
938
+ const searchService = new MockSearchService({
939
+ asyncResponse: true,
940
+ resultsSpy,
941
+ });
942
+
943
+ const el = await fixture<CollectionBrowser>(
944
+ html`<collection-browser .searchService=${searchService}>
945
+ </collection-browser>`,
946
+ );
947
+
948
+ el.baseQuery = 'with-sort';
949
+ el.selectedSort = SortField.date;
950
+ el.sortDirection = 'asc';
951
+ await el.updateComplete;
952
+
953
+ // We want to spy exclusively on the first set of results, not the second
954
+ searchService.asyncResponse = false;
955
+ searchService.resultsSpy = () => {};
956
+
957
+ el.selectedSort = SortField.default;
958
+ await el.updateComplete;
959
+ await el.initialSearchComplete;
960
+
961
+ // If the different sort param causes the results to be discarded,
962
+ // the first results array should never be read.
963
+ expect(resultsSpy.callCount).to.equal(0);
964
+ });
965
+
966
+ it('sets sort properties when user changes sort', async () => {
967
+ const searchService = new MockSearchService();
968
+ const el = await fixture<CollectionBrowser>(
969
+ html`<collection-browser .searchService=${searchService}>
970
+ </collection-browser>`,
971
+ );
972
+
973
+ expect(el.selectedSort).to.equal(SortField.default);
974
+
975
+ el.baseQuery = 'foo';
976
+ await el.updateComplete;
977
+ await nextTick();
978
+
979
+ const sortBar = el.shadowRoot?.querySelector(
980
+ '#content-container sort-filter-bar',
981
+ ) as SortFilterBar;
982
+ expect(sortBar, 'sort bar').to.exist;
983
+
984
+ // Switch to title sort
985
+ sortBar.setSelectedSort(SortField.title);
986
+ await el.updateComplete;
987
+ expect(el.selectedSort).to.equal(SortField.title);
988
+ expect(el.sortDirection).to.equal('asc'); // Default for title sort
989
+
990
+ // Switch to descending sort order
991
+ sortBar.setSortDirection('desc');
992
+ await el.updateComplete;
993
+ expect(el.selectedSort).to.equal(SortField.title);
994
+ expect(el.sortDirection).to.equal('desc');
995
+ });
996
+
997
+ it('sets sort filter properties when user selects title filter', async () => {
998
+ const searchService = new MockSearchService();
999
+ const el = await fixture<CollectionBrowser>(
1000
+ html`<collection-browser .searchService=${searchService}>
1001
+ </collection-browser>`,
1002
+ );
1003
+
1004
+ el.baseQuery = 'first-title';
1005
+ el.selectedSort = 'title' as SortField;
1006
+ el.sortDirection = 'asc';
1007
+ el.selectedTitleFilter = 'X';
1008
+ await el.updateComplete;
1009
+ await el.initialSearchComplete;
1010
+
1011
+ expect(searchService.searchParams?.query).to.equal('first-title');
1012
+ expect(searchService.searchParams?.filters?.firstTitle?.X).to.equal(
1013
+ FilterConstraint.INCLUDE,
1014
+ );
1015
+ });
1016
+
1017
+ it('sets sort filter properties when user selects creator filter', async () => {
1018
+ const searchService = new MockSearchService();
1019
+ const el = await fixture<CollectionBrowser>(
1020
+ html`<collection-browser .searchService=${searchService}>
1021
+ </collection-browser>`,
1022
+ );
1023
+
1024
+ el.baseQuery = 'first-creator';
1025
+ el.selectedSort = 'creator' as SortField;
1026
+ el.sortDirection = 'asc';
1027
+ el.selectedCreatorFilter = 'X';
1028
+ await el.updateComplete;
1029
+ await el.initialSearchComplete;
1030
+
1031
+ expect(searchService.searchParams?.query).to.equal('first-creator');
1032
+ expect(searchService.searchParams?.filters?.firstCreator?.X).to.equal(
1033
+ FilterConstraint.INCLUDE,
1034
+ );
1035
+ });
1036
+
1037
+ it('sets sort filter properties simultaneous with facets and date range', async () => {
1038
+ const searchService = new MockSearchService();
1039
+ const selectedFacets: SelectedFacets = {
1040
+ collection: { foo: { key: 'foo', state: 'selected', count: 1 } },
1041
+ creator: {},
1042
+ language: {},
1043
+ lending: {},
1044
+ mediatype: {},
1045
+ subject: {},
1046
+ year: {},
1047
+ };
1048
+
1049
+ const el = await fixture<CollectionBrowser>(
1050
+ html`<collection-browser .searchService=${searchService}>
1051
+ </collection-browser>`,
1052
+ );
1053
+
1054
+ el.baseQuery = 'first-creator';
1055
+ el.selectedSort = 'creator' as SortField;
1056
+ el.selectedFacets = selectedFacets;
1057
+ el.minSelectedDate = '1950';
1058
+ el.maxSelectedDate = '1970';
1059
+ el.sortDirection = 'asc';
1060
+ el.selectedCreatorFilter = 'X';
1061
+ await el.updateComplete;
1062
+ await el.initialSearchComplete;
1063
+
1064
+ expect(searchService.searchParams?.query).to.equal('first-creator');
1065
+ expect(searchService.searchParams?.filters).to.deep.equal({
1066
+ collection: {
1067
+ foo: 'inc',
1068
+ },
1069
+ year: {
1070
+ '1950': 'gte',
1071
+ '1970': 'lte',
1072
+ },
1073
+ firstCreator: {
1074
+ X: 'inc',
1075
+ },
1076
+ });
1077
+ });
1078
+
1079
+ it('applies correct TV search filter for commercials', async () => {
1080
+ const searchService = new MockSearchService();
1081
+ const el = await fixture<CollectionBrowser>(
1082
+ html`<collection-browser .searchService=${searchService}>
1083
+ </collection-browser>`,
1084
+ );
1085
+
1086
+ el.baseQuery = 'tv-fields';
1087
+ el.searchType = SearchType.TV;
1088
+ el.selectedFacets = {
1089
+ clip_type: {
1090
+ commercial: { key: 'commercial', count: 1, state: 'selected' },
1091
+ },
1092
+ };
1093
+ await el.updateComplete;
1094
+ await el.initialSearchComplete;
1095
+
1096
+ expect(searchService.searchParams?.filters?.clip_type?.commercial).to.equal(
1097
+ FilterConstraint.INCLUDE,
1098
+ );
1099
+ });
1100
+
1101
+ it('applies correct TV search filter for fact checks', async () => {
1102
+ const searchService = new MockSearchService();
1103
+ const el = await fixture<CollectionBrowser>(
1104
+ html`<collection-browser .searchService=${searchService}>
1105
+ </collection-browser>`,
1106
+ );
1107
+
1108
+ el.baseQuery = 'tv-fields';
1109
+ el.searchType = SearchType.TV;
1110
+ el.selectedFacets = {
1111
+ clip_type: {
1112
+ 'fact check': { key: 'fact check', count: 1, state: 'selected' },
1113
+ },
1114
+ };
1115
+ await el.updateComplete;
1116
+ await el.initialSearchComplete;
1117
+
1118
+ expect(
1119
+ searchService.searchParams?.filters?.clip_type?.['fact check'],
1120
+ ).to.equal(FilterConstraint.INCLUDE);
1121
+ });
1122
+
1123
+ it('applies correct TV search filter for quotes', async () => {
1124
+ const searchService = new MockSearchService();
1125
+ const el = await fixture<CollectionBrowser>(
1126
+ html`<collection-browser .searchService=${searchService}>
1127
+ </collection-browser>`,
1128
+ );
1129
+
1130
+ el.baseQuery = 'tv-fields';
1131
+ el.searchType = SearchType.TV;
1132
+ el.selectedFacets = {
1133
+ clip_type: { quote: { key: 'quote', count: 1, state: 'selected' } },
1134
+ };
1135
+ await el.updateComplete;
1136
+ await el.initialSearchComplete;
1137
+
1138
+ expect(searchService.searchParams?.filters?.clip_type?.quote).to.equal(
1139
+ FilterConstraint.INCLUDE,
1140
+ );
1141
+ });
1142
+
1143
+ it('resets letter filters when query changes', async () => {
1144
+ const searchService = new MockSearchService();
1145
+ const el = await fixture<CollectionBrowser>(
1146
+ html`<collection-browser .searchService=${searchService}>
1147
+ </collection-browser>`,
1148
+ );
1149
+
1150
+ el.baseQuery = 'first-creator';
1151
+ el.selectedSort = 'creator' as SortField;
1152
+ el.sortDirection = 'asc';
1153
+ el.selectedCreatorFilter = 'X';
1154
+ await el.updateComplete;
1155
+ await el.initialSearchComplete;
1156
+ await nextTick();
1157
+
1158
+ expect(searchService.searchParams?.query).to.equal('first-creator');
1159
+ expect(searchService.searchParams?.filters?.firstCreator?.X).to.equal(
1160
+ FilterConstraint.INCLUDE,
1161
+ );
1162
+
1163
+ el.baseQuery = 'collection:foo';
1164
+ await el.updateComplete;
1165
+ await nextTick();
1166
+
1167
+ expect(searchService.searchParams?.query).to.equal('collection:foo');
1168
+ expect(searchService.searchParams?.filters?.firstCreator).not.to.exist;
1169
+ });
1170
+
1171
+ it('sets date range query when date picker selection changed', async () => {
1172
+ const searchService = new MockSearchService();
1173
+ const mockAnalyticsHandler = new MockAnalyticsHandler();
1174
+ const el = await fixture<CollectionBrowser>(
1175
+ html`<collection-browser
1176
+ .searchService=${searchService}
1177
+ .analyticsHandler=${mockAnalyticsHandler}
1178
+ .suppressPlaceholders=${true}
1179
+ >
1180
+ </collection-browser>`,
1181
+ );
1182
+
1183
+ el.baseQuery = 'years'; // Includes year_histogram aggregation in response
1184
+ el.showHistogramDatePicker = true;
1185
+ await el.updateComplete;
1186
+
1187
+ const facets = el.shadowRoot?.querySelector(
1188
+ 'collection-facets',
1189
+ ) as CollectionFacets;
1190
+ await facets?.updateComplete;
1191
+
1192
+ // Wait for the date picker to be rendered (which may take until the next tick)
1193
+ await nextTick();
1194
+
1195
+ const histogram = facets?.shadowRoot?.querySelector(
1196
+ 'histogram-date-range',
1197
+ ) as HistogramDateRange;
1198
+
1199
+ expect(histogram, 'histogram exists').to.exist;
1200
+
1201
+ // Enter a new min date into the date picker
1202
+ const minDateInput = histogram.shadowRoot?.querySelector(
1203
+ '#date-min',
1204
+ ) as HTMLInputElement;
1205
+
1206
+ const pressEnterEvent = new KeyboardEvent('keyup', {
1207
+ key: 'Enter',
1208
+ });
1209
+
1210
+ minDateInput.value = '1960';
1211
+ minDateInput.dispatchEvent(pressEnterEvent);
1212
+
1213
+ // Wait for the histogram's update delay
1214
+ await aTimeout(histogram.updateDelay + 50);
1215
+
1216
+ // Ensure that the histogram change propagated to the collection browser's
1217
+ // date query correctly.
1218
+ await el.updateComplete;
1219
+ expect(el.minSelectedDate).to.equal('1960');
1220
+ expect(el.maxSelectedDate).to.equal('2009');
1221
+ });
1222
+
1223
+ it('sets date range query when monthly date picker selection changed', async () => {
1224
+ const searchService = new MockSearchService();
1225
+ const el = await fixture<CollectionBrowser>(
1226
+ html`<collection-browser
1227
+ .searchService=${searchService}
1228
+ .suppressPlaceholders=${true}
1229
+ >
1230
+ </collection-browser>`,
1231
+ );
1232
+
1233
+ el.baseQuery = 'months'; // Includes date_histogram aggregation in response
1234
+ el.searchType = SearchType.TV;
1235
+ el.showHistogramDatePicker = true;
1236
+ await el.updateComplete;
1237
+
1238
+ const facets = el.shadowRoot?.querySelector(
1239
+ 'collection-facets',
1240
+ ) as CollectionFacets;
1241
+ await facets?.updateComplete;
1242
+
1243
+ // Wait for the date picker to be rendered (which may take until the next tick)
1244
+ await nextTick();
1245
+
1246
+ const histogram = facets?.shadowRoot?.querySelector(
1247
+ 'histogram-date-range',
1248
+ ) as HistogramDateRange;
1249
+
1250
+ expect(histogram, 'histogram exists').to.exist;
1251
+
1252
+ // Enter a new min date into the date picker
1253
+ const minDateInput = histogram.shadowRoot?.querySelector(
1254
+ '#date-min',
1255
+ ) as HTMLInputElement;
1256
+
1257
+ const pressEnterEvent = new KeyboardEvent('keyup', {
1258
+ key: 'Enter',
1259
+ });
1260
+
1261
+ minDateInput.value = '2001-02';
1262
+ minDateInput.dispatchEvent(pressEnterEvent);
1263
+
1264
+ // Wait for the histogram's update delay
1265
+ await aTimeout(histogram.updateDelay + 50);
1266
+
1267
+ // Ensure that the histogram change propagated to the collection browser's
1268
+ // date query correctly.
1269
+ await el.updateComplete;
1270
+ expect(el.minSelectedDate).to.equal('2001-02');
1271
+ expect(el.maxSelectedDate).to.equal('2002-12');
1272
+ });
1273
+
1274
+ it('emits event when results start and end loading', async () => {
1275
+ const spy = sinon.spy();
1276
+ const searchService = new MockSearchService();
1277
+ const el = await fixture<CollectionBrowser>(
1278
+ html`<collection-browser
1279
+ .searchService=${searchService}
1280
+ @searchResultsLoadingChanged=${spy}
1281
+ ></collection-browser>`,
1282
+ );
1283
+ spy.resetHistory();
1284
+
1285
+ el.baseQuery = 'collection:foo';
1286
+ await el.updateComplete;
1287
+ await el.initialSearchComplete;
1288
+
1289
+ // Should initially emit loading=true, then later emit loading=false
1290
+ expect(spy.callCount).to.equal(2);
1291
+ expect(spy.firstCall.firstArg?.detail?.loading).to.equal(true);
1292
+ expect(spy.secondCall.firstArg?.detail?.loading).to.equal(false);
1293
+ });
1294
+
1295
+ it('collapses extra set of quotes around href field', async () => {
1296
+ const searchService = new MockSearchService();
1297
+ const el = await fixture<CollectionBrowser>(
1298
+ html`<collection-browser
1299
+ .searchService=${searchService}
1300
+ .baseNavigationUrl=${''}
1301
+ ></collection-browser>`,
1302
+ );
1303
+
1304
+ el.baseQuery = 'extra-quoted-href';
1305
+ await el.updateComplete;
1306
+ await el.initialSearchComplete;
1307
+ await el.updateComplete;
1308
+ await aTimeout(50);
1309
+
1310
+ // Original href q param starts/ends with %22%22, but should be collapsed to %22 before render
1311
+ expect(el.dataSource.getTileModelAt(0)?.href).to.equal(
1312
+ '/details/foo?q=%22quoted+query%22',
1313
+ );
1314
+ });
1315
+
1316
+ it('sets default sort from collection metadata', async () => {
1317
+ const searchService = new MockSearchService();
1318
+ const el = await fixture<CollectionBrowser>(
1319
+ html`<collection-browser
1320
+ .searchService=${searchService}
1321
+ .baseNavigationUrl=${''}
1322
+ ></collection-browser>`,
1323
+ );
1324
+
1325
+ el.withinCollection = 'default-sort';
1326
+ await el.updateComplete;
1327
+ await el.initialSearchComplete;
1328
+ await el.updateComplete;
1329
+ await aTimeout(50);
1330
+
1331
+ const sortBar = el.shadowRoot?.querySelector(
1332
+ 'sort-filter-bar',
1333
+ ) as SortFilterBar;
1334
+ expect(sortBar).to.exist;
1335
+ expect(sortBar.defaultSortField).to.equal(SortField.title);
1336
+ expect(sortBar.defaultSortDirection).to.equal('asc');
1337
+ expect(sortBar.selectedSort).to.equal(SortField.default);
1338
+ expect(sortBar.sortDirection).to.be.null;
1339
+ });
1340
+
1341
+ it('sets default sort from collection metadata in "-field" format', async () => {
1342
+ const searchService = new MockSearchService();
1343
+ const el = await fixture<CollectionBrowser>(
1344
+ html`<collection-browser
1345
+ .searchService=${searchService}
1346
+ .baseNavigationUrl=${''}
1347
+ ></collection-browser>`,
1348
+ );
1349
+
1350
+ el.withinCollection = 'default-sort-concise';
1351
+ await el.updateComplete;
1352
+ await el.initialSearchComplete;
1353
+ await el.updateComplete;
1354
+ await aTimeout(50);
1355
+
1356
+ const sortBar = el.shadowRoot?.querySelector(
1357
+ 'sort-filter-bar',
1358
+ ) as SortFilterBar;
1359
+ expect(sortBar).to.exist;
1360
+ expect(sortBar.defaultSortField).to.equal(SortField.dateadded);
1361
+ expect(sortBar.defaultSortDirection).to.equal('desc');
1362
+ expect(sortBar.selectedSort).to.equal(SortField.default);
1363
+ expect(sortBar.sortDirection).to.be.null;
1364
+ });
1365
+
1366
+ it('falls back to weekly views default sorting on profiles when tab not set', async () => {
1367
+ const el = await fixture<CollectionBrowser>(
1368
+ html`<collection-browser
1369
+ .withinProfile=${'@foobar'}
1370
+ ></collection-browser>`,
1371
+ );
1372
+
1373
+ el.applyDefaultProfileSort();
1374
+ expect(el.defaultSortParam).to.deep.equal({
1375
+ field: 'week',
1376
+ direction: 'desc',
1377
+ });
1378
+ });
1379
+
1380
+ it('uses relevance sort as default when a query is set', async () => {
1381
+ const searchService = new MockSearchService();
1382
+ const el = await fixture<CollectionBrowser>(
1383
+ html`<collection-browser
1384
+ .searchService=${searchService}
1385
+ .baseNavigationUrl=${''}
1386
+ ></collection-browser>`,
1387
+ );
1388
+
1389
+ el.withinCollection = 'default-sort';
1390
+ el.baseQuery = 'default-sort';
1391
+ await el.updateComplete;
1392
+ await el.initialSearchComplete;
1393
+ await el.updateComplete;
1394
+ await aTimeout(50);
1395
+
1396
+ const sortBar = el.shadowRoot?.querySelector(
1397
+ 'sort-filter-bar',
1398
+ ) as SortFilterBar;
1399
+ expect(sortBar).to.exist;
1400
+ expect(sortBar.defaultSortField).to.equal(SortField.relevance);
1401
+ expect(sortBar.defaultSortDirection).to.be.null;
1402
+ expect(sortBar.selectedSort).to.equal(SortField.default);
1403
+ expect(sortBar.sortDirection).to.be.null;
1404
+ });
1405
+
1406
+ it('uses date favorited sort as default when targeting fav- collection', async () => {
1407
+ const searchService = new MockSearchService();
1408
+ const el = await fixture<CollectionBrowser>(
1409
+ html`<collection-browser
1410
+ .searchService=${searchService}
1411
+ .baseNavigationUrl=${''}
1412
+ ></collection-browser>`,
1413
+ );
1414
+
1415
+ el.withinCollection = 'fav-sort';
1416
+ await el.updateComplete;
1417
+ await el.initialSearchComplete;
1418
+ await el.updateComplete;
1419
+ await aTimeout(50);
1420
+
1421
+ const sortBar = el.shadowRoot?.querySelector(
1422
+ 'sort-filter-bar',
1423
+ ) as SortFilterBar;
1424
+ expect(sortBar).to.exist;
1425
+ expect(sortBar.defaultSortField).to.equal(SortField.datefavorited);
1426
+ expect(sortBar.defaultSortDirection).to.equal('desc');
1427
+ expect(sortBar.selectedSort).to.equal(SortField.default);
1428
+ expect(sortBar.sortDirection).to.be.null;
1429
+ });
1430
+
1431
+ it('scrolls to page', async () => {
1432
+ const searchService = new MockSearchService();
1433
+ const el = await fixture<CollectionBrowser>(
1434
+ html`<collection-browser .searchService=${searchService}>
1435
+ </collection-browser>`,
1436
+ );
1437
+
1438
+ // Infinite scroller won't exist unless there's a base query.
1439
+ // First ensure that we don't throw errors when it doesn't exist.
1440
+ await el.goToPage(1);
1441
+
1442
+ // And make sure it correctly calls scrollToCell when the
1443
+ // infinite scroller does exist.
1444
+ el.baseQuery = 'collection:foo';
1445
+ await el.updateComplete;
1446
+ await nextTick();
1447
+
1448
+ const infiniteScroller = el.shadowRoot?.querySelector(
1449
+ 'infinite-scroller',
1450
+ ) as InfiniteScroller;
1451
+ expect(infiniteScroller).to.exist;
1452
+
1453
+ const oldScrollToCell = infiniteScroller.scrollToCell;
1454
+ const spy = sinon.spy();
1455
+ infiniteScroller.scrollToCell = spy;
1456
+
1457
+ await el.goToPage(1);
1458
+ expect(spy.callCount, 'scroll to page fires once').to.equal(1);
1459
+
1460
+ infiniteScroller.scrollToCell = oldScrollToCell;
1461
+ });
1462
+
1463
+ it('shows mobile facets in mobile view', async () => {
1464
+ const searchService = new MockSearchService();
1465
+ const el = await fixture<CollectionBrowser>(
1466
+ html`<collection-browser
1467
+ .searchService=${searchService}
1468
+ .mobileBreakpoint=${9999}
1469
+ ></collection-browser>`,
1470
+ );
1471
+
1472
+ el.baseQuery = 'collection:foo';
1473
+ await el.updateComplete;
1474
+
1475
+ const contentContainer = el.shadowRoot?.querySelector(
1476
+ '#content-container',
1477
+ ) as HTMLElement;
1478
+
1479
+ el.handleResize({
1480
+ target: contentContainer,
1481
+ contentRect: contentContainer.getBoundingClientRect(),
1482
+ borderBoxSize: [],
1483
+ contentBoxSize: [],
1484
+ devicePixelContentBoxSize: [],
1485
+ });
1486
+ await el.updateComplete;
1487
+
1488
+ const mobileFacets = el.shadowRoot?.querySelector(
1489
+ '#mobile-filter-collapse',
1490
+ );
1491
+ expect(mobileFacets).to.exist;
1492
+ });
1493
+
1494
+ it('fires analytics when mobile facets toggled', async () => {
1495
+ const searchService = new MockSearchService();
1496
+ const analyticsHandler = new MockAnalyticsHandler();
1497
+ const el = await fixture<CollectionBrowser>(
1498
+ html`<collection-browser
1499
+ .searchService=${searchService}
1500
+ .analyticsHandler=${analyticsHandler}
1501
+ .searchContext=${'foobar-context'}
1502
+ .mobileBreakpoint=${9999}
1503
+ ></collection-browser>`,
1504
+ );
1505
+
1506
+ el.baseQuery = 'collection:foo';
1507
+ await el.updateComplete;
1508
+
1509
+ const contentContainer = el.shadowRoot?.querySelector(
1510
+ '#content-container',
1511
+ ) as HTMLElement;
1512
+
1513
+ el.handleResize({
1514
+ target: contentContainer,
1515
+ contentRect: contentContainer.getBoundingClientRect(),
1516
+ borderBoxSize: [],
1517
+ contentBoxSize: [],
1518
+ devicePixelContentBoxSize: [],
1519
+ });
1520
+ await el.updateComplete;
1521
+
1522
+ const mobileFacets = el.shadowRoot?.querySelector(
1523
+ '#mobile-filter-collapse',
1524
+ ) as HTMLDetailsElement;
1525
+ expect(mobileFacets).to.exist;
1526
+
1527
+ // We set up a Promise to wait for the 'toggle' event on the collapser,
1528
+ // which is what triggers the analytics.
1529
+ let facetsToggled = new Promise(resolve => {
1530
+ mobileFacets.addEventListener('toggle', resolve);
1531
+ });
1532
+
1533
+ // Open the mobile facets accordion & check analytics
1534
+ const mobileFacetsHeader = mobileFacets.querySelector('summary');
1535
+ expect(mobileFacetsHeader).to.exist;
1536
+ mobileFacetsHeader!.click();
1537
+ await facetsToggled;
1538
+ expect(analyticsHandler.callCategory).to.equal('foobar-context');
1539
+ expect(analyticsHandler.callAction).to.equal(
1540
+ analyticsActions.mobileFacetsToggled,
1541
+ );
1542
+ expect(analyticsHandler.callLabel).to.equal('open');
1543
+
1544
+ // Close the mobile facets accordion & check analytics
1545
+ facetsToggled = new Promise(resolve => {
1546
+ mobileFacets.addEventListener('toggle', resolve);
1547
+ });
1548
+ mobileFacetsHeader!.click();
1549
+ await facetsToggled;
1550
+ expect(analyticsHandler.callCategory).to.equal('foobar-context');
1551
+ expect(analyticsHandler.callAction).to.equal(
1552
+ analyticsActions.mobileFacetsToggled,
1553
+ );
1554
+ expect(analyticsHandler.callLabel).to.equal('closed');
1555
+ });
1556
+
1557
+ it('sets parent collections to prop when searching a collection', async () => {
1558
+ const searchService = new MockSearchService();
1559
+ const el = await fixture<CollectionBrowser>(
1560
+ html`<collection-browser
1561
+ .searchService=${searchService}
1562
+ .withinCollection=${'fake'}
1563
+ ></collection-browser>`,
1564
+ );
1565
+
1566
+ el.baseQuery = 'parent-collections';
1567
+ await el.updateComplete;
1568
+ await el.initialSearchComplete;
1569
+ await aTimeout(0);
1570
+
1571
+ expect(el.dataSource.parentCollections).to.deep.equal(['foo', 'bar']);
1572
+ });
1573
+
1574
+ it('recognizes TV collections', async () => {
1575
+ const searchService = new MockSearchService();
1576
+ const el = await fixture<CollectionBrowser>(
1577
+ html`<collection-browser
1578
+ .searchService=${searchService}
1579
+ .withinCollection=${'TV-FOO'}
1580
+ ></collection-browser>`,
1581
+ );
1582
+
1583
+ el.baseQuery = 'tv-collection';
1584
+ await el.updateComplete;
1585
+ await el.initialSearchComplete;
1586
+ await aTimeout(0);
1587
+
1588
+ expect(el.isTVCollection).to.be.true;
1589
+ });
1590
+
1591
+ it('refreshes when certain properties change - with some analytics event sampling', async () => {
1592
+ const mockAnalyticsHandler = new MockAnalyticsHandler();
1593
+ const searchService = new MockSearchService();
1594
+ const el = await fixture<CollectionBrowser>(
1595
+ html`<collection-browser
1596
+ .analyticsHandler=${mockAnalyticsHandler}
1597
+ .searchService=${searchService}
1598
+ ></collection-browser>`,
1599
+ );
1600
+ const infiniteScrollerRefreshSpy = sinon.spy();
1601
+
1602
+ // Infinite scroller won't exist unless there's a base query
1603
+ el.baseQuery = 'collection:foo';
1604
+ await el.updateComplete;
1605
+ await nextTick();
1606
+
1607
+ const infiniteScroller = el.shadowRoot?.querySelector('infinite-scroller');
1608
+ (infiniteScroller as InfiniteScroller).reload = infiniteScrollerRefreshSpy;
1609
+ expect(infiniteScrollerRefreshSpy.called).to.be.false;
1610
+ expect(infiniteScrollerRefreshSpy.callCount).to.equal(0);
1611
+
1612
+ // testing: `loggedIn`
1613
+ el.loggedIn = true;
1614
+ await el.updateComplete;
1615
+
1616
+ expect(infiniteScrollerRefreshSpy.called, 'Infinite Scroller Refresh').to.be
1617
+ .true;
1618
+ expect(
1619
+ infiniteScrollerRefreshSpy.callCount,
1620
+ 'Infinite Scroller Refresh call count',
1621
+ ).to.equal(1);
1622
+
1623
+ el.loggedIn = false;
1624
+ await el.updateComplete;
1625
+
1626
+ expect(
1627
+ infiniteScrollerRefreshSpy.callCount,
1628
+ '2nd Infinite Scroller Refresh',
1629
+ ).to.equal(2);
1630
+
1631
+ // testing: `displayMode`
1632
+ el.displayMode = 'list-compact';
1633
+ el.searchContext = 'beta-search';
1634
+ await el.updateComplete;
1635
+ expect(
1636
+ infiniteScrollerRefreshSpy.callCount,
1637
+ '3rd Infinite Scroller Refresh',
1638
+ ).to.equal(3);
1639
+
1640
+ expect(mockAnalyticsHandler.callCategory).to.equal('beta-search');
1641
+ expect(mockAnalyticsHandler.callAction).to.equal('displayMode');
1642
+ expect(mockAnalyticsHandler.callLabel).to.equal('list-compact');
1643
+
1644
+ el.displayMode = 'list-detail';
1645
+ await el.updateComplete;
1646
+ expect(
1647
+ infiniteScrollerRefreshSpy.callCount,
1648
+ '4th Infinite Scroller Refresh',
1649
+ ).to.equal(4);
1650
+
1651
+ expect(mockAnalyticsHandler.callCategory).to.equal('beta-search');
1652
+ expect(mockAnalyticsHandler.callAction).to.equal('displayMode');
1653
+ expect(mockAnalyticsHandler.callLabel).to.equal('list-detail');
1654
+
1655
+ // testing: `baseNavigationUrl`
1656
+ el.baseNavigationUrl = 'https://funtestsite.com';
1657
+ await el.updateComplete;
1658
+ expect(
1659
+ infiniteScrollerRefreshSpy.callCount,
1660
+ '5th Infinite Scroller Refresh',
1661
+ ).to.equal(5);
1662
+
1663
+ // testing: `baseImageUrl`
1664
+ el.baseImageUrl = 'https://funtestsiteforimages.com';
1665
+ await el.updateComplete;
1666
+ expect(
1667
+ infiniteScrollerRefreshSpy.callCount,
1668
+ '6th Infinite Scroller Refresh',
1669
+ ).to.equal(6);
1670
+ });
1671
+
1672
+ it('query the search service for single result', async () => {
1673
+ const searchService = new MockSearchService();
1674
+
1675
+ const el = await fixture<CollectionBrowser>(
1676
+ html`<collection-browser .searchService=${searchService}>
1677
+ </collection-browser>`,
1678
+ );
1679
+
1680
+ el.baseQuery = 'single-result';
1681
+ await el.updateComplete;
1682
+ await el.initialSearchComplete;
1683
+
1684
+ expect(
1685
+ el.shadowRoot?.querySelector('#big-results-label')?.textContent,
1686
+ ).to.contains('Result');
1687
+ });
1688
+
1689
+ it('`searchContext` prop helps describe where component is being used', async () => {
1690
+ const el = await fixture<CollectionBrowser>(
1691
+ html`<collection-browser></collection-browser>`,
1692
+ );
1693
+
1694
+ expect(el.searchContext).to.equal(analyticsCategories.default);
1695
+
1696
+ el.searchContext = 'unicorn-search';
1697
+ await el.updateComplete;
1698
+
1699
+ expect(el.searchContext).to.equal('unicorn-search');
1700
+
1701
+ // property is reflected as attribute
1702
+ expect(el.getAttribute('searchcontext')).to.equal('unicorn-search');
1703
+ });
1704
+
1705
+ it('respects the initial set of URL parameters for a general search', async () => {
1706
+ const url = new URL(window.location.href);
1707
+ const { searchParams } = url;
1708
+ searchParams.set('query', 'foo');
1709
+ searchParams.set('sin', 'TXT');
1710
+ searchParams.set('sort', 'title');
1711
+ searchParams.append('not[]', 'mediatype:"data"');
1712
+ searchParams.append('and[]', 'subject:"baz"');
1713
+ searchParams.append('and[]', 'firstTitle:X');
1714
+ searchParams.append('and[]', 'year:[2000 TO 2010]');
1715
+ window.history.replaceState({}, '', url);
1716
+
1717
+ const searchService = new MockSearchService();
1718
+ const el = await fixture<CollectionBrowser>(
1719
+ html`<collection-browser .searchService=${searchService}>
1720
+ </collection-browser>`,
1721
+ );
1722
+
1723
+ await el.initialSearchComplete;
1724
+ await el.updateComplete;
1725
+
1726
+ expect(el.baseQuery).to.equal('foo');
1727
+ expect(el.searchType).to.equal(SearchType.FULLTEXT);
1728
+ expect(el.selectedSort).to.equal(SortField.title);
1729
+ expect(el.selectedFacets?.mediatype?.data?.state).to.equal('hidden');
1730
+ expect(el.selectedFacets?.subject?.baz?.state).to.equal('selected');
1731
+ expect(el.selectedTitleFilter).to.equal('X');
1732
+ expect(el.minSelectedDate).to.equal('2000');
1733
+ expect(el.maxSelectedDate).to.equal('2010');
1734
+ });
1735
+
1736
+ it('respects the initial set of URL parameters within a collection', async () => {
1737
+ const url = new URL(window.location.href);
1738
+ const { searchParams } = url;
1739
+ searchParams.set('query', 'foo');
1740
+ searchParams.set('sin', 'TXT');
1741
+ searchParams.set('sort', 'title');
1742
+ searchParams.append('not[]', 'mediatype:"data"');
1743
+ searchParams.append('and[]', 'subject:"baz"');
1744
+ searchParams.append('and[]', 'firstTitle:X');
1745
+ searchParams.append('and[]', 'year:[2000 TO 2010]');
1746
+ window.history.replaceState({}, '', url);
1747
+
1748
+ const searchService = new MockSearchService();
1749
+ const el = await fixture<CollectionBrowser>(
1750
+ html`<collection-browser
1751
+ .searchService=${searchService}
1752
+ .withinCollection=${'foobar'}
1753
+ >
1754
+ </collection-browser>`,
1755
+ );
1756
+
1757
+ await el.initialSearchComplete;
1758
+ await el.updateComplete;
1759
+
1760
+ expect(el.withinCollection).to.equal('foobar');
1761
+ expect(el.baseQuery).to.equal('foo');
1762
+ expect(el.searchType).to.equal(SearchType.FULLTEXT);
1763
+ expect(el.selectedSort).to.equal(SortField.title);
1764
+ expect(el.selectedFacets?.mediatype?.data?.state).to.equal('hidden');
1765
+ expect(el.selectedFacets?.subject?.baz?.state).to.equal('selected');
1766
+ expect(el.selectedTitleFilter).to.equal('X');
1767
+ expect(el.minSelectedDate).to.equal('2000');
1768
+ expect(el.maxSelectedDate).to.equal('2010');
1769
+ });
1770
+
1771
+ it('respects the initial set of URL parameters within a profile page', async () => {
1772
+ const url = new URL(window.location.href);
1773
+ const { searchParams } = url;
1774
+ searchParams.set('query', 'foo');
1775
+ searchParams.append('not[]', 'mediatype:"data"');
1776
+ searchParams.append('and[]', 'subject:"baz"');
1777
+ searchParams.append('and[]', 'firstTitle:X');
1778
+ searchParams.append('and[]', 'year:[2000 TO 2010]');
1779
+ window.history.replaceState({}, '', url);
1780
+
1781
+ const searchService = new MockSearchService();
1782
+ const el = await fixture<CollectionBrowser>(
1783
+ html`<collection-browser
1784
+ .searchService=${searchService}
1785
+ .withinProfile=${'@foobar'}
1786
+ .profileElement=${'uploads'}
1787
+ >
1788
+ </collection-browser>`,
1789
+ );
1790
+
1791
+ await el.initialSearchComplete;
1792
+ await el.updateComplete;
1793
+
1794
+ expect(el.withinProfile).to.equal('@foobar');
1795
+ expect(el.profileElement).to.equal('uploads');
1796
+ expect(el.baseQuery).to.equal('foo');
1797
+ expect(el.searchType).to.equal(SearchType.DEFAULT);
1798
+ expect(el.selectedFacets?.mediatype?.data?.state).to.equal('hidden');
1799
+ expect(el.selectedFacets?.subject?.baz?.state).to.equal('selected');
1800
+ expect(el.selectedTitleFilter).to.equal('X');
1801
+ expect(el.minSelectedDate).to.equal('2000');
1802
+ expect(el.maxSelectedDate).to.equal('2010');
1803
+ });
1804
+
1805
+ it('clears filters except sort when query changes for a general search', async () => {
1806
+ const url = new URL(window.location.href);
1807
+ const { searchParams } = url;
1808
+ searchParams.set('query', 'foo');
1809
+ searchParams.set('sin', 'TXT');
1810
+ searchParams.set('sort', 'title');
1811
+ searchParams.append('not[]', 'mediatype:"data"');
1812
+ searchParams.append('and[]', 'subject:"baz"');
1813
+ searchParams.append('and[]', 'firstTitle:X');
1814
+ searchParams.append('and[]', 'year:[2000 TO 2010]');
1815
+ window.history.replaceState({}, '', url);
1816
+
1817
+ const searchService = new MockSearchService();
1818
+ const el = await fixture<CollectionBrowser>(
1819
+ html`<collection-browser .searchService=${searchService}>
1820
+ </collection-browser>`,
1821
+ );
1822
+
1823
+ await el.initialSearchComplete;
1824
+ await el.updateComplete;
1825
+
1826
+ el.baseQuery = 'bar';
1827
+ await el.updateComplete;
1828
+
1829
+ expect(el.baseQuery).to.equal('bar');
1830
+ expect(el.searchType).to.equal(SearchType.FULLTEXT);
1831
+ expect(el.selectedSort).to.equal(SortField.title);
1832
+ expect(el.selectedFacets?.mediatype?.data).not.to.exist;
1833
+ expect(el.selectedFacets?.subject?.baz).not.to.exist;
1834
+ expect(el.selectedTitleFilter).not.to.exist;
1835
+ expect(el.minSelectedDate).not.to.exist;
1836
+ expect(el.maxSelectedDate).not.to.exist;
1837
+ });
1838
+
1839
+ it('clears filters except sort when query changes within a collection', async () => {
1840
+ const url = new URL(window.location.href);
1841
+ const { searchParams } = url;
1842
+ searchParams.set('query', 'foo');
1843
+ searchParams.set('sin', 'TXT');
1844
+ searchParams.set('sort', 'title');
1845
+ searchParams.append('not[]', 'mediatype:"data"');
1846
+ searchParams.append('and[]', 'subject:"baz"');
1847
+ searchParams.append('and[]', 'firstTitle:X');
1848
+ searchParams.append('and[]', 'year:[2000 TO 2010]');
1849
+ window.history.replaceState({}, '', url);
1850
+
1851
+ const searchService = new MockSearchService();
1852
+ const el = await fixture<CollectionBrowser>(
1853
+ html`<collection-browser
1854
+ .searchService=${searchService}
1855
+ .withinCollection=${'foobar'}
1856
+ >
1857
+ </collection-browser>`,
1858
+ );
1859
+
1860
+ el.baseQuery = 'bar';
1861
+ await el.updateComplete;
1862
+
1863
+ expect(el.withinCollection).to.equal('foobar');
1864
+ expect(el.baseQuery).to.equal('bar');
1865
+ expect(el.searchType).to.equal(SearchType.FULLTEXT);
1866
+ expect(el.selectedSort).to.equal(SortField.title);
1867
+ expect(el.selectedFacets?.mediatype?.data).not.to.exist;
1868
+ expect(el.selectedFacets?.subject?.baz).not.to.exist;
1869
+ expect(el.selectedTitleFilter).not.to.exist;
1870
+ expect(el.minSelectedDate).not.to.exist;
1871
+ expect(el.maxSelectedDate).not.to.exist;
1872
+ });
1873
+
1874
+ it('clears filters *including* sort when target collection changes', async () => {
1875
+ const url = new URL(window.location.href);
1876
+ const { searchParams } = url;
1877
+ searchParams.set('query', 'foo');
1878
+ searchParams.set('sin', 'TXT');
1879
+ searchParams.set('sort', 'title');
1880
+ searchParams.append('not[]', 'mediatype:"data"');
1881
+ searchParams.append('and[]', 'subject:"baz"');
1882
+ searchParams.append('and[]', 'firstTitle:X');
1883
+ searchParams.append('and[]', 'year:[2000 TO 2010]');
1884
+ window.history.replaceState({}, '', url);
1885
+
1886
+ const searchService = new MockSearchService();
1887
+ const el = await fixture<CollectionBrowser>(
1888
+ html`<collection-browser
1889
+ .searchService=${searchService}
1890
+ .withinCollection=${'foobar'}
1891
+ >
1892
+ </collection-browser>`,
1893
+ );
1894
+
1895
+ el.withinCollection = 'bar';
1896
+ await el.updateComplete;
1897
+
1898
+ expect(el.withinCollection).to.equal('bar');
1899
+ expect(el.baseQuery).to.equal('foo');
1900
+ expect(el.searchType).to.equal(SearchType.FULLTEXT);
1901
+ expect(el.selectedSort).to.equal(SortField.default);
1902
+ expect(el.selectedFacets?.mediatype?.data).not.to.exist;
1903
+ expect(el.selectedFacets?.subject?.baz).not.to.exist;
1904
+ expect(el.selectedTitleFilter).not.to.exist;
1905
+ expect(el.minSelectedDate).not.to.exist;
1906
+ expect(el.maxSelectedDate).not.to.exist;
1907
+ });
1908
+
1909
+ it('correctly retrieves web archive hits', async () => {
1910
+ const searchService = new MockSearchService();
1911
+ const el = await fixture<CollectionBrowser>(
1912
+ html`<collection-browser
1913
+ .searchService=${searchService}
1914
+ .withinProfile=${'@foo'}
1915
+ .profileElement=${'web_archives'}
1916
+ >
1917
+ </collection-browser>`,
1918
+ );
1919
+
1920
+ el.baseQuery = 'web-archive';
1921
+ await el.updateComplete;
1922
+ await el.initialSearchComplete;
1923
+ await nextTick();
1924
+
1925
+ console.log(
1926
+ '\n\n*****\n\n*****\n\n',
1927
+ el.dataSource.getAllPages(),
1928
+ '\n\n*****\n\n*****\n\n',
1929
+ );
1930
+ expect(el.dataSource.totalResults, 'total results').to.equal(1);
1931
+ expect(el.dataSource.getTileModelAt(0)?.title).to.equal(
1932
+ 'https://example.com',
1933
+ );
1934
+ expect(
1935
+ el.dataSource.getTileModelAt(0)?.captureDates?.length,
1936
+ 'capture dates',
1937
+ ).to.equal(1);
1938
+ });
1939
+
1940
+ it('shows dropdown accordion in facet sidebar when opt-in strategy is specified', async () => {
1941
+ const searchService = new MockSearchService();
1942
+ const el = await fixture<CollectionBrowser>(
1943
+ html`<collection-browser
1944
+ .searchService=${searchService}
1945
+ facetLoadStrategy=${'opt-in'}
1946
+ >
1947
+ </collection-browser>`,
1948
+ );
1949
+
1950
+ el.baseQuery = 'foo';
1951
+ await el.updateComplete;
1952
+ await el.initialSearchComplete;
1953
+
1954
+ const facetsDropdown = el.shadowRoot?.querySelector(
1955
+ '.desktop-facets-dropdown',
1956
+ );
1957
+ expect(facetsDropdown).to.exist;
1958
+ });
1959
+
1960
+ it('shows temporarily unavailable message when facets suppressed', async () => {
1961
+ const searchService = new MockSearchService();
1962
+ const el = await fixture<CollectionBrowser>(
1963
+ html`<collection-browser
1964
+ .searchService=${searchService}
1965
+ facetLoadStrategy=${'off'}
1966
+ >
1967
+ </collection-browser>`,
1968
+ );
1969
+
1970
+ el.baseQuery = 'foo';
1971
+ await el.updateComplete;
1972
+ await el.initialSearchComplete;
1973
+
1974
+ const facetsMsg = el.shadowRoot?.querySelector('.facets-message');
1975
+ expect(facetsMsg).to.exist;
1976
+ expect(facetsMsg?.textContent?.trim()).to.equal(
1977
+ 'Facets are temporarily unavailable.',
1978
+ );
1979
+ });
1980
+
1981
+ it('shows manage bar interface instead of sort bar when in manage view', async () => {
1982
+ const searchService = new MockSearchService();
1983
+ const el = await fixture<CollectionBrowser>(
1984
+ html`<collection-browser .searchService=${searchService}>
1985
+ </collection-browser>`,
1986
+ );
1987
+
1988
+ el.baseQuery = 'foo';
1989
+ await el.updateComplete;
1990
+ await el.initialSearchComplete;
1991
+
1992
+ el.isManageView = true;
1993
+ await el.updateComplete;
1994
+
1995
+ expect(el.shadowRoot?.querySelector('manage-bar')).to.exist;
1996
+ expect(el.shadowRoot?.querySelector('sort-filter-bar')).not.to.exist;
1997
+
1998
+ el.isManageView = false;
1999
+ await el.updateComplete;
2000
+
2001
+ expect(el.shadowRoot?.querySelector('manage-bar')).not.to.exist;
2002
+ expect(el.shadowRoot?.querySelector('sort-filter-bar')).to.exist;
2003
+ });
2004
+
2005
+ it('switches to grid display mode when manage view activated', async () => {
2006
+ const searchService = new MockSearchService();
2007
+ const el = await fixture<CollectionBrowser>(
2008
+ html`<collection-browser
2009
+ .searchService=${searchService}
2010
+ .baseQuery=${'foo'}
2011
+ .displayMode=${'list-detail'}
2012
+ >
2013
+ </collection-browser>`,
2014
+ );
2015
+
2016
+ el.isManageView = true;
2017
+ await el.updateComplete;
2018
+
2019
+ expect(el.displayMode).to.equal('grid');
2020
+ });
2021
+
2022
+ it('can remove all checked tiles', async () => {
2023
+ const searchService = new MockSearchService();
2024
+ const el = await fixture<CollectionBrowser>(
2025
+ html`<collection-browser
2026
+ .searchService=${searchService}
2027
+ .baseNavigationUrl=${''}
2028
+ >
2029
+ </collection-browser>`,
2030
+ );
2031
+
2032
+ el.baseQuery = 'foo';
2033
+ el.pageSize = 1; // To hit the edge case of a page break while offsetting tiles
2034
+ await el.updateComplete;
2035
+ await el.initialSearchComplete;
2036
+
2037
+ el.isManageView = true;
2038
+ await el.updateComplete;
2039
+
2040
+ const infiniteScroller = el.shadowRoot?.querySelector('infinite-scroller');
2041
+ expect(infiniteScroller).to.exist;
2042
+
2043
+ let tiles =
2044
+ infiniteScroller!.shadowRoot?.querySelectorAll('tile-dispatcher');
2045
+ expect(tiles).to.exist;
2046
+ expect(tiles?.length).to.equal(2);
2047
+
2048
+ const firstTile = tiles![0] as TileDispatcher;
2049
+ const firstTileLink = firstTile.shadowRoot?.querySelector(
2050
+ 'a[href]',
2051
+ ) as HTMLAnchorElement;
2052
+ expect(firstTile.model?.identifier).to.equal('foo');
2053
+ expect(firstTileLink).to.exist;
2054
+
2055
+ // No effect if no tiles checked
2056
+ el.removeCheckedTiles();
2057
+ await el.updateComplete;
2058
+ tiles = infiniteScroller!.shadowRoot?.querySelectorAll('tile-dispatcher');
2059
+ expect(tiles).to.exist;
2060
+ expect(tiles?.length).to.equal(2);
2061
+
2062
+ // Check the first tile
2063
+ firstTileLink!.click();
2064
+ expect(firstTile.model?.checked).to.be.true;
2065
+
2066
+ // Remove checked tiles and verify that we only kept the second tile
2067
+ el.removeCheckedTiles();
2068
+ await el.updateComplete;
2069
+ expect(el?.dataSource?.size, 'data source count').to.equal(1);
2070
+
2071
+ tiles = el.shadowRoot
2072
+ ?.querySelector('infinite-scroller')!
2073
+ .shadowRoot?.querySelectorAll('tile-dispatcher');
2074
+ expect(tiles).to.exist;
2075
+ expect(
2076
+ tiles!.length,
2077
+ 'tile count after `el.removeCheckedTiles()`',
2078
+ ).to.equal(1);
2079
+ });
2080
+
2081
+ it('can check/uncheck all tiles', async () => {
2082
+ const searchService = new MockSearchService();
2083
+ const el = await fixture<CollectionBrowser>(
2084
+ html`<collection-browser
2085
+ .searchService=${searchService}
2086
+ .baseNavigationUrl=${''}
2087
+ >
2088
+ </collection-browser>`,
2089
+ );
2090
+
2091
+ el.baseQuery = 'foo';
2092
+ await el.updateComplete;
2093
+ await el.initialSearchComplete;
2094
+
2095
+ el.isManageView = true;
2096
+ await el.updateComplete;
2097
+
2098
+ expect(el.dataSource.checkedTileModels.length).to.equal(0);
2099
+ expect(el.dataSource.uncheckedTileModels.length).to.equal(2);
2100
+
2101
+ el.dataSource.checkAllTiles();
2102
+ expect(el.dataSource.checkedTileModels.length).to.equal(2);
2103
+ expect(el.dataSource.uncheckedTileModels.length).to.equal(0);
2104
+
2105
+ el.dataSource.uncheckAllTiles();
2106
+ expect(el.dataSource.checkedTileModels.length).to.equal(0);
2107
+ expect(el.dataSource.uncheckedTileModels.length).to.equal(2);
2108
+ });
2109
+
2110
+ it('emits event when manage view state changes', async () => {
2111
+ const spy = sinon.spy();
2112
+ const searchService = new MockSearchService();
2113
+ const el = await fixture<CollectionBrowser>(
2114
+ html`<collection-browser
2115
+ .searchService=${searchService}
2116
+ .baseNavigationUrl=${''}
2117
+ @manageModeChanged=${spy}
2118
+ ></collection-browser>`,
2119
+ );
2120
+
2121
+ el.isManageView = true;
2122
+ await el.updateComplete;
2123
+
2124
+ expect(spy.callCount).to.equal(1);
2125
+ expect(spy.args[0][0]?.detail).to.be.true;
2126
+
2127
+ el.isManageView = false;
2128
+ await el.updateComplete;
2129
+
2130
+ expect(spy.callCount).to.equal(2);
2131
+ expect(spy.args[1][0]?.detail).to.be.false;
2132
+ });
2133
+
2134
+ it('emits event when item removal requested', async () => {
2135
+ const spy = sinon.spy();
2136
+ const searchService = new MockSearchService();
2137
+ const el = await fixture<CollectionBrowser>(
2138
+ html`<collection-browser
2139
+ .searchService=${searchService}
2140
+ .baseNavigationUrl=${''}
2141
+ @itemRemovalRequested=${spy}
2142
+ >
2143
+ </collection-browser>`,
2144
+ );
2145
+
2146
+ el.baseQuery = 'foo';
2147
+ await el.updateComplete;
2148
+ await el.initialSearchComplete;
2149
+
2150
+ el.isManageView = true;
2151
+ await el.updateComplete;
2152
+
2153
+ const infiniteScroller = el.shadowRoot?.querySelector('infinite-scroller');
2154
+ expect(infiniteScroller).to.exist;
2155
+
2156
+ const tiles =
2157
+ infiniteScroller!.shadowRoot?.querySelectorAll('tile-dispatcher');
2158
+ expect(tiles).to.exist.and.have.length(2);
2159
+
2160
+ const firstTile = tiles![0] as TileDispatcher;
2161
+ const firstTileLink = firstTile.shadowRoot?.querySelector(
2162
+ 'a[href]',
2163
+ ) as HTMLAnchorElement;
2164
+ expect(firstTile.model?.identifier).to.equal('foo');
2165
+ expect(firstTileLink).to.exist;
2166
+
2167
+ // Check the first tile
2168
+ firstTileLink!.click();
2169
+ await el.updateComplete;
2170
+
2171
+ const manageBar = el.shadowRoot?.querySelector('manage-bar');
2172
+ expect(manageBar).to.exist;
2173
+
2174
+ // Emit remove event from manage bar
2175
+ manageBar!.dispatchEvent(new CustomEvent('removeItems'));
2176
+
2177
+ await el.updateComplete;
2178
+ expect(spy.callCount).to.equal(1);
2179
+ expect(spy.args[0].length).to.equal(1);
2180
+ expect(spy.args[0][0].detail.items[0]).to.equal('foo');
2181
+ });
2182
+
2183
+ it('disables manage view when manage bar cancelled', async () => {
2184
+ const searchService = new MockSearchService();
2185
+ const el = await fixture<CollectionBrowser>(
2186
+ html`<collection-browser
2187
+ .searchService=${searchService}
2188
+ .baseNavigationUrl=${''}
2189
+ >
2190
+ </collection-browser>`,
2191
+ );
2192
+
2193
+ el.baseQuery = 'foo';
2194
+ await el.updateComplete;
2195
+ await el.initialSearchComplete;
2196
+
2197
+ el.isManageView = true;
2198
+ await el.updateComplete;
2199
+
2200
+ const manageBar = el.shadowRoot?.querySelector('manage-bar');
2201
+ expect(manageBar).to.exist;
2202
+
2203
+ // Emit remove event from manage bar
2204
+ manageBar!.dispatchEvent(new CustomEvent('cancel'));
2205
+
2206
+ await el.updateComplete;
2207
+ expect(el.isManageView).to.be.false;
2208
+ });
2209
+
2210
+ it('enable/disable manage view delete button when you selectAll/unselectAll', async () => {
2211
+ const searchService = new MockSearchService();
2212
+ const el = await fixture<CollectionBrowser>(
2213
+ html`<collection-browser .searchService=${searchService}>
2214
+ </collection-browser>`,
2215
+ );
2216
+
2217
+ el.baseQuery = 'foo';
2218
+ await el.updateComplete;
2219
+ await el.initialSearchComplete;
2220
+
2221
+ el.isManageView = true;
2222
+ await el.updateComplete;
2223
+
2224
+ const manageBar = el.shadowRoot?.querySelector('manage-bar');
2225
+ expect(manageBar).to.exist;
2226
+
2227
+ // disable button exists
2228
+ expect(manageBar?.shadowRoot?.querySelector('.danger:disabled')).to.be
2229
+ .exist;
2230
+
2231
+ // Emit remove event from manage bar
2232
+ manageBar!.dispatchEvent(new CustomEvent('selectAll'));
2233
+ await el.updateComplete;
2234
+
2235
+ // disable button does not exists
2236
+ expect(manageBar?.shadowRoot?.querySelector('.danger:disabled')).to.be.not
2237
+ .exist;
2238
+
2239
+ // Emit remove event from manage bar
2240
+ manageBar!.dispatchEvent(new CustomEvent('unselectAll'));
2241
+ await el.updateComplete;
2242
+
2243
+ // disable button exists again
2244
+ expect(manageBar?.shadowRoot?.querySelector('.danger:disabled')).to.be
2245
+ .exist;
2246
+ });
2247
+
2248
+ it('shows Blurring checkbox for admin users', async () => {
2249
+ const searchService = new MockSearchService();
2250
+ const el = await fixture<CollectionBrowser>(
2251
+ html`<collection-browser
2252
+ .baseNavigationUrl=${''}
2253
+ .searchService=${searchService}
2254
+ >
2255
+ </collection-browser>`,
2256
+ );
2257
+
2258
+ el.baseQuery = 'archive-org-user-loggedin';
2259
+ await el.updateComplete;
2260
+ await el.initialSearchComplete;
2261
+
2262
+ const blurringCheck = el.shadowRoot?.querySelector(
2263
+ '#tile-blur-check',
2264
+ ) as HTMLInputElement;
2265
+ expect(blurringCheck).to.exist;
2266
+ expect(blurringCheck.checked).to.be.true;
2267
+ });
2268
+
2269
+ it('unchecks Blurring checkbox for admin users with blurring preference off', async () => {
2270
+ const searchService = new MockSearchService();
2271
+ const el = await fixture<CollectionBrowser>(
2272
+ html`<collection-browser
2273
+ .baseNavigationUrl=${''}
2274
+ .searchService=${searchService}
2275
+ >
2276
+ </collection-browser>`,
2277
+ );
2278
+
2279
+ el.baseQuery = 'archive-org-user-loggedin-noblur';
2280
+ await el.updateComplete;
2281
+ await el.initialSearchComplete;
2282
+
2283
+ const blurringCheck = el.shadowRoot?.querySelector(
2284
+ '#tile-blur-check',
2285
+ ) as HTMLInputElement;
2286
+ expect(blurringCheck).to.exist;
2287
+ expect(blurringCheck.checked).to.be.false;
2288
+ });
2289
+
2290
+ it('toggles blur state when Blurring checkbox is toggled', async () => {
2291
+ const searchService = new MockSearchService();
2292
+ const el = await fixture<CollectionBrowser>(
2293
+ html`<collection-browser
2294
+ .baseNavigationUrl=${''}
2295
+ .searchService=${searchService}
2296
+ >
2297
+ </collection-browser>`,
2298
+ );
2299
+
2300
+ el.baseQuery = 'archive-org-user-loggedin';
2301
+ await el.updateComplete;
2302
+ await el.initialSearchComplete;
2303
+
2304
+ const blurringCheck = el.shadowRoot?.querySelector(
2305
+ '#tile-blur-check',
2306
+ ) as HTMLInputElement;
2307
+ expect(blurringCheck).to.exist;
2308
+
2309
+ blurringCheck.dispatchEvent(new PointerEvent('click'));
2310
+ await el.updateComplete;
2311
+
2312
+ const infiniteScroller = el.shadowRoot?.querySelector(
2313
+ 'infinite-scroller',
2314
+ ) as InfiniteScroller;
2315
+ const firstTile = infiniteScroller?.shadowRoot?.querySelector(
2316
+ 'tile-dispatcher',
2317
+ ) as TileDispatcher;
2318
+ expect(firstTile.suppressBlurring).to.be.true;
2319
+ });
2320
+
2321
+ it('applies loans tab properties to sort bar', async () => {
2322
+ const searchService = new MockSearchService();
2323
+ const el = await fixture<CollectionBrowser>(
2324
+ html`<collection-browser
2325
+ .baseNavigationUrl=${''}
2326
+ .searchService=${searchService}
2327
+ .enableSortOptionsSlot=${true}
2328
+ >
2329
+ </collection-browser>`,
2330
+ );
2331
+
2332
+ el.baseQuery = 'collection:foo';
2333
+ await el.updateComplete;
2334
+ await aTimeout(10);
2335
+
2336
+ const sortBar = el.shadowRoot?.querySelector(
2337
+ 'sort-filter-bar',
2338
+ ) as SortFilterBar;
2339
+ expect(sortBar?.enableSortOptionsSlot, 'show loans in sort bar').to.be.true;
2340
+ expect(el.enableSortOptionsSlot, 'collection browser is loans tab').to.be
2341
+ .true;
2342
+
2343
+ const loansTabSlot = sortBar.querySelector('slot[name="sort-options"]');
2344
+ expect(loansTabSlot).to.exist;
2345
+ });
2346
+
2347
+ it('can suppress presence of result count', async () => {
2348
+ const searchService = new MockSearchService();
2349
+ const el = await fixture<CollectionBrowser>(
2350
+ html`<collection-browser
2351
+ .searchService=${searchService}
2352
+ suppressResultCount
2353
+ ></collection-browser>`,
2354
+ );
2355
+
2356
+ el.baseQuery = 'collection:foo';
2357
+ await el.updateComplete;
2358
+ await el.initialSearchComplete;
2359
+
2360
+ const resultCount = el.shadowRoot?.querySelector('#results-total');
2361
+ expect(resultCount).not.to.exist;
2362
+ });
2363
+
2364
+ it('can suppress presence of result tiles', async () => {
2365
+ const searchService = new MockSearchService();
2366
+ const el = await fixture<CollectionBrowser>(
2367
+ html`<collection-browser
2368
+ .searchService=${searchService}
2369
+ suppressResultTiles
2370
+ ></collection-browser>`,
2371
+ );
2372
+
2373
+ el.baseQuery = 'collection:foo';
2374
+ await el.updateComplete;
2375
+
2376
+ const infiniteScroller = el.shadowRoot?.querySelector('infinite-scroller');
2377
+ expect(infiniteScroller).not.to.exist;
2378
+ });
2379
+
2380
+ it('fetch larger result on search page for admin user to manage items', async () => {
2381
+ const resultsSpy = sinon.spy();
2382
+ const searchService = new MockSearchService({
2383
+ asyncResponse: true,
2384
+ resultsSpy,
2385
+ });
2386
+
2387
+ const el = await fixture<CollectionBrowser>(
2388
+ html`<collection-browser .searchService=${searchService}>
2389
+ </collection-browser>`,
2390
+ );
2391
+
2392
+ const numberOfPages = 15;
2393
+
2394
+ el.baseQuery = 'jack';
2395
+ el.isManageView = true;
2396
+ await el.dataSource.fetchPage(1, numberOfPages);
2397
+ await el.updateComplete;
2398
+
2399
+ const initialResults = el.dataSource.getAllPages();
2400
+ expect(Object.keys(initialResults).length).to.deep.equal(numberOfPages);
2401
+ });
2402
+ });