@internetarchive/collection-browser 4.3.1-alpha-webdev8257.0 → 4.3.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (41) hide show
  1. package/dist/index.d.ts +0 -1
  2. package/dist/index.js.map +1 -1
  3. package/dist/src/app-root.d.ts +0 -8
  4. package/dist/src/app-root.js +672 -698
  5. package/dist/src/app-root.js.map +1 -1
  6. package/dist/src/collection-browser.d.ts +0 -8
  7. package/dist/src/collection-browser.js +762 -779
  8. package/dist/src/collection-browser.js.map +1 -1
  9. package/dist/src/tiles/base-tile-component.d.ts +1 -17
  10. package/dist/src/tiles/base-tile-component.js +1 -48
  11. package/dist/src/tiles/base-tile-component.js.map +1 -1
  12. package/dist/src/tiles/grid/item-tile.js +138 -139
  13. package/dist/src/tiles/grid/item-tile.js.map +1 -1
  14. package/dist/src/tiles/list/tile-list-compact-header.js +46 -66
  15. package/dist/src/tiles/list/tile-list-compact-header.js.map +1 -1
  16. package/dist/src/tiles/list/tile-list-compact.d.ts +1 -1
  17. package/dist/src/tiles/list/tile-list-compact.js +100 -132
  18. package/dist/src/tiles/list/tile-list-compact.js.map +1 -1
  19. package/dist/src/tiles/list/tile-list.d.ts +1 -1
  20. package/dist/src/tiles/list/tile-list.js +298 -316
  21. package/dist/src/tiles/list/tile-list.js.map +1 -1
  22. package/dist/src/tiles/models.d.ts +0 -14
  23. package/dist/src/tiles/models.js.map +1 -1
  24. package/dist/src/tiles/tile-dispatcher.d.ts +0 -14
  25. package/dist/src/tiles/tile-dispatcher.js +216 -319
  26. package/dist/src/tiles/tile-dispatcher.js.map +1 -1
  27. package/index.ts +28 -29
  28. package/package.json +2 -2
  29. package/src/app-root.ts +1251 -1281
  30. package/src/collection-browser.ts +3049 -3063
  31. package/src/tiles/base-tile-component.ts +65 -121
  32. package/src/tiles/grid/item-tile.ts +346 -347
  33. package/src/tiles/list/tile-list-compact-header.ts +86 -106
  34. package/src/tiles/list/tile-list-compact.ts +239 -273
  35. package/src/tiles/list/tile-list.ts +700 -718
  36. package/src/tiles/models.ts +8 -24
  37. package/src/tiles/tile-dispatcher.ts +527 -637
  38. package/dist/src/styles/tile-action-styles.d.ts +0 -14
  39. package/dist/src/styles/tile-action-styles.js +0 -52
  40. package/dist/src/styles/tile-action-styles.js.map +0 -1
  41. package/src/styles/tile-action-styles.ts +0 -52
