@internetarchive/collection-browser 4.4.0 → 4.5.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 (68) hide show
  1. package/dist/index.d.ts +1 -0
  2. package/dist/index.js.map +1 -1
  3. package/dist/src/app-root.d.ts +8 -0
  4. package/dist/src/app-root.js +26 -0
  5. package/dist/src/app-root.js.map +1 -1
  6. package/dist/src/collection-browser.d.ts +8 -0
  7. package/dist/src/collection-browser.js +19 -1
  8. package/dist/src/collection-browser.js.map +1 -1
  9. package/dist/src/collection-facets/facet-row.d.ts +6 -0
  10. package/dist/src/collection-facets/facet-row.js +158 -140
  11. package/dist/src/collection-facets/facet-row.js.map +1 -1
  12. package/dist/src/collection-facets/facets-template.js +25 -23
  13. package/dist/src/collection-facets/facets-template.js.map +1 -1
  14. package/dist/src/data-source/collection-browser-data-source.js.map +1 -1
  15. package/dist/src/styles/tile-action-styles.d.ts +14 -0
  16. package/dist/src/styles/tile-action-styles.js +59 -0
  17. package/dist/src/styles/tile-action-styles.js.map +1 -0
  18. package/dist/src/tiles/base-tile-component.d.ts +17 -1
  19. package/dist/src/tiles/base-tile-component.js +50 -1
  20. package/dist/src/tiles/base-tile-component.js.map +1 -1
  21. package/dist/src/tiles/grid/item-tile.js +139 -138
  22. package/dist/src/tiles/grid/item-tile.js.map +1 -1
  23. package/dist/src/tiles/item-image.js +28 -28
  24. package/dist/src/tiles/item-image.js.map +1 -1
  25. package/dist/src/tiles/list/tile-list-compact-header.js +71 -46
  26. package/dist/src/tiles/list/tile-list-compact-header.js.map +1 -1
  27. package/dist/src/tiles/list/tile-list-compact.d.ts +1 -1
  28. package/dist/src/tiles/list/tile-list-compact.js +138 -100
  29. package/dist/src/tiles/list/tile-list-compact.js.map +1 -1
  30. package/dist/src/tiles/list/tile-list.d.ts +1 -1
  31. package/dist/src/tiles/list/tile-list.js +316 -298
  32. package/dist/src/tiles/list/tile-list.js.map +1 -1
  33. package/dist/src/tiles/models.d.ts +11 -0
  34. package/dist/src/tiles/models.js.map +1 -1
  35. package/dist/src/tiles/tile-dispatcher.d.ts +14 -0
  36. package/dist/src/tiles/tile-dispatcher.js +319 -216
  37. package/dist/src/tiles/tile-dispatcher.js.map +1 -1
  38. package/dist/src/tiles/tile-display-value-provider.js +2 -1
  39. package/dist/src/tiles/tile-display-value-provider.js.map +1 -1
  40. package/dist/test/collection-facets/facet-row.test.js +55 -23
  41. package/dist/test/collection-facets/facet-row.test.js.map +1 -1
  42. package/dist/test/tiles/grid/item-tile.test.js +79 -79
  43. package/dist/test/tiles/grid/item-tile.test.js.map +1 -1
  44. package/dist/test/tiles/list/tile-list.test.js +136 -136
  45. package/dist/test/tiles/list/tile-list.test.js.map +1 -1
  46. package/dist/test/tiles/tile-dispatcher.test.js +101 -87
  47. package/dist/test/tiles/tile-dispatcher.test.js.map +1 -1
  48. package/index.ts +29 -28
  49. package/package.json +2 -2
  50. package/src/app-root.ts +30 -0
  51. package/src/collection-browser.ts +16 -1
  52. package/src/collection-facets/facet-row.ts +309 -299
  53. package/src/collection-facets/facets-template.ts +85 -83
  54. package/src/data-source/collection-browser-data-source.ts +1465 -1465
  55. package/src/styles/tile-action-styles.ts +59 -0
  56. package/src/tiles/base-tile-component.ts +124 -65
  57. package/src/tiles/grid/item-tile.ts +347 -346
  58. package/src/tiles/item-image.ts +214 -214
  59. package/src/tiles/list/tile-list-compact-header.ts +112 -86
  60. package/src/tiles/list/tile-list-compact.ts +278 -239
  61. package/src/tiles/list/tile-list.ts +718 -700
  62. package/src/tiles/models.ts +21 -8
  63. package/src/tiles/tile-dispatcher.ts +637 -527
  64. package/src/tiles/tile-display-value-provider.ts +133 -134
  65. package/test/collection-facets/facet-row.test.ts +421 -375
  66. package/test/tiles/grid/item-tile.test.ts +520 -520
  67. package/test/tiles/list/tile-list.test.ts +576 -576
  68. package/test/tiles/tile-dispatcher.test.ts +320 -300
