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