@@ -1,637 +1,527 @@
1
- import { css, html, nothing, PropertyValues } from 'lit';
2
- import { customElement, property, query } from 'lit/decorators.js';
3
- import { classMap } from 'lit/directives/class-map.js';
4
- import { ifDefined } from 'lit/directives/if-defined.js';
5
- import { msg } from '@lit/localize';
6
- import type {
7
- SharedResizeObserverInterface,
8
- SharedResizeObserverResizeHandlerInterface,
9
- } from '@internetarchive/shared-resize-observer';
10
- import type { TileDisplayMode } from '../models';
11
- import type { CollectionTitles } from '../data-source/models';
12
- import './grid/collection-tile';
13
- import './grid/item-tile';
14
- import './grid/account-tile';
15
- import './grid/search-tile';
16
- import './hover/tile-hover-pane';
17
- import './list/tile-list';
18
- import './list/tile-list-compact';
19
- import './list/tile-list-compact-header';
20
- import type { TileHoverPane } from './hover/tile-hover-pane';
21
- import { BaseTileComponent } from './base-tile-component';
22
- import { LayoutType } from './models';
23
- import {
24
- HoverPaneController,
25
- HoverPaneControllerInterface,
26
- HoverPaneProperties,
27
- HoverPaneProviderInterface,
28
- } from './hover/hover-pane-controller';
29
- import { srOnlyStyle } from '../styles/sr-only';
30
- import { tileActionStyles } from '../styles/tile-action-styles';
31
-
32
- @customElement('tile-dispatcher')
33
- export class TileDispatcher
34
- extends BaseTileComponent
35
- implements
36
- SharedResizeObserverResizeHandlerInterface,
37
- HoverPaneProviderInterface
38
- {
39
- /*
40
- * Reactive properties inherited from BaseTileComponent:
41
- * - model?: TileModel;
42
- * - tileActions: TileAction[] = [];
43
- * - currentWidth?: number;
44
- * - currentHeight?: number;
45
- * - baseNavigationUrl?: string;
46
- * - baseImageUrl?: string;
47
- * - collectionPagePath?: string;
48
- * - sortParam: SortParam | null = null;
49
- * - defaultSortParam: SortParam | null = null;
50
- * - creatorFilter?: string;
51
- * - mobileBreakpoint?: number;
52
- * - loggedIn = false;
53
- * - suppressTileBlurring = false;
54
- * - useLocalTime = false;
55
- */
56
-
57
- @property({ type: String }) tileDisplayMode?: TileDisplayMode;
58
-
59
- @property({ type: Boolean }) isManageView = false;
60
-
61
- @property({ type: Object }) resizeObserver?: SharedResizeObserverInterface;
62
-
63
- @property({ type: Object })
64
- collectionTitles?: CollectionTitles;
65
-
66
- @property({ type: Boolean }) showTvClips = false;
67
-
68
- /** What type of simplified layout to use in grid mode, if any */
69
- @property({ type: String }) layoutType: LayoutType = 'default';
70
-
71
- /** Whether this tile should include a hover pane at all (for applicable tile modes) */
72
- @property({ type: Boolean }) enableHoverPane = false;
73
-
74
- @property({ type: String }) manageCheckTitle = msg(
75
- 'Remove this item from the list',
76
- );
77
-
78
- private hoverPaneController?: HoverPaneControllerInterface;
79
-
80
- @query('#container')
81
- private container!: HTMLDivElement;
82
-
83
- @query('tile-hover-pane')
84
- private hoverPane?: TileHoverPane;
85
-
86
- @query('.tile-link')
87
- private tileLinkElement?: HTMLAnchorElement;
88
-
89
- acquireFocus(): void {
90
- this.tileLinkElement?.focus();
91
- }
92
-
93
- releaseFocus(): void {
94
- this.tileLinkElement?.blur();
95
- }
96
-
97
- /** Maps each display mode to whether hover panes should appear in that mode */
98
- private static readonly HOVER_PANE_DISPLAY_MODES: Record<
99
- TileDisplayMode,
100
- boolean
101
- > = {
102
- grid: true,
103
- 'list-compact': true,
104
- 'list-detail': false,
105
- 'list-header': false,
106
- };
107
-
108
- render() {
109
- const isGridMode = this.tileDisplayMode === 'grid';
110
- const hasTileActions = isGridMode && this.showGridTileActions;
111
- const hoverPaneTemplate =
112
- this.hoverPaneController?.getTemplate() ?? nothing;
113
- const containerClasses = classMap({
114
- hoverable: isGridMode,
115
- 'has-tile-actions': hasTileActions,
116
- });
117
- return html`
118
- <div id="container" class=${containerClasses}>
119
- ${this.tileDisplayMode === 'list-header'
120
- ? this.headerTemplate
121
- : this.tileTemplate}
122
- ${this.gridTileActionsTemplate} ${this.manageCheckTemplate}
123
- ${hoverPaneTemplate}
124
- </div>
125
- `;
126
- }
127
-
128
- protected firstUpdated(): void {
129
- if (this.shouldPrepareHoverPane) {
130
- this.hoverPaneController = new HoverPaneController(this, {
131
- mobileBreakpoint: this.mobileBreakpoint,
132
- enableLongPress: false,
133
- });
134
- }
135
- }
136
-
137
- private get headerTemplate() {
138
- const { currentWidth, sortParam, defaultSortParam, mobileBreakpoint } =
139
- this;
140
- return html`
141
- <tile-list-compact-header
142
- class="header"
143
- .currentWidth=${currentWidth}
144
- .sortParam=${sortParam ?? defaultSortParam}
145
- .mobileBreakpoint=${mobileBreakpoint}
146
- .tileActions=${this.tileActions}
147
- >
148
- </tile-list-compact-header>
149
- `;
150
- }
151
-
152
- private get tileTemplate() {
153
- return html`
154
- ${this.tileDisplayMode === 'list-detail'
155
- ? this.tile
156
- : this.linkTileTemplate}
157
- `;
158
- }
159
-
160
- private get linkTileTemplate() {
161
- return html`
162
- <a
163
- href=${this.linkTileHref}
164
- aria-label=${this.model?.title ?? 'Untitled item'}
165
- aria-describedby="link-aria-description"
166
- aria-haspopup=${this.shouldPrepareHoverPane ? 'dialog' : 'false'}
167
- title=${this.shouldPrepareHoverPane
168
- ? nothing // Don't show title tooltips when we have the tile info popups
169
- : ifDefined(this.model?.title)}
170
- @click=${this.handleLinkClicked}
171
- @contextmenu=${this.handleLinkContextMenu}
172
- class="tile-link"
173
- >
174
- ${this.tile}
175
- </a>
176
- <div id="link-aria-description" class="sr-only">
177
- ${msg('Press Down Arrow to preview item details')}
178
- </div>
179
- `;
180
- }
181
-
182
- private get linkTileHref(): string | typeof nothing {
183
- if (!this.model?.identifier || this.baseNavigationUrl == null)
184
- return nothing;
185
-
186
- // Use the server-specified href if available.
187
- // Otherwise, construct a details page URL from the item identifier.
188
- if (this.model.href) {
189
- return `${this.baseNavigationUrl}${this.model.href}`;
190
- }
191
-
192
- return this.displayValueProvider.itemPageUrl(
193
- this.model.identifier,
194
- this.model.mediatype === 'collection',
195
- );
196
- }
197
-
198
- private get manageCheckTemplate() {
199
- if (!this.isManageView || this.tileDisplayMode !== 'grid') return nothing;
200
-
201
- return html`
202
- <div class="manage-check">
203
- <input
204
- type="checkbox"
205
- title=${this.manageCheckTitle}
206
- ?checked=${this.model?.checked}
207
- @change=${this.handleLinkClicked}
208
- />
209
- </div>
210
- `;
211
- }
212
-
213
- /**
214
- * Whether hover pane behavior should be prepared for this tile
215
- * (e.g., whether mouse listeners should be attached, etc.)
216
- */
217
- private get shouldPrepareHoverPane(): boolean {
218
- return (
219
- this.enableHoverPane &&
220
- !!this.tileDisplayMode &&
221
- TileDispatcher.HOVER_PANE_DISPLAY_MODES[this.tileDisplayMode] &&
222
- this.model?.mediatype !== 'search' && // don't show hover panes on search tiles
223
- !this.model?.captureDates // don't show hover panes on web archive tiles
224
- );
225
- }
226
-
227
- private get isHoverEnabled(): boolean {
228
- return window.matchMedia('(hover: hover)').matches;
229
- }
230
-
231
- /**
232
- * Whether the info button should be shown on this tile.
233
- * Only shown on touch/non-hover devices where a hover pane is available,
234
- * so the button always has something to toggle.
235
- */
236
- private get shouldShowInfoButton(): boolean {
237
- return !this.isHoverEnabled && this.shouldPrepareHoverPane;
238
- }
239
-
240
- /** @inheritdoc */
241
- getHoverPane(): TileHoverPane | undefined {
242
- return this.hoverPane;
243
- }
244
-
245
- /** @inheritdoc */
246
- getHoverPaneProps(): HoverPaneProperties {
247
- return this;
248
- }
249
-
250
- handleResize(entry: ResizeObserverEntry): void {
251
- this.currentWidth = entry.contentRect.width;
252
- this.currentHeight = entry.contentRect.height;
253
- }
254
-
255
- disconnectedCallback(): void {
256
- this.stopResizeObservation(this.resizeObserver);
257
- }
258
-
259
- private stopResizeObservation(observer?: SharedResizeObserverInterface) {
260
- observer?.removeObserver({
261
- handler: this,
262
- target: this.container,
263
- });
264
- }
265
-
266
- private startResizeObservation() {
267
- this.stopResizeObservation(this.resizeObserver);
268
- this.resizeObserver?.addObserver({
269
- handler: this,
270
- target: this.container,
271
- });
272
- }
273
-
274
- updated(props: PropertyValues) {
275
- if (props.has('resizeObserver')) {
276
- const previousObserver = props.get(
277
- 'resizeObserver',
278
- ) as SharedResizeObserverInterface;
279
- this.stopResizeObservation(previousObserver);
280
- this.startResizeObservation();
281
- }
282
- }
283
-
284
- /**
285
- * Handler for when the tile link is left-clicked. Emits the `resultSelected` event.
286
- * In manage view, it also checks/unchecks the tile.
287
- */
288
- private handleLinkClicked(e: Event): void {
289
- if (this.isManageView) {
290
- e.preventDefault();
291
- if (this.model) this.model.checked = !this.model.checked;
292
- }
293
-
294
- this.dispatchEvent(
295
- new CustomEvent('resultSelected', { detail: this.model }),
296
- );
297
- }
298
-
299
- /**
300
- * Handler for when the tile link is right-clicked.
301
- * In manage view, it opens the item in a new tab. Otherwise, does nothing.
302
- */
303
- private handleLinkContextMenu(e: Event): void {
304
- if (this.isManageView && this.linkTileHref !== nothing) {
305
- e.preventDefault();
306
- window.open(this.linkTileHref, '_blank');
307
- }
308
- }
309
-
310
- private tileInfoButtonPressed(
311
- e: CustomEvent<{ x: number; y: number }>,
312
- ): void {
313
- this.hoverPaneController?.toggleHoverPane({
314
- coords: e.detail,
315
- enableTouchBackdrop: true,
316
- });
317
- }
318
-
319
- /** Whether tile action buttons should be rendered in grid mode */
320
- private get showGridTileActions(): boolean {
321
- return (
322
- this.tileActions.length > 0 &&
323
- !this.isManageView &&
324
- this.tileDisplayMode === 'grid'
325
- );
326
- }
327
-
328
- /**
329
- * Template for the grid-mode action buttons. Rendered alongside the inner
330
- * tile link inside the dispatcher's shadow root so the action buttons can
331
- * suppress the hover pane on hover.
332
- */
333
- private get gridTileActionsTemplate() {
334
- if (!this.showGridTileActions) return nothing;
335
-
336
- return html`
337
- <div
338
- class="tile-actions grid-tile-actions"
339
- @mouseenter=${this.handleGridActionsMouseEnter}
340
- @mousemove=${(e: Event) => e.stopPropagation()}
341
- >
342
- ${this.tileActions.map(
343
- action => html`
344
- <button
345
- class="tile-action-btn"
346
- @click=${(e: Event) => this.handleTileActionClick(e, action)}
347
- >
348
- ${action.label}
349
- </button>
350
- `,
351
- )}
352
- </div>
353
- `;
354
- }
355
-
356
- /**
357
- * When the mouse enters the grid-mode tile actions area, dispatch a
358
- * synthetic mouseleave on the host to cancel the hover pane's show timer
359
- * and hide any visible pane.
360
- */
361
- private handleGridActionsMouseEnter = (): void => {
362
- this.dispatchEvent(new MouseEvent('mouseleave', { bubbles: false }));
363
- };
364
-
365
- private get tile() {
366
- const {
367
- model,
368
- collectionPagePath,
369
- baseNavigationUrl,
370
- currentWidth,
371
- currentHeight,
372
- sortParam,
373
- creatorFilter,
374
- mobileBreakpoint,
375
- defaultSortParam,
376
- } = this;
377
-
378
- if (!model) return nothing;
379
-
380
- switch (this.tileDisplayMode) {
381
- case 'grid':
382
- switch (model.mediatype) {
383
- case 'collection':
384
- return html`<collection-tile
385
- .model=${model}
386
- .collectionPagePath=${collectionPagePath}
387
- .baseImageUrl=${this.baseImageUrl}
388
- .currentWidth=${currentWidth}
389
- .currentHeight=${currentHeight}
390
- .creatorFilter=${creatorFilter}
391
- .suppressBlurring=${this.suppressBlurring}
392
- .isManageView=${this.isManageView}
393
- .layoutType=${this.layoutType}
394
- ?showInfoButton=${this.shouldShowInfoButton}
395
- @infoButtonPressed=${this.tileInfoButtonPressed}
396
- >
397
- </collection-tile>`;
398
- case 'account':
399
- return html`<account-tile
400
- .model=${model}
401
- .collectionPagePath=${collectionPagePath}
402
- .baseImageUrl=${this.baseImageUrl}
403
- .currentWidth=${currentWidth}
404
- .currentHeight=${currentHeight}
405
- .creatorFilter=${creatorFilter}
406
- .suppressBlurring=${this.suppressBlurring}
407
- .isManageView=${this.isManageView}
408
- ?showInfoButton=${this.shouldShowInfoButton}
409
- @infoButtonPressed=${this.tileInfoButtonPressed}
410
- >
411
- </account-tile>`;
412
- case 'search':
413
- return html`<search-tile
414
- .model=${model}
415
- .collectionPagePath=${collectionPagePath}
416
- .baseImageUrl=${this.baseImageUrl}
417
- .currentWidth=${currentWidth}
418
- .currentHeight=${currentHeight}
419
- .creatorFilter=${creatorFilter}
420
- .suppressBlurring=${this.suppressBlurring}
421
- .isManageView=${this.isManageView}
422
- ?showInfoButton=${false}
423
- @infoButtonPressed=${this.tileInfoButtonPressed}
424
- >
425
- </search-tile>`;
426
- default:
427
- return html`<item-tile
428
- .model=${model}
429
- .collectionPagePath=${collectionPagePath}
430
- .currentWidth=${this.currentWidth}
431
- .currentHeight=${this.currentHeight}
432
- .baseImageUrl=${this.baseImageUrl}
433
- .sortParam=${sortParam}
434
- .defaultSortParam=${defaultSortParam}
435
- .creatorFilter=${creatorFilter}
436
- .loggedIn=${this.loggedIn}
437
- .suppressBlurring=${this.suppressBlurring}
438
- .isManageView=${this.isManageView}
439
- .layoutType=${this.layoutType}
440
- ?showTvClips=${this.showTvClips}
441
- ?showInfoButton=${this.shouldShowInfoButton}
442
- ?useLocalTime=${this.useLocalTime}
443
- @infoButtonPressed=${this.tileInfoButtonPressed}
444
- >
445
- </item-tile>`;
446
- }
447
- case 'list-compact':
448
- return html`<tile-list-compact
449
- .model=${model}
450
- .collectionPagePath=${collectionPagePath}
451
- .currentWidth=${currentWidth}
452
- .currentHeight=${currentHeight}
453
- .baseNavigationUrl=${baseNavigationUrl}
454
- .sortParam=${sortParam}
455
- .defaultSortParam=${defaultSortParam}
456
- .creatorFilter=${creatorFilter}
457
- .mobileBreakpoint=${mobileBreakpoint}
458
- .baseImageUrl=${this.baseImageUrl}
459
- .loggedIn=${this.loggedIn}
460
- .suppressBlurring=${this.suppressBlurring}
461
- .tileActions=${this.isManageView ? [] : this.tileActions}
462
- ?useLocalTime=${this.useLocalTime}
463
- >
464
- </tile-list-compact>`;
465
- case 'list-detail':
466
- return html`<tile-list
467
- .model=${model}
468
- .collectionPagePath=${collectionPagePath}
469
- .collectionTitles=${this.collectionTitles}
470
- .currentWidth=${currentWidth}
471
- .currentHeight=${currentHeight}
472
- .baseNavigationUrl=${baseNavigationUrl}
473
- .sortParam=${sortParam}
474
- .defaultSortParam=${defaultSortParam}
475
- .creatorFilter=${creatorFilter}
476
- .mobileBreakpoint=${mobileBreakpoint}
477
- .baseImageUrl=${this.baseImageUrl}
478
- .loggedIn=${this.loggedIn}
479
- .suppressBlurring=${this.suppressBlurring}
480
- .tileActions=${this.isManageView ? [] : this.tileActions}
481
- ?useLocalTime=${this.useLocalTime}
482
- >
483
- </tile-list>`;
484
- default:
485
- return nothing;
486
- }
487
- }
488
-
489
- static get styles() {
490
- return [
491
- srOnlyStyle,
492
- tileActionStyles,
493
- css`
494
- :host {
495
- display: block;
496
- height: 100%;
497
- }
498
-
499
- collection-tile {
500
- --tileBorderColor: #555555;
501
- --tileBackgroundColor: #666666;
502
- --imageBlockBackgroundColor: #666666;
503
- }
504
-
505
- account-tile {
506
- --tileBorderColor: #dddddd;
507
- --imageBlockBackgroundColor: #fcf5e6;
508
- }
509
-
510
- item-tile {
511
- --tileBorderColor: #dddddd;
512
- --imageBlockBackgroundColor: #f1f1f4;
513
- }
514
-
515
- search-tile {
516
- --tileBorderColor: #555555;
517
- --tileBackgroundColor: #666666;
518
- --imageBlockBackgroundColor: #666666;
519
- --iconFillColor: #2c2c2c;
520
- }
521
-
522
- #container {
523
- position: relative;
524
- height: 100%;
525
- border-radius: 4px;
526
- }
527
-
528
- /*
529
- * When tile actions are present, the container takes on the role of
530
- * the tile's visual card so the tile content and action row appear
531
- * as a single unified element. The inner tile's own shadow/radius
532
- * are disabled via CSS variable overrides to avoid visual
533
- * duplication, and the action row sits as a footer inside the same
534
- * card.
535
- */
536
- #container.has-tile-actions {
537
- display: flex;
538
- flex-direction: column;
539
- overflow: hidden;
540
- box-shadow: var(--tileShadow, 1px 1px 2px 0);
541
- --tileBoxShadow: none;
542
- --tileCornerRadius: 0;
543
- }
544
-
545
- #container.has-tile-actions .tile-link {
546
- flex: 1;
547
- min-height: 0;
548
- overflow: hidden;
549
- border-radius: 0;
550
- }
551
-
552
- /* Normal hover shadow lives on the inner anchor for plain tiles */
553
- #container.hoverable:not(.has-tile-actions) a:focus,
554
- #container.hoverable:not(.has-tile-actions) a:hover {
555
- box-shadow: var(
556
- --tileHoverBoxShadow,
557
- 0 0 6px 2px rgba(8, 8, 32, 0.8)
558
- );
559
- transition: box-shadow 0.1s ease;
560
- }
561
-
562
- /*
563
- * When the container owns the card visuals, the hover shadow needs
564
- * to move up to the container so it wraps the action row too.
565
- */
566
- #container.hoverable.has-tile-actions:hover {
567
- box-shadow: var(
568
- --tileHoverBoxShadow,
569
- 0 0 6px 2px rgba(8, 8, 32, 0.8)
570
- );
571
- transition: box-shadow 0.1s ease;
572
- }
573
-
574
- a {
575
- display: block;
576
- height: 100%;
577
- color: unset;
578
- text-decoration: none;
579
- transition: transform 0.05s ease;
580
- border-radius: 4px;
581
- outline: none;
582
- }
583
-
584
- a :first-child {
585
- display: block;
586
- height: 100%;
587
- }
588
-
589
- .manage-check {
590
- position: absolute;
591
- right: 0;
592
- top: 0;
593
- border: 5px solid #2c2c2c;
594
- border-radius: 3px;
595
- background-color: #2c2c2c;
596
- z-index: 1;
597
- }
598
-
599
- .manage-check > input[type='checkbox'] {
600
- display: block;
601
- margin: 0;
602
- }
603
-
604
- #touch-backdrop {
605
- position: fixed;
606
- width: 100vw;
607
- height: 100vh;
608
- top: 0;
609
- left: 0;
610
- z-index: 2;
611
- background: transparent;
612
- }
613
-
614
- tile-hover-pane {
615
- position: absolute;
616
- top: 0;
617
- left: -9999px;
618
- z-index: 2;
619
- }
620
-
621
- /*
622
- * Grid-mode action row sits flush against the bottom of the card —
623
- * the buttons' own borders form the visible bottom edge. The outer
624
- * buttons get rounded bottom corners to match the container so the
625
- * red border traces cleanly around the card's bottom corners.
626
- */
627
- .grid-tile-actions .tile-action-btn:first-child {
628
- border-bottom-left-radius: 4px;
629
- }
630
-
631
- .grid-tile-actions .tile-action-btn:last-child {
632
- border-bottom-right-radius: 4px;
633
- }
634
- `,
635
- ];
636
- }
637
- }
1
+ import { css, html, nothing, PropertyValues } from 'lit';
2
+ import { customElement, property, query } from 'lit/decorators.js';
3
+ import { ifDefined } from 'lit/directives/if-defined.js';
4
+ import { msg } from '@lit/localize';
5
+ import type {
6
+ SharedResizeObserverInterface,
7
+ SharedResizeObserverResizeHandlerInterface,
8
+ } from '@internetarchive/shared-resize-observer';
9
+ import type { TileDisplayMode } from '../models';
10
+ import type { CollectionTitles } from '../data-source/models';
11
+ import './grid/collection-tile';
12
+ import './grid/item-tile';
13
+ import './grid/account-tile';
14
+ import './grid/search-tile';
15
+ import './hover/tile-hover-pane';
16
+ import './list/tile-list';
17
+ import './list/tile-list-compact';
18
+ import './list/tile-list-compact-header';
19
+ import type { TileHoverPane } from './hover/tile-hover-pane';
20
+ import { BaseTileComponent } from './base-tile-component';
21
+ import { LayoutType } from './models';
22
+ import {
23
+ HoverPaneController,
24
+ HoverPaneControllerInterface,
25
+ HoverPaneProperties,
26
+ HoverPaneProviderInterface,
27
+ } from './hover/hover-pane-controller';
28
+ import { srOnlyStyle } from '../styles/sr-only';
29
+
30
+ @customElement('tile-dispatcher')
31
+ export class TileDispatcher
32
+ extends BaseTileComponent
33
+ implements
34
+ SharedResizeObserverResizeHandlerInterface,
35
+ HoverPaneProviderInterface
36
+ {
37
+ /*
38
+ * Reactive properties inherited from BaseTileComponent:
39
+ * - model?: TileModel;
40
+ * - currentWidth?: number;
41
+ * - currentHeight?: number;
42
+ * - baseNavigationUrl?: string;
43
+ * - baseImageUrl?: string;
44
+ * - collectionPagePath?: string;
45
+ * - sortParam: SortParam | null = null;
46
+ * - defaultSortParam: SortParam | null = null;
47
+ * - creatorFilter?: string;
48
+ * - mobileBreakpoint?: number;
49
+ * - loggedIn = false;
50
+ * - suppressTileBlurring = false;
51
+ * - useLocalTime = false;
52
+ */
53
+
54
+ @property({ type: String }) tileDisplayMode?: TileDisplayMode;
55
+
56
+ @property({ type: Boolean }) isManageView = false;
57
+
58
+ @property({ type: Object }) resizeObserver?: SharedResizeObserverInterface;
59
+
60
+ @property({ type: Object })
61
+ collectionTitles?: CollectionTitles;
62
+
63
+ @property({ type: Boolean }) showTvClips = false;
64
+
65
+ /** What type of simplified layout to use in grid mode, if any */
66
+ @property({ type: String }) layoutType: LayoutType = 'default';
67
+
68
+ /** Whether this tile should include a hover pane at all (for applicable tile modes) */
69
+ @property({ type: Boolean }) enableHoverPane = false;
70
+
71
+ @property({ type: String }) manageCheckTitle = msg(
72
+ 'Remove this item from the list',
73
+ );
74
+
75
+ private hoverPaneController?: HoverPaneControllerInterface;
76
+
77
+ @query('#container')
78
+ private container!: HTMLDivElement;
79
+
80
+ @query('tile-hover-pane')
81
+ private hoverPane?: TileHoverPane;
82
+
83
+ @query('.tile-link')
84
+ private tileLinkElement?: HTMLAnchorElement;
85
+
86
+ acquireFocus(): void {
87
+ this.tileLinkElement?.focus();
88
+ }
89
+
90
+ releaseFocus(): void {
91
+ this.tileLinkElement?.blur();
92
+ }
93
+
94
+ /** Maps each display mode to whether hover panes should appear in that mode */
95
+ private static readonly HOVER_PANE_DISPLAY_MODES: Record<
96
+ TileDisplayMode,
97
+ boolean
98
+ > = {
99
+ grid: true,
100
+ 'list-compact': true,
101
+ 'list-detail': false,
102
+ 'list-header': false,
103
+ };
104
+
105
+ render() {
106
+ const isGridMode = this.tileDisplayMode === 'grid';
107
+ const hoverPaneTemplate =
108
+ this.hoverPaneController?.getTemplate() ?? nothing;
109
+ return html`
110
+ <div id="container" class=${isGridMode ? 'hoverable' : ''}>
111
+ ${this.tileDisplayMode === 'list-header'
112
+ ? this.headerTemplate
113
+ : this.tileTemplate}
114
+ ${this.manageCheckTemplate} ${hoverPaneTemplate}
115
+ </div>
116
+ `;
117
+ }
118
+
119
+ protected firstUpdated(): void {
120
+ if (this.shouldPrepareHoverPane) {
121
+ this.hoverPaneController = new HoverPaneController(this, {
122
+ mobileBreakpoint: this.mobileBreakpoint,
123
+ enableLongPress: false,
124
+ });
125
+ }
126
+ }
127
+
128
+ private get headerTemplate() {
129
+ const { currentWidth, sortParam, defaultSortParam, mobileBreakpoint } =
130
+ this;
131
+ return html`
132
+ <tile-list-compact-header
133
+ class="header"
134
+ .currentWidth=${currentWidth}
135
+ .sortParam=${sortParam ?? defaultSortParam}
136
+ .mobileBreakpoint=${mobileBreakpoint}
137
+ >
138
+ </tile-list-compact-header>
139
+ `;
140
+ }
141
+
142
+ private get tileTemplate() {
143
+ return html`
144
+ ${this.tileDisplayMode === 'list-detail'
145
+ ? this.tile
146
+ : this.linkTileTemplate}
147
+ `;
148
+ }
149
+
150
+ private get linkTileTemplate() {
151
+ return html`
152
+ <a
153
+ href=${this.linkTileHref}
154
+ aria-label=${this.model?.title ?? 'Untitled item'}
155
+ aria-describedby="link-aria-description"
156
+ aria-haspopup=${this.shouldPrepareHoverPane ? 'dialog' : 'false'}
157
+ title=${this.shouldPrepareHoverPane
158
+ ? nothing // Don't show title tooltips when we have the tile info popups
159
+ : ifDefined(this.model?.title)}
160
+ @click=${this.handleLinkClicked}
161
+ @contextmenu=${this.handleLinkContextMenu}
162
+ class="tile-link"
163
+ >
164
+ ${this.tile}
165
+ </a>
166
+ <div id="link-aria-description" class="sr-only">
167
+ ${msg('Press Down Arrow to preview item details')}
168
+ </div>
169
+ `;
170
+ }
171
+
172
+ private get linkTileHref(): string | typeof nothing {
173
+ if (!this.model?.identifier || this.baseNavigationUrl == null)
174
+ return nothing;
175
+
176
+ // Use the server-specified href if available.
177
+ // Otherwise, construct a details page URL from the item identifier.
178
+ if (this.model.href) {
179
+ return `${this.baseNavigationUrl}${this.model.href}`;
180
+ }
181
+
182
+ return this.displayValueProvider.itemPageUrl(
183
+ this.model.identifier,
184
+ this.model.mediatype === 'collection',
185
+ );
186
+ }
187
+
188
+ private get manageCheckTemplate() {
189
+ if (!this.isManageView || this.tileDisplayMode !== 'grid') return nothing;
190
+
191
+ return html`
192
+ <div class="manage-check">
193
+ <input
194
+ type="checkbox"
195
+ title=${this.manageCheckTitle}
196
+ ?checked=${this.model?.checked}
197
+ @change=${this.handleLinkClicked}
198
+ />
199
+ </div>
200
+ `;
201
+ }
202
+
203
+ /**
204
+ * Whether hover pane behavior should be prepared for this tile
205
+ * (e.g., whether mouse listeners should be attached, etc.)
206
+ */
207
+ private get shouldPrepareHoverPane(): boolean {
208
+ return (
209
+ this.enableHoverPane &&
210
+ !!this.tileDisplayMode &&
211
+ TileDispatcher.HOVER_PANE_DISPLAY_MODES[this.tileDisplayMode] &&
212
+ this.model?.mediatype !== 'search' && // don't show hover panes on search tiles
213
+ !this.model?.captureDates // don't show hover panes on web archive tiles
214
+ );
215
+ }
216
+
217
+ private get isHoverEnabled(): boolean {
218
+ return window.matchMedia('(hover: hover)').matches;
219
+ }
220
+
221
+ /**
222
+ * Whether the info button should be shown on this tile.
223
+ * Only shown on touch/non-hover devices where a hover pane is available,
224
+ * so the button always has something to toggle.
225
+ */
226
+ private get shouldShowInfoButton(): boolean {
227
+ return !this.isHoverEnabled && this.shouldPrepareHoverPane;
228
+ }
229
+
230
+ /** @inheritdoc */
231
+ getHoverPane(): TileHoverPane | undefined {
232
+ return this.hoverPane;
233
+ }
234
+
235
+ /** @inheritdoc */
236
+ getHoverPaneProps(): HoverPaneProperties {
237
+ return this;
238
+ }
239
+
240
+ handleResize(entry: ResizeObserverEntry): void {
241
+ this.currentWidth = entry.contentRect.width;
242
+ this.currentHeight = entry.contentRect.height;
243
+ }
244
+
245
+ disconnectedCallback(): void {
246
+ this.stopResizeObservation(this.resizeObserver);
247
+ }
248
+
249
+ private stopResizeObservation(observer?: SharedResizeObserverInterface) {
250
+ observer?.removeObserver({
251
+ handler: this,
252
+ target: this.container,
253
+ });
254
+ }
255
+
256
+ private startResizeObservation() {
257
+ this.stopResizeObservation(this.resizeObserver);
258
+ this.resizeObserver?.addObserver({
259
+ handler: this,
260
+ target: this.container,
261
+ });
262
+ }
263
+
264
+ updated(props: PropertyValues) {
265
+ if (props.has('resizeObserver')) {
266
+ const previousObserver = props.get(
267
+ 'resizeObserver',
268
+ ) as SharedResizeObserverInterface;
269
+ this.stopResizeObservation(previousObserver);
270
+ this.startResizeObservation();
271
+ }
272
+ }
273
+
274
+ /**
275
+ * Handler for when the tile link is left-clicked. Emits the `resultSelected` event.
276
+ * In manage view, it also checks/unchecks the tile.
277
+ */
278
+ private handleLinkClicked(e: Event): void {
279
+ if (this.isManageView) {
280
+ e.preventDefault();
281
+ if (this.model) this.model.checked = !this.model.checked;
282
+ }
283
+
284
+ this.dispatchEvent(
285
+ new CustomEvent('resultSelected', { detail: this.model }),
286
+ );
287
+ }
288
+
289
+ /**
290
+ * Handler for when the tile link is right-clicked.
291
+ * In manage view, it opens the item in a new tab. Otherwise, does nothing.
292
+ */
293
+ private handleLinkContextMenu(e: Event): void {
294
+ if (this.isManageView && this.linkTileHref !== nothing) {
295
+ e.preventDefault();
296
+ window.open(this.linkTileHref, '_blank');
297
+ }
298
+ }
299
+
300
+ private tileInfoButtonPressed(
301
+ e: CustomEvent<{ x: number; y: number }>,
302
+ ): void {
303
+ this.hoverPaneController?.toggleHoverPane({
304
+ coords: e.detail,
305
+ enableTouchBackdrop: true,
306
+ });
307
+ }
308
+
309
+ private get tile() {
310
+ const {
311
+ model,
312
+ collectionPagePath,
313
+ baseNavigationUrl,
314
+ currentWidth,
315
+ currentHeight,
316
+ sortParam,
317
+ creatorFilter,
318
+ mobileBreakpoint,
319
+ defaultSortParam,
320
+ } = this;
321
+
322
+ if (!model) return nothing;
323
+
324
+ switch (this.tileDisplayMode) {
325
+ case 'grid':
326
+ switch (model.mediatype) {
327
+ case 'collection':
328
+ return html`<collection-tile
329
+ .model=${model}
330
+ .collectionPagePath=${collectionPagePath}
331
+ .baseImageUrl=${this.baseImageUrl}
332
+ .currentWidth=${currentWidth}
333
+ .currentHeight=${currentHeight}
334
+ .creatorFilter=${creatorFilter}
335
+ .suppressBlurring=${this.suppressBlurring}
336
+ .isManageView=${this.isManageView}
337
+ .layoutType=${this.layoutType}
338
+ ?showInfoButton=${this.shouldShowInfoButton}
339
+ @infoButtonPressed=${this.tileInfoButtonPressed}
340
+ >
341
+ </collection-tile>`;
342
+ case 'account':
343
+ return html`<account-tile
344
+ .model=${model}
345
+ .collectionPagePath=${collectionPagePath}
346
+ .baseImageUrl=${this.baseImageUrl}
347
+ .currentWidth=${currentWidth}
348
+ .currentHeight=${currentHeight}
349
+ .creatorFilter=${creatorFilter}
350
+ .suppressBlurring=${this.suppressBlurring}
351
+ .isManageView=${this.isManageView}
352
+ ?showInfoButton=${this.shouldShowInfoButton}
353
+ @infoButtonPressed=${this.tileInfoButtonPressed}
354
+ >
355
+ </account-tile>`;
356
+ case 'search':
357
+ return html`<search-tile
358
+ .model=${model}
359
+ .collectionPagePath=${collectionPagePath}
360
+ .baseImageUrl=${this.baseImageUrl}
361
+ .currentWidth=${currentWidth}
362
+ .currentHeight=${currentHeight}
363
+ .creatorFilter=${creatorFilter}
364
+ .suppressBlurring=${this.suppressBlurring}
365
+ .isManageView=${this.isManageView}
366
+ ?showInfoButton=${false}
367
+ @infoButtonPressed=${this.tileInfoButtonPressed}
368
+ >
369
+ </search-tile>`;
370
+ default:
371
+ return html`<item-tile
372
+ .model=${model}
373
+ .collectionPagePath=${collectionPagePath}
374
+ .currentWidth=${this.currentWidth}
375
+ .currentHeight=${this.currentHeight}
376
+ .baseImageUrl=${this.baseImageUrl}
377
+ .sortParam=${sortParam}
378
+ .defaultSortParam=${defaultSortParam}
379
+ .creatorFilter=${creatorFilter}
380
+ .loggedIn=${this.loggedIn}
381
+ .suppressBlurring=${this.suppressBlurring}
382
+ .isManageView=${this.isManageView}
383
+ .layoutType=${this.layoutType}
384
+ ?showTvClips=${this.showTvClips}
385
+ ?showInfoButton=${this.shouldShowInfoButton}
386
+ ?useLocalTime=${this.useLocalTime}
387
+ @infoButtonPressed=${this.tileInfoButtonPressed}
388
+ >
389
+ </item-tile>`;
390
+ }
391
+ case 'list-compact':
392
+ return html`<tile-list-compact
393
+ .model=${model}
394
+ .collectionPagePath=${collectionPagePath}
395
+ .currentWidth=${currentWidth}
396
+ .currentHeight=${currentHeight}
397
+ .baseNavigationUrl=${baseNavigationUrl}
398
+ .sortParam=${sortParam}
399
+ .defaultSortParam=${defaultSortParam}
400
+ .creatorFilter=${creatorFilter}
401
+ .mobileBreakpoint=${mobileBreakpoint}
402
+ .baseImageUrl=${this.baseImageUrl}
403
+ .loggedIn=${this.loggedIn}
404
+ .suppressBlurring=${this.suppressBlurring}
405
+ ?useLocalTime=${this.useLocalTime}
406
+ >
407
+ </tile-list-compact>`;
408
+ case 'list-detail':
409
+ return html`<tile-list
410
+ .model=${model}
411
+ .collectionPagePath=${collectionPagePath}
412
+ .collectionTitles=${this.collectionTitles}
413
+ .currentWidth=${currentWidth}
414
+ .currentHeight=${currentHeight}
415
+ .baseNavigationUrl=${baseNavigationUrl}
416
+ .sortParam=${sortParam}
417
+ .defaultSortParam=${defaultSortParam}
418
+ .creatorFilter=${creatorFilter}
419
+ .mobileBreakpoint=${mobileBreakpoint}
420
+ .baseImageUrl=${this.baseImageUrl}
421
+ .loggedIn=${this.loggedIn}
422
+ .suppressBlurring=${this.suppressBlurring}
423
+ ?useLocalTime=${this.useLocalTime}
424
+ >
425
+ </tile-list>`;
426
+ default:
427
+ return nothing;
428
+ }
429
+ }
430
+
431
+ static get styles() {
432
+ return [
433
+ srOnlyStyle,
434
+ css`
435
+ :host {
436
+ display: block;
437
+ height: 100%;
438
+ }
439
+
440
+ collection-tile {
441
+ --tileBorderColor: #555555;
442
+ --tileBackgroundColor: #666666;
443
+ --imageBlockBackgroundColor: #666666;
444
+ }
445
+
446
+ account-tile {
447
+ --tileBorderColor: #dddddd;
448
+ --imageBlockBackgroundColor: #fcf5e6;
449
+ }
450
+
451
+ item-tile {
452
+ --tileBorderColor: #dddddd;
453
+ --imageBlockBackgroundColor: #f1f1f4;
454
+ }
455
+
456
+ search-tile {
457
+ --tileBorderColor: #555555;
458
+ --tileBackgroundColor: #666666;
459
+ --imageBlockBackgroundColor: #666666;
460
+ --iconFillColor: #2c2c2c;
461
+ }
462
+
463
+ #container {
464
+ position: relative;
465
+ height: 100%;
466
+ border-radius: 4px;
467
+ }
468
+
469
+ #container.hoverable a:focus,
470
+ #container.hoverable a:hover {
471
+ box-shadow: var(
472
+ --tileHoverBoxShadow,
473
+ 0 0 6px 2px rgba(8, 8, 32, 0.8)
474
+ );
475
+ transition: box-shadow 0.1s ease;
476
+ }
477
+
478
+ a {
479
+ display: block;
480
+ height: 100%;
481
+ color: unset;
482
+ text-decoration: none;
483
+ transition: transform 0.05s ease;
484
+ border-radius: 4px;
485
+ outline: none;
486
+ }
487
+
488
+ a :first-child {
489
+ display: block;
490
+ height: 100%;
491
+ }
492
+
493
+ .manage-check {
494
+ position: absolute;
495
+ right: 0;
496
+ top: 0;
497
+ border: 5px solid #2c2c2c;
498
+ border-radius: 3px;
499
+ background-color: #2c2c2c;
500
+ z-index: 1;
501
+ }
502
+
503
+ .manage-check > input[type='checkbox'] {
504
+ display: block;
505
+ margin: 0;
506
+ }
507
+
508
+ #touch-backdrop {
509
+ position: fixed;
510
+ width: 100vw;
511
+ height: 100vh;
512
+ top: 0;
513
+ left: 0;
514
+ z-index: 2;
515
+ background: transparent;
516
+ }
517
+
518
+ tile-hover-pane {
519
+ position: absolute;
520
+ top: 0;
521
+ left: -9999px;
522
+ z-index: 2;
523
+ }
524
+ `,
525
+ ];
526
+ }
527
+ }