@@ -1,375 +1,421 @@
1
- import { expect, fixture } from '@open-wc/testing';
2
- import sinon from 'sinon';
3
- import { html } from 'lit';
4
- import type { FacetRow } from '../../src/collection-facets/facet-row';
5
- import '../../src/collection-facets/facet-row';
6
- import type { FacetState } from '../../src/models';
7
-
8
- describe('Facet row', () => {
9
- it('renders nothing if no bucket provided', async () => {
10
- const el = await fixture<FacetRow>(
11
- html`<facet-row .facetType=${'subject'}></facet-row>`,
12
- );
13
-
14
- expect(el.shadowRoot?.querySelector('.facet-row-container')).not.to.exist;
15
- });
16
-
17
- it('renders nothing if no facet type provided', async () => {
18
- const bucket = {
19
- key: 'foo',
20
- state: 'none' as FacetState,
21
- count: 5,
22
- };
23
-
24
- const el = await fixture<FacetRow>(
25
- html`<facet-row .bucket=${bucket}></facet-row>`,
26
- );
27
-
28
- expect(el.shadowRoot?.querySelector('.facet-row-container')).not.to.exist;
29
- });
30
-
31
- it('renders provided bucket as facet row', async () => {
32
- const bucket = {
33
- key: 'foo',
34
- state: 'none' as FacetState,
35
- count: 5,
36
- };
37
-
38
- const el = await fixture<FacetRow>(
39
- html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
40
- );
41
-
42
- expect(el.shadowRoot?.querySelector('.facet-row-container')).to.exist;
43
- expect(
44
- el.shadowRoot?.querySelectorAll('input[type="checkbox"]'),
45
- ).to.have.length(2);
46
- expect(el.shadowRoot?.querySelectorAll('label')).to.have.length(2);
47
-
48
- const labelText = el.shadowRoot?.querySelector('.facet-info-display');
49
- expect(labelText?.textContent?.trim()).to.match(/^foo\s*5$/);
50
- });
51
-
52
- it('renders locale-appropriate facet count', async () => {
53
- const bucket = {
54
- key: 'foo',
55
- state: 'none' as FacetState,
56
- count: 54321,
57
- };
58
-
59
- const el = await fixture<FacetRow>(
60
- html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
61
- );
62
-
63
- const facetCount = el.shadowRoot?.querySelector('.facet-count');
64
- expect(facetCount?.textContent).to.equal('54,321');
65
- });
66
-
67
- it('renders selected facets with checked checkbox', async () => {
68
- const bucket = {
69
- key: 'foo',
70
- state: 'selected' as FacetState,
71
- count: 5,
72
- };
73
-
74
- const el = await fixture<FacetRow>(
75
- html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
76
- );
77
-
78
- // "Positive" checkbox is checked; "Negative" checkbox is not checked
79
- const selectCheckbox = el.shadowRoot?.querySelector(
80
- '.select-facet-checkbox',
81
- ) as HTMLInputElement;
82
- const hideCheckbox = el.shadowRoot?.querySelector(
83
- '.hide-facet-checkbox',
84
- ) as HTMLInputElement;
85
- expect(selectCheckbox?.checked).to.be.true;
86
- expect(hideCheckbox?.checked).to.be.false;
87
-
88
- // Eye icon is not in its active state
89
- expect(
90
- el.shadowRoot?.querySelector('.hide-facet-icon'),
91
- ).to.exist.and.satisfy(
92
- (icon: HTMLElement) => !icon.classList.contains('active'),
93
- );
94
- });
95
-
96
- it('renders hidden facets with closed eye icon', async () => {
97
- const bucket = {
98
- key: 'foo',
99
- state: 'hidden' as FacetState,
100
- count: 5,
101
- };
102
-
103
- const el = await fixture<FacetRow>(
104
- html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
105
- );
106
-
107
- // "Positive" checkbox is not checked; "Negative" checkbox is checked
108
- const selectCheckbox = el.shadowRoot?.querySelector(
109
- '.select-facet-checkbox',
110
- ) as HTMLInputElement;
111
- const hideCheckbox = el.shadowRoot?.querySelector(
112
- '.hide-facet-checkbox',
113
- ) as HTMLInputElement;
114
- expect(selectCheckbox?.checked).to.be.false;
115
- expect(hideCheckbox?.checked).to.be.true;
116
-
117
- // Eye icon is in its "active" state
118
- expect(
119
- el.shadowRoot?.querySelector('.hide-facet-icon'),
120
- ).to.exist.and.satisfy((icon: HTMLElement) =>
121
- icon.classList.contains('active'),
122
- );
123
- });
124
-
125
- it('renders correct accessible label for unchecked negative facets', async () => {
126
- const bucket = {
127
- key: 'foo',
128
- state: 'none' as FacetState,
129
- count: 5,
130
- };
131
-
132
- const el = await fixture<FacetRow>(
133
- html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
134
- );
135
-
136
- const hideFacetLabel = el.shadowRoot?.querySelector('.hide-facet-icon');
137
- expect(hideFacetLabel?.textContent?.trim()).to.match(/^Hide subject: foo$/);
138
- });
139
-
140
- it('renders correct accessible label for checked negative facets', async () => {
141
- const bucket = {
142
- key: 'foo',
143
- state: 'hidden' as FacetState,
144
- count: 5,
145
- };
146
-
147
- const el = await fixture<FacetRow>(
148
- html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
149
- );
150
-
151
- const hideFacetLabel = el.shadowRoot?.querySelector('.hide-facet-icon');
152
- expect(hideFacetLabel?.textContent?.trim()).to.match(
153
- /^Unhide subject: foo$/,
154
- );
155
- });
156
-
157
- it('renders collection facets as links', async () => {
158
- const bucket = {
159
- key: 'foo',
160
- state: 'none' as FacetState,
161
- count: 5,
162
- };
163
-
164
- const el = await fixture<FacetRow>(
165
- html`<facet-row
166
- .facetType=${'collection'}
167
- .bucket=${bucket}
168
- ></facet-row>`,
169
- );
170
-
171
- const collectionName = el.shadowRoot?.querySelector(
172
- '.facet-title > a:link',
173
- );
174
- expect(collectionName).to.exist;
175
- expect(collectionName?.getAttribute('href')).to.equal('/details/foo');
176
- });
177
-
178
- it('does not render non-collection facets as links', async () => {
179
- const bucket = {
180
- key: 'foo',
181
- state: 'none' as FacetState,
182
- count: 5,
183
- };
184
-
185
- const el = await fixture<FacetRow>(
186
- html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
187
- );
188
-
189
- expect(el.shadowRoot?.querySelector('a:link')).not.to.exist;
190
- });
191
-
192
- it('emits event when facet checkbox is clicked', async () => {
193
- const facetClickSpy = sinon.spy();
194
- const bucket = {
195
- key: 'foo',
196
- state: 'none' as FacetState,
197
- count: 5,
198
- };
199
-
200
- const el = await fixture<FacetRow>(
201
- html`<facet-row
202
- .facetType=${'subject'}
203
- .bucket=${bucket}
204
- @facetClick=${facetClickSpy}
205
- ></facet-row>`,
206
- );
207
-
208
- const positiveFacetCheck = el.shadowRoot?.querySelector(
209
- '.select-facet-checkbox',
210
- ) as HTMLInputElement;
211
- expect(positiveFacetCheck).to.exist;
212
- positiveFacetCheck.click();
213
-
214
- expect(facetClickSpy.callCount).to.equal(1);
215
- expect(facetClickSpy.lastCall.args[0]?.detail).to.deep.equal({
216
- facetType: 'subject',
217
- bucket: {
218
- key: 'foo',
219
- state: 'selected',
220
- count: 5,
221
- },
222
- negative: false,
223
- });
224
- });
225
-
226
- it('emits event when facet checkbox is unchecked', async () => {
227
- const facetClickSpy = sinon.spy();
228
- const bucket = {
229
- key: 'foo',
230
- state: 'selected' as FacetState,
231
- count: 5,
232
- };
233
-
234
- const el = await fixture<FacetRow>(
235
- html`<facet-row
236
- .facetType=${'subject'}
237
- .bucket=${bucket}
238
- @facetClick=${facetClickSpy}
239
- ></facet-row>`,
240
- );
241
-
242
- const positiveFacetCheck = el.shadowRoot?.querySelector(
243
- '.select-facet-checkbox',
244
- ) as HTMLInputElement;
245
- expect(positiveFacetCheck).to.exist;
246
- positiveFacetCheck.click();
247
-
248
- expect(facetClickSpy.callCount).to.equal(1);
249
- expect(facetClickSpy.lastCall.args[0]?.detail).to.deep.equal({
250
- facetType: 'subject',
251
- bucket: {
252
- key: 'foo',
253
- state: 'none',
254
- count: 5,
255
- },
256
- negative: false,
257
- });
258
- });
259
-
260
- it('emits event when facet negative icon is clicked', async () => {
261
- const facetClickSpy = sinon.spy();
262
- const bucket = {
263
- key: 'foo',
264
- state: 'none' as FacetState,
265
- count: 5,
266
- };
267
-
268
- const el = await fixture<FacetRow>(
269
- html`<facet-row
270
- .facetType=${'subject'}
271
- .bucket=${bucket}
272
- @facetClick=${facetClickSpy}
273
- ></facet-row>`,
274
- );
275
-
276
- const negativeFacetIcon = el.shadowRoot?.querySelector(
277
- '.hide-facet-icon',
278
- ) as HTMLLabelElement;
279
- expect(negativeFacetIcon).to.exist;
280
- negativeFacetIcon.click();
281
-
282
- expect(facetClickSpy.callCount).to.equal(1);
283
- expect(facetClickSpy.lastCall.args[0]?.detail).to.deep.equal({
284
- facetType: 'subject',
285
- bucket: {
286
- key: 'foo',
287
- state: 'hidden',
288
- count: 5,
289
- },
290
- negative: true,
291
- });
292
- });
293
-
294
- it('emits event when facet negative icon is unchecked', async () => {
295
- const facetClickSpy = sinon.spy();
296
- const bucket = {
297
- key: 'foo',
298
- state: 'hidden' as FacetState,
299
- count: 5,
300
- };
301
-
302
- const el = await fixture<FacetRow>(
303
- html`<facet-row
304
- .facetType=${'subject'}
305
- .bucket=${bucket}
306
- @facetClick=${facetClickSpy}
307
- ></facet-row>`,
308
- );
309
-
310
- const negativeFacetIcon = el.shadowRoot?.querySelector(
311
- '.hide-facet-icon',
312
- ) as HTMLLabelElement;
313
- expect(negativeFacetIcon).to.exist;
314
- negativeFacetIcon.click();
315
-
316
- expect(facetClickSpy.callCount).to.equal(1);
317
- expect(facetClickSpy.lastCall.args[0]?.detail).to.deep.equal({
318
- facetType: 'subject',
319
- bucket: {
320
- key: 'foo',
321
- state: 'none',
322
- count: 5,
323
- },
324
- negative: true,
325
- });
326
- });
327
-
328
- it('selects/deselects facet when label is clicked', async () => {
329
- const facetClickSpy = sinon.spy();
330
- const bucket = {
331
- key: 'foo',
332
- state: 'none' as FacetState,
333
- count: 5,
334
- };
335
-
336
- const el = await fixture<FacetRow>(
337
- html`<facet-row
338
- .facetType=${'subject'}
339
- .bucket=${bucket}
340
- @facetClick=${facetClickSpy}
341
- ></facet-row>`,
342
- );
343
-
344
- const facetLabel = el.shadowRoot?.querySelector(
345
- '.facet-info-display',
346
- ) as HTMLLabelElement;
347
- expect(facetLabel).to.exist;
348
-
349
- // Select facet by clicking label
350
- facetLabel.click();
351
- expect(facetClickSpy.callCount).to.equal(1);
352
- expect(facetClickSpy.lastCall.args[0]?.detail).to.deep.equal({
353
- facetType: 'subject',
354
- bucket: {
355
- key: 'foo',
356
- state: 'selected',
357
- count: 5,
358
- },
359
- negative: false,
360
- });
361
-
362
- // Deselect facet by clicking label
363
- facetLabel.click();
364
- expect(facetClickSpy.callCount).to.equal(2);
365
- expect(facetClickSpy.lastCall.args[0]?.detail).to.deep.equal({
366
- facetType: 'subject',
367
- bucket: {
368
- key: 'foo',
369
- state: 'none',
370
- count: 5,
371
- },
372
- negative: false,
373
- });
374
- });
375
- });
1
+ import { expect, fixture } from '@open-wc/testing';
2
+ import sinon from 'sinon';
3
+ import { html } from 'lit';
4
+ import type { FacetRow } from '../../src/collection-facets/facet-row';
5
+ import '../../src/collection-facets/facet-row';
6
+ import type { FacetState } from '../../src/models';
7
+
8
+ describe('Facet row', () => {
9
+ it('renders nothing if no bucket provided', async () => {
10
+ const el = await fixture<FacetRow>(
11
+ html`<facet-row .facetType=${'subject'}></facet-row>`,
12
+ );
13
+
14
+ expect(el.shadowRoot?.querySelector('.facet-row-container')).not.to.exist;
15
+ });
16
+
17
+ it('renders nothing if no facet type provided', async () => {
18
+ const bucket = {
19
+ key: 'foo',
20
+ state: 'none' as FacetState,
21
+ count: 5,
22
+ };
23
+
24
+ const el = await fixture<FacetRow>(
25
+ html`<facet-row .bucket=${bucket}></facet-row>`,
26
+ );
27
+
28
+ expect(el.shadowRoot?.querySelector('.facet-row-container')).not.to.exist;
29
+ });
30
+
31
+ it('renders provided bucket as facet row', async () => {
32
+ const bucket = {
33
+ key: 'foo',
34
+ state: 'none' as FacetState,
35
+ count: 5,
36
+ };
37
+
38
+ const el = await fixture<FacetRow>(
39
+ html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
40
+ );
41
+
42
+ expect(el.shadowRoot?.querySelector('.facet-row-container')).to.exist;
43
+ expect(
44
+ el.shadowRoot?.querySelectorAll('input[type="checkbox"]'),
45
+ ).to.have.length(2);
46
+ expect(el.shadowRoot?.querySelectorAll('label')).to.have.length(2);
47
+
48
+ const labelText = el.shadowRoot?.querySelector('.facet-info-display');
49
+ expect(labelText?.textContent?.trim()).to.match(/^foo\s*5$/);
50
+ });
51
+
52
+ it('renders locale-appropriate facet count', async () => {
53
+ const bucket = {
54
+ key: 'foo',
55
+ state: 'none' as FacetState,
56
+ count: 54321,
57
+ };
58
+
59
+ const el = await fixture<FacetRow>(
60
+ html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
61
+ );
62
+
63
+ const facetCount = el.shadowRoot?.querySelector('.facet-count');
64
+ expect(facetCount?.textContent).to.equal('54,321');
65
+ });
66
+
67
+ it('renders selected facets with checked checkbox', async () => {
68
+ const bucket = {
69
+ key: 'foo',
70
+ state: 'selected' as FacetState,
71
+ count: 5,
72
+ };
73
+
74
+ const el = await fixture<FacetRow>(
75
+ html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
76
+ );
77
+
78
+ // "Positive" checkbox is checked; "Negative" checkbox is not checked
79
+ const selectCheckbox = el.shadowRoot?.querySelector(
80
+ '.select-facet-checkbox',
81
+ ) as HTMLInputElement;
82
+ const hideCheckbox = el.shadowRoot?.querySelector(
83
+ '.hide-facet-checkbox',
84
+ ) as HTMLInputElement;
85
+ expect(selectCheckbox?.checked).to.be.true;
86
+ expect(hideCheckbox?.checked).to.be.false;
87
+
88
+ // Eye icon is not in its active state
89
+ expect(
90
+ el.shadowRoot?.querySelector('.hide-facet-icon'),
91
+ ).to.exist.and.satisfy(
92
+ (icon: HTMLElement) => !icon.classList.contains('active'),
93
+ );
94
+ });
95
+
96
+ it('renders hidden facets with closed eye icon', async () => {
97
+ const bucket = {
98
+ key: 'foo',
99
+ state: 'hidden' as FacetState,
100
+ count: 5,
101
+ };
102
+
103
+ const el = await fixture<FacetRow>(
104
+ html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
105
+ );
106
+
107
+ // "Positive" checkbox is not checked; "Negative" checkbox is checked
108
+ const selectCheckbox = el.shadowRoot?.querySelector(
109
+ '.select-facet-checkbox',
110
+ ) as HTMLInputElement;
111
+ const hideCheckbox = el.shadowRoot?.querySelector(
112
+ '.hide-facet-checkbox',
113
+ ) as HTMLInputElement;
114
+ expect(selectCheckbox?.checked).to.be.false;
115
+ expect(hideCheckbox?.checked).to.be.true;
116
+
117
+ // Eye icon is in its "active" state
118
+ expect(
119
+ el.shadowRoot?.querySelector('.hide-facet-icon'),
120
+ ).to.exist.and.satisfy((icon: HTMLElement) =>
121
+ icon.classList.contains('active'),
122
+ );
123
+ });
124
+
125
+ it('omits hide button when property is specified (and facet state is not hidden)', async () => {
126
+ const bucket = {
127
+ key: 'foo',
128
+ state: 'none' as FacetState,
129
+ count: 5,
130
+ };
131
+
132
+ const el = await fixture<FacetRow>(
133
+ html`<facet-row
134
+ .facetType=${'subject'}
135
+ .bucket=${bucket}
136
+ omitHideButton
137
+ ></facet-row>`,
138
+ );
139
+
140
+ // Hide button's container is hidden
141
+ expect(
142
+ el.shadowRoot
143
+ ?.querySelector('.hide-facet-container')
144
+ ?.hasAttribute('hidden'),
145
+ ).to.be.true;
146
+ });
147
+
148
+ it('does not omit hide button when facet state is hidden, even if property specified', async () => {
149
+ const bucket = {
150
+ key: 'foo',
151
+ state: 'hidden' as FacetState,
152
+ count: 5,
153
+ };
154
+
155
+ const el = await fixture<FacetRow>(
156
+ html`<facet-row
157
+ .facetType=${'subject'}
158
+ .bucket=${bucket}
159
+ omitHideButton
160
+ ></facet-row>`,
161
+ );
162
+
163
+ // Hide button's container is NOT hidden (i.e., we SHOW the hide button like normal)
164
+ expect(
165
+ el.shadowRoot
166
+ ?.querySelector('.hide-facet-container')
167
+ ?.hasAttribute('hidden'),
168
+ ).to.be.false;
169
+ });
170
+
171
+ it('renders correct accessible label for unchecked negative facets', async () => {
172
+ const bucket = {
173
+ key: 'foo',
174
+ state: 'none' as FacetState,
175
+ count: 5,
176
+ };
177
+
178
+ const el = await fixture<FacetRow>(
179
+ html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
180
+ );
181
+
182
+ const hideFacetLabel = el.shadowRoot?.querySelector('.hide-facet-icon');
183
+ expect(hideFacetLabel?.textContent?.trim()).to.match(/^Hide subject: foo$/);
184
+ });
185
+
186
+ it('renders correct accessible label for checked negative facets', async () => {
187
+ const bucket = {
188
+ key: 'foo',
189
+ state: 'hidden' as FacetState,
190
+ count: 5,
191
+ };
192
+
193
+ const el = await fixture<FacetRow>(
194
+ html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
195
+ );
196
+
197
+ const hideFacetLabel = el.shadowRoot?.querySelector('.hide-facet-icon');
198
+ expect(hideFacetLabel?.textContent?.trim()).to.match(
199
+ /^Unhide subject: foo$/,
200
+ );
201
+ });
202
+
203
+ it('renders collection facets as links', async () => {
204
+ const bucket = {
205
+ key: 'foo',
206
+ state: 'none' as FacetState,
207
+ count: 5,
208
+ };
209
+
210
+ const el = await fixture<FacetRow>(
211
+ html`<facet-row
212
+ .facetType=${'collection'}
213
+ .bucket=${bucket}
214
+ ></facet-row>`,
215
+ );
216
+
217
+ const collectionName = el.shadowRoot?.querySelector(
218
+ '.facet-title > a:link',
219
+ );
220
+ expect(collectionName).to.exist;
221
+ expect(collectionName?.getAttribute('href')).to.equal('/details/foo');
222
+ });
223
+
224
+ it('does not render non-collection facets as links', async () => {
225
+ const bucket = {
226
+ key: 'foo',
227
+ state: 'none' as FacetState,
228
+ count: 5,
229
+ };
230
+
231
+ const el = await fixture<FacetRow>(
232
+ html`<facet-row .facetType=${'subject'} .bucket=${bucket}></facet-row>`,
233
+ );
234
+
235
+ expect(el.shadowRoot?.querySelector('a:link')).not.to.exist;
236
+ });
237
+
238
+ it('emits event when facet checkbox is clicked', async () => {
239
+ const facetClickSpy = sinon.spy();
240
+ const bucket = {
241
+ key: 'foo',
242
+ state: 'none' as FacetState,
243
+ count: 5,
244
+ };
245
+
246
+ const el = await fixture<FacetRow>(
247
+ html`<facet-row
248
+ .facetType=${'subject'}
249
+ .bucket=${bucket}
250
+ @facetClick=${facetClickSpy}
251
+ ></facet-row>`,
252
+ );
253
+
254
+ const positiveFacetCheck = el.shadowRoot?.querySelector(
255
+ '.select-facet-checkbox',
256
+ ) as HTMLInputElement;
257
+ expect(positiveFacetCheck).to.exist;
258
+ positiveFacetCheck.click();
259
+
260
+ expect(facetClickSpy.callCount).to.equal(1);
261
+ expect(facetClickSpy.lastCall.args[0]?.detail).to.deep.equal({
262
+ facetType: 'subject',
263
+ bucket: {
264
+ key: 'foo',
265
+ state: 'selected',
266
+ count: 5,
267
+ },
268
+ negative: false,
269
+ });
270
+ });
271
+
272
+ it('emits event when facet checkbox is unchecked', async () => {
273
+ const facetClickSpy = sinon.spy();
274
+ const bucket = {
275
+ key: 'foo',
276
+ state: 'selected' as FacetState,
277
+ count: 5,
278
+ };
279
+
280
+ const el = await fixture<FacetRow>(
281
+ html`<facet-row
282
+ .facetType=${'subject'}
283
+ .bucket=${bucket}
284
+ @facetClick=${facetClickSpy}
285
+ ></facet-row>`,
286
+ );
287
+
288
+ const positiveFacetCheck = el.shadowRoot?.querySelector(
289
+ '.select-facet-checkbox',
290
+ ) as HTMLInputElement;
291
+ expect(positiveFacetCheck).to.exist;
292
+ positiveFacetCheck.click();
293
+
294
+ expect(facetClickSpy.callCount).to.equal(1);
295
+ expect(facetClickSpy.lastCall.args[0]?.detail).to.deep.equal({
296
+ facetType: 'subject',
297
+ bucket: {
298
+ key: 'foo',
299
+ state: 'none',
300
+ count: 5,
301
+ },
302
+ negative: false,
303
+ });
304
+ });
305
+
306
+ it('emits event when facet negative icon is clicked', async () => {
307
+ const facetClickSpy = sinon.spy();
308
+ const bucket = {
309
+ key: 'foo',
310
+ state: 'none' as FacetState,
311
+ count: 5,
312
+ };
313
+
314
+ const el = await fixture<FacetRow>(
315
+ html`<facet-row
316
+ .facetType=${'subject'}
317
+ .bucket=${bucket}
318
+ @facetClick=${facetClickSpy}
319
+ ></facet-row>`,
320
+ );
321
+
322
+ const negativeFacetIcon = el.shadowRoot?.querySelector(
323
+ '.hide-facet-icon',
324
+ ) as HTMLLabelElement;
325
+ expect(negativeFacetIcon).to.exist;
326
+ negativeFacetIcon.click();
327
+
328
+ expect(facetClickSpy.callCount).to.equal(1);
329
+ expect(facetClickSpy.lastCall.args[0]?.detail).to.deep.equal({
330
+ facetType: 'subject',
331
+ bucket: {
332
+ key: 'foo',
333
+ state: 'hidden',
334
+ count: 5,
335
+ },
336
+ negative: true,
337
+ });
338
+ });
339
+
340
+ it('emits event when facet negative icon is unchecked', async () => {
341
+ const facetClickSpy = sinon.spy();
342
+ const bucket = {
343
+ key: 'foo',
344
+ state: 'hidden' as FacetState,
345
+ count: 5,
346
+ };
347
+
348
+ const el = await fixture<FacetRow>(
349
+ html`<facet-row
350
+ .facetType=${'subject'}
351
+ .bucket=${bucket}
352
+ @facetClick=${facetClickSpy}
353
+ ></facet-row>`,
354
+ );
355
+
356
+ const negativeFacetIcon = el.shadowRoot?.querySelector(
357
+ '.hide-facet-icon',
358
+ ) as HTMLLabelElement;
359
+ expect(negativeFacetIcon).to.exist;
360
+ negativeFacetIcon.click();
361
+
362
+ expect(facetClickSpy.callCount).to.equal(1);
363
+ expect(facetClickSpy.lastCall.args[0]?.detail).to.deep.equal({
364
+ facetType: 'subject',
365
+ bucket: {
366
+ key: 'foo',
367
+ state: 'none',
368
+ count: 5,
369
+ },
370
+ negative: true,
371
+ });
372
+ });
373
+
374
+ it('selects/deselects facet when label is clicked', async () => {
375
+ const facetClickSpy = sinon.spy();
376
+ const bucket = {
377
+ key: 'foo',
378
+ state: 'none' as FacetState,
379
+ count: 5,
380
+ };
381
+
382
+ const el = await fixture<FacetRow>(
383
+ html`<facet-row
384
+ .facetType=${'subject'}
385
+ .bucket=${bucket}
386
+ @facetClick=${facetClickSpy}
387
+ ></facet-row>`,
388
+ );
389
+
390
+ const facetLabel = el.shadowRoot?.querySelector(
391
+ '.facet-info-display',
392
+ ) as HTMLLabelElement;
393
+ expect(facetLabel).to.exist;
394
+
395
+ // Select facet by clicking label
396
+ facetLabel.click();
397
+ expect(facetClickSpy.callCount).to.equal(1);
398
+ expect(facetClickSpy.lastCall.args[0]?.detail).to.deep.equal({
399
+ facetType: 'subject',
400
+ bucket: {
401
+ key: 'foo',
402
+ state: 'selected',
403
+ count: 5,
404
+ },
405
+ negative: false,
406
+ });
407
+
408
+ // Deselect facet by clicking label
409
+ facetLabel.click();
410
+ expect(facetClickSpy.callCount).to.equal(2);
411
+ expect(facetClickSpy.lastCall.args[0]?.detail).to.deep.equal({
412
+ facetType: 'subject',
413
+ bucket: {
414
+ key: 'foo',
415
+ state: 'none',
416
+ count: 5,
417
+ },
418
+ negative: false,
419
+ });
420
+ });
421
+ });