@internetarchive/collection-browser 3.3.2 → 3.3.4-alpha-webdev7761.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 (543) 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/index.d.ts +16 -0
  11. package/dist/index.js.map +1 -0
  12. package/dist/src/app-root.d.ts +105 -0
  13. package/dist/src/app-root.js +1076 -0
  14. package/dist/src/app-root.js.map +1 -0
  15. package/dist/src/assets/img/icons/arrow-left.d.ts +2 -0
  16. package/dist/src/assets/img/icons/arrow-left.js +10 -0
  17. package/dist/src/assets/img/icons/arrow-left.js.map +1 -0
  18. package/dist/src/assets/img/icons/arrow-right.d.ts +2 -0
  19. package/dist/src/assets/img/icons/arrow-right.js +10 -0
  20. package/dist/src/assets/img/icons/arrow-right.js.map +1 -0
  21. package/dist/src/assets/img/icons/chevron.d.ts +2 -0
  22. package/dist/src/assets/img/icons/chevron.js +4 -0
  23. package/dist/src/assets/img/icons/chevron.js.map +1 -0
  24. package/dist/src/assets/img/icons/close-circle-dark.d.ts +2 -0
  25. package/dist/src/assets/img/icons/close-circle-dark.js +5 -0
  26. package/dist/src/assets/img/icons/close-circle-dark.js.map +1 -0
  27. package/dist/src/assets/img/icons/contract.d.ts +2 -0
  28. package/dist/src/assets/img/icons/contract.js +9 -0
  29. package/dist/src/assets/img/icons/contract.js.map +1 -0
  30. package/dist/src/assets/img/icons/empty-query.d.ts +2 -0
  31. package/dist/src/assets/img/icons/empty-query.js +5 -0
  32. package/dist/src/assets/img/icons/empty-query.js.map +1 -0
  33. package/dist/src/assets/img/icons/expand.d.ts +2 -0
  34. package/dist/src/assets/img/icons/expand.js +9 -0
  35. package/dist/src/assets/img/icons/expand.js.map +1 -0
  36. package/dist/src/assets/img/icons/eye-closed.d.ts +2 -0
  37. package/dist/src/assets/img/icons/eye-closed.js +5 -0
  38. package/dist/src/assets/img/icons/eye-closed.js.map +1 -0
  39. package/dist/src/assets/img/icons/eye.d.ts +2 -0
  40. package/dist/src/assets/img/icons/eye.js +5 -0
  41. package/dist/src/assets/img/icons/eye.js.map +1 -0
  42. package/dist/src/assets/img/icons/favorite-filled.d.ts +1 -0
  43. package/dist/src/assets/img/icons/favorite-filled.js +10 -0
  44. package/dist/src/assets/img/icons/favorite-filled.js.map +1 -0
  45. package/dist/src/assets/img/icons/favorite-unfilled.d.ts +1 -0
  46. package/dist/src/assets/img/icons/favorite-unfilled.js +9 -0
  47. package/dist/src/assets/img/icons/favorite-unfilled.js.map +1 -0
  48. package/dist/src/assets/img/icons/filter.d.ts +2 -0
  49. package/dist/src/assets/img/icons/filter.js +10 -0
  50. package/dist/src/assets/img/icons/filter.js.map +1 -0
  51. package/dist/src/assets/img/icons/login-required.d.ts +1 -0
  52. package/dist/src/assets/img/icons/login-required.js +18 -0
  53. package/dist/src/assets/img/icons/login-required.js.map +1 -0
  54. package/dist/src/assets/img/icons/mediatype/account.d.ts +1 -0
  55. package/dist/src/assets/img/icons/mediatype/account.js +14 -0
  56. package/dist/src/assets/img/icons/mediatype/account.js.map +1 -0
  57. package/dist/src/assets/img/icons/mediatype/audio.d.ts +1 -0
  58. package/dist/src/assets/img/icons/mediatype/audio.js +14 -0
  59. package/dist/src/assets/img/icons/mediatype/audio.js.map +1 -0
  60. package/dist/src/assets/img/icons/mediatype/collection.d.ts +1 -0
  61. package/dist/src/assets/img/icons/mediatype/collection.js +12 -0
  62. package/dist/src/assets/img/icons/mediatype/collection.js.map +1 -0
  63. package/dist/src/assets/img/icons/mediatype/data.d.ts +1 -0
  64. package/dist/src/assets/img/icons/mediatype/data.js +15 -0
  65. package/dist/src/assets/img/icons/mediatype/data.js.map +1 -0
  66. package/dist/src/assets/img/icons/mediatype/etree.d.ts +1 -0
  67. package/dist/src/assets/img/icons/mediatype/etree.js +14 -0
  68. package/dist/src/assets/img/icons/mediatype/etree.js.map +1 -0
  69. package/dist/src/assets/img/icons/mediatype/film.d.ts +1 -0
  70. package/dist/src/assets/img/icons/mediatype/film.js +14 -0
  71. package/dist/src/assets/img/icons/mediatype/film.js.map +1 -0
  72. package/dist/src/assets/img/icons/mediatype/images.d.ts +1 -0
  73. package/dist/src/assets/img/icons/mediatype/images.js +13 -0
  74. package/dist/src/assets/img/icons/mediatype/images.js.map +1 -0
  75. package/dist/src/assets/img/icons/mediatype/radio.d.ts +1 -0
  76. package/dist/src/assets/img/icons/mediatype/radio.js +15 -0
  77. package/dist/src/assets/img/icons/mediatype/radio.js.map +1 -0
  78. package/dist/src/assets/img/icons/mediatype/search.d.ts +1 -0
  79. package/dist/src/assets/img/icons/mediatype/search.js +14 -0
  80. package/dist/src/assets/img/icons/mediatype/search.js.map +1 -0
  81. package/dist/src/assets/img/icons/mediatype/software.d.ts +1 -0
  82. package/dist/src/assets/img/icons/mediatype/software.js +13 -0
  83. package/dist/src/assets/img/icons/mediatype/software.js.map +1 -0
  84. package/dist/src/assets/img/icons/mediatype/texts.d.ts +1 -0
  85. package/dist/src/assets/img/icons/mediatype/texts.js +13 -0
  86. package/dist/src/assets/img/icons/mediatype/texts.js.map +1 -0
  87. package/dist/src/assets/img/icons/mediatype/tv-commercial.d.ts +1 -0
  88. package/dist/src/assets/img/icons/mediatype/tv-commercial.js +12 -0
  89. package/dist/src/assets/img/icons/mediatype/tv-commercial.js.map +1 -0
  90. package/dist/src/assets/img/icons/mediatype/tv-fact-check.d.ts +1 -0
  91. package/dist/src/assets/img/icons/mediatype/tv-fact-check.js +12 -0
  92. package/dist/src/assets/img/icons/mediatype/tv-fact-check.js.map +1 -0
  93. package/dist/src/assets/img/icons/mediatype/tv-quote.d.ts +1 -0
  94. package/dist/src/assets/img/icons/mediatype/tv-quote.js +12 -0
  95. package/dist/src/assets/img/icons/mediatype/tv-quote.js.map +1 -0
  96. package/dist/src/assets/img/icons/mediatype/tv.d.ts +1 -0
  97. package/dist/src/assets/img/icons/mediatype/tv.js +14 -0
  98. package/dist/src/assets/img/icons/mediatype/tv.js.map +1 -0
  99. package/dist/src/assets/img/icons/mediatype/video.d.ts +1 -0
  100. package/dist/src/assets/img/icons/mediatype/video.js +14 -0
  101. package/dist/src/assets/img/icons/mediatype/video.js.map +1 -0
  102. package/dist/src/assets/img/icons/mediatype/web.d.ts +1 -0
  103. package/dist/src/assets/img/icons/mediatype/web.js +13 -0
  104. package/dist/src/assets/img/icons/mediatype/web.js.map +1 -0
  105. package/dist/src/assets/img/icons/null-result.d.ts +2 -0
  106. package/dist/src/assets/img/icons/null-result.js +5 -0
  107. package/dist/src/assets/img/icons/null-result.js.map +1 -0
  108. package/dist/src/assets/img/icons/quote.d.ts +1 -0
  109. package/dist/src/assets/img/icons/quote.js +7 -0
  110. package/dist/src/assets/img/icons/quote.js.map +1 -0
  111. package/dist/src/assets/img/icons/restricted.d.ts +1 -0
  112. package/dist/src/assets/img/icons/restricted.js +13 -0
  113. package/dist/src/assets/img/icons/restricted.js.map +1 -0
  114. package/dist/src/assets/img/icons/reviews.d.ts +1 -0
  115. package/dist/src/assets/img/icons/reviews.js +11 -0
  116. package/dist/src/assets/img/icons/reviews.js.map +1 -0
  117. package/dist/src/assets/img/icons/upload.d.ts +1 -0
  118. package/dist/src/assets/img/icons/upload.js +12 -0
  119. package/dist/src/assets/img/icons/upload.js.map +1 -0
  120. package/dist/src/assets/img/icons/views.d.ts +1 -0
  121. package/dist/src/assets/img/icons/views.js +11 -0
  122. package/dist/src/assets/img/icons/views.js.map +1 -0
  123. package/dist/src/circular-activity-indicator.d.ts +5 -0
  124. package/dist/src/circular-activity-indicator.js +66 -0
  125. package/dist/src/circular-activity-indicator.js.map +1 -0
  126. package/dist/src/collection-browser.d.ts +692 -0
  127. package/dist/src/collection-browser.js +2669 -0
  128. package/dist/src/collection-browser.js.map +1 -0
  129. package/dist/src/collection-facets/facet-row.d.ts +30 -0
  130. package/dist/src/collection-facets/facet-row.js +266 -0
  131. package/dist/src/collection-facets/facet-row.js.map +1 -0
  132. package/dist/src/collection-facets/facet-tombstone-row.d.ts +5 -0
  133. package/dist/src/collection-facets/facet-tombstone-row.js +43 -0
  134. package/dist/src/collection-facets/facet-tombstone-row.js.map +1 -0
  135. package/dist/src/collection-facets/facets-template.d.ts +13 -0
  136. package/dist/src/collection-facets/facets-template.js +68 -0
  137. package/dist/src/collection-facets/facets-template.js.map +1 -0
  138. package/dist/src/collection-facets/models.d.ts +9 -0
  139. package/dist/src/collection-facets/models.js +10 -0
  140. package/dist/src/collection-facets/models.js.map +1 -0
  141. package/dist/src/collection-facets/more-facets-content.d.ts +109 -0
  142. package/dist/src/collection-facets/more-facets-content.js +547 -0
  143. package/dist/src/collection-facets/more-facets-content.js.map +1 -0
  144. package/dist/src/collection-facets/more-facets-pagination.d.ts +36 -0
  145. package/dist/src/collection-facets/more-facets-pagination.js +264 -0
  146. package/dist/src/collection-facets/more-facets-pagination.js.map +1 -0
  147. package/dist/src/collection-facets/smart-facets/dedupe.d.ts +10 -0
  148. package/dist/src/collection-facets/smart-facets/dedupe.js +35 -0
  149. package/dist/src/collection-facets/smart-facets/dedupe.js.map +1 -0
  150. package/dist/src/collection-facets/smart-facets/heuristics/browser-language/browser-language-heuristic.d.ts +5 -0
  151. package/dist/src/collection-facets/smart-facets/heuristics/browser-language/browser-language-heuristic.js +24 -0
  152. package/dist/src/collection-facets/smart-facets/heuristics/browser-language/browser-language-heuristic.js.map +1 -0
  153. package/dist/src/collection-facets/smart-facets/heuristics/index.d.ts +3 -0
  154. package/dist/src/collection-facets/smart-facets/heuristics/index.js +4 -0
  155. package/dist/src/collection-facets/smart-facets/heuristics/index.js.map +1 -0
  156. package/dist/src/collection-facets/smart-facets/heuristics/query-keywords/query-keywords-heuristic.d.ts +4 -0
  157. package/dist/src/collection-facets/smart-facets/heuristics/query-keywords/query-keywords-heuristic.js +14 -0
  158. package/dist/src/collection-facets/smart-facets/heuristics/query-keywords/query-keywords-heuristic.js.map +1 -0
  159. package/dist/src/collection-facets/smart-facets/heuristics/query-keywords/query-keywords-map.d.ts +6 -0
  160. package/dist/src/collection-facets/smart-facets/heuristics/query-keywords/query-keywords-map.js +68 -0
  161. package/dist/src/collection-facets/smart-facets/heuristics/query-keywords/query-keywords-map.js.map +1 -0
  162. package/dist/src/collection-facets/smart-facets/heuristics/wikidata/wikidata-entity-map.d.ts +9 -0
  163. package/dist/src/collection-facets/smart-facets/heuristics/wikidata/wikidata-entity-map.js +69 -0
  164. package/dist/src/collection-facets/smart-facets/heuristics/wikidata/wikidata-entity-map.js.map +1 -0
  165. package/dist/src/collection-facets/smart-facets/heuristics/wikidata/wikidata-heuristic.d.ts +21 -0
  166. package/dist/src/collection-facets/smart-facets/heuristics/wikidata/wikidata-heuristic.js +76 -0
  167. package/dist/src/collection-facets/smart-facets/heuristics/wikidata/wikidata-heuristic.js.map +1 -0
  168. package/dist/src/collection-facets/smart-facets/models.d.ts +30 -0
  169. package/dist/src/collection-facets/smart-facets/models.js +2 -0
  170. package/dist/src/collection-facets/smart-facets/models.js.map +1 -0
  171. package/dist/src/collection-facets/smart-facets/smart-facet-bar.d.ts +54 -0
  172. package/dist/src/collection-facets/smart-facets/smart-facet-bar.js +383 -0
  173. package/dist/src/collection-facets/smart-facets/smart-facet-bar.js.map +1 -0
  174. package/dist/src/collection-facets/smart-facets/smart-facet-button.d.ts +11 -0
  175. package/dist/src/collection-facets/smart-facets/smart-facet-button.js +133 -0
  176. package/dist/src/collection-facets/smart-facets/smart-facet-button.js.map +1 -0
  177. package/dist/src/collection-facets/smart-facets/smart-facet-dropdown.d.ts +28 -0
  178. package/dist/src/collection-facets/smart-facets/smart-facet-dropdown.js +172 -0
  179. package/dist/src/collection-facets/smart-facets/smart-facet-dropdown.js.map +1 -0
  180. package/dist/src/collection-facets/smart-facets/smart-facet-equals.d.ts +2 -0
  181. package/dist/src/collection-facets/smart-facets/smart-facet-equals.js +13 -0
  182. package/dist/src/collection-facets/smart-facets/smart-facet-equals.js.map +1 -0
  183. package/dist/src/collection-facets/smart-facets/smart-facet-heuristics.d.ts +5 -0
  184. package/dist/src/collection-facets/smart-facets/smart-facet-heuristics.js +18 -0
  185. package/dist/src/collection-facets/smart-facets/smart-facet-heuristics.js.map +1 -0
  186. package/dist/src/collection-facets/toggle-switch.d.ts +41 -0
  187. package/dist/src/collection-facets/toggle-switch.js +174 -0
  188. package/dist/src/collection-facets/toggle-switch.js.map +1 -0
  189. package/dist/src/collection-facets.d.ts +113 -0
  190. package/dist/src/collection-facets.js +884 -0
  191. package/dist/src/collection-facets.js.map +1 -0
  192. package/dist/src/data-source/collection-browser-data-source-interface.d.ts +270 -0
  193. package/dist/src/data-source/collection-browser-data-source-interface.js +2 -0
  194. package/dist/src/data-source/collection-browser-data-source-interface.js.map +1 -0
  195. package/dist/src/data-source/collection-browser-data-source.d.ts +418 -0
  196. package/dist/src/data-source/collection-browser-data-source.js +1121 -0
  197. package/dist/src/data-source/collection-browser-data-source.js.map +1 -0
  198. package/dist/src/data-source/collection-browser-query-state.d.ts +48 -0
  199. package/dist/src/data-source/collection-browser-query-state.js +2 -0
  200. package/dist/src/data-source/collection-browser-query-state.js.map +1 -0
  201. package/dist/src/data-source/models.d.ts +43 -0
  202. package/dist/src/data-source/models.js +9 -0
  203. package/dist/src/data-source/models.js.map +1 -0
  204. package/dist/src/empty-placeholder.d.ts +23 -0
  205. package/dist/src/empty-placeholder.js +165 -0
  206. package/dist/src/empty-placeholder.js.map +1 -0
  207. package/dist/src/expanded-date-picker.d.ts +50 -0
  208. package/dist/src/expanded-date-picker.js +182 -0
  209. package/dist/src/expanded-date-picker.js.map +1 -0
  210. package/dist/src/language-code-handler/language-code-handler.d.ts +37 -0
  211. package/dist/src/language-code-handler/language-code-handler.js +27 -0
  212. package/dist/src/language-code-handler/language-code-handler.js.map +1 -0
  213. package/dist/src/language-code-handler/language-code-mapping.d.ts +1 -0
  214. package/dist/src/language-code-handler/language-code-mapping.js +563 -0
  215. package/dist/src/language-code-handler/language-code-mapping.js.map +1 -0
  216. package/dist/src/manage/manage-bar.d.ts +58 -0
  217. package/dist/src/manage/manage-bar.js +237 -0
  218. package/dist/src/manage/manage-bar.js.map +1 -0
  219. package/dist/src/manage/remove-items-modal-content.d.ts +9 -0
  220. package/dist/src/manage/remove-items-modal-content.js +104 -0
  221. package/dist/src/manage/remove-items-modal-content.js.map +1 -0
  222. package/dist/src/mediatype/mediatype-config.d.ts +11 -0
  223. package/dist/src/mediatype/mediatype-config.js +116 -0
  224. package/dist/src/mediatype/mediatype-config.js.map +1 -0
  225. package/dist/src/models.d.ts +298 -0
  226. package/dist/src/models.js +507 -0
  227. package/dist/src/models.js.map +1 -0
  228. package/dist/src/restoration-state-handler.d.ts +74 -0
  229. package/dist/src/restoration-state-handler.js +397 -0
  230. package/dist/src/restoration-state-handler.js.map +1 -0
  231. package/dist/src/sort-filter-bar/alpha-bar-tooltip.d.ts +6 -0
  232. package/dist/src/sort-filter-bar/alpha-bar-tooltip.js +60 -0
  233. package/dist/src/sort-filter-bar/alpha-bar-tooltip.js.map +1 -0
  234. package/dist/src/sort-filter-bar/alpha-bar.d.ts +21 -0
  235. package/dist/src/sort-filter-bar/alpha-bar.js +241 -0
  236. package/dist/src/sort-filter-bar/alpha-bar.js.map +1 -0
  237. package/dist/src/sort-filter-bar/img/compact.d.ts +1 -0
  238. package/dist/src/sort-filter-bar/img/compact.js +5 -0
  239. package/dist/src/sort-filter-bar/img/compact.js.map +1 -0
  240. package/dist/src/sort-filter-bar/img/list.d.ts +1 -0
  241. package/dist/src/sort-filter-bar/img/list.js +5 -0
  242. package/dist/src/sort-filter-bar/img/list.js.map +1 -0
  243. package/dist/src/sort-filter-bar/img/sort-toggle-disabled.d.ts +1 -0
  244. package/dist/src/sort-filter-bar/img/sort-toggle-disabled.js +15 -0
  245. package/dist/src/sort-filter-bar/img/sort-toggle-disabled.js.map +1 -0
  246. package/dist/src/sort-filter-bar/img/sort-toggle-down.d.ts +1 -0
  247. package/dist/src/sort-filter-bar/img/sort-toggle-down.js +17 -0
  248. package/dist/src/sort-filter-bar/img/sort-toggle-down.js.map +1 -0
  249. package/dist/src/sort-filter-bar/img/sort-toggle-up.d.ts +1 -0
  250. package/dist/src/sort-filter-bar/img/sort-toggle-up.js +17 -0
  251. package/dist/src/sort-filter-bar/img/sort-toggle-up.js.map +1 -0
  252. package/dist/src/sort-filter-bar/img/sort-triangle.d.ts +1 -0
  253. package/dist/src/sort-filter-bar/img/sort-triangle.js +5 -0
  254. package/dist/src/sort-filter-bar/img/sort-triangle.js.map +1 -0
  255. package/dist/src/sort-filter-bar/img/tile.d.ts +1 -0
  256. package/dist/src/sort-filter-bar/img/tile.js +5 -0
  257. package/dist/src/sort-filter-bar/img/tile.js.map +1 -0
  258. package/dist/src/sort-filter-bar/sort-filter-bar.d.ts +278 -0
  259. package/dist/src/sort-filter-bar/sort-filter-bar.js +1161 -0
  260. package/dist/src/sort-filter-bar/sort-filter-bar.js.map +1 -0
  261. package/dist/src/styles/ia-button.d.ts +2 -0
  262. package/dist/src/styles/ia-button.js +134 -0
  263. package/dist/src/styles/ia-button.js.map +1 -0
  264. package/dist/src/styles/item-image-styles.d.ts +8 -0
  265. package/dist/src/styles/item-image-styles.js +123 -0
  266. package/dist/src/styles/item-image-styles.js.map +1 -0
  267. package/dist/src/styles/sr-only.d.ts +1 -0
  268. package/dist/src/styles/sr-only.js +18 -0
  269. package/dist/src/styles/sr-only.js.map +1 -0
  270. package/dist/src/tiles/base-tile-component.d.ts +27 -0
  271. package/dist/src/tiles/base-tile-component.js +82 -0
  272. package/dist/src/tiles/base-tile-component.js.map +1 -0
  273. package/dist/src/tiles/collection-browser-loading-tile.d.ts +5 -0
  274. package/dist/src/tiles/collection-browser-loading-tile.js +29 -0
  275. package/dist/src/tiles/collection-browser-loading-tile.js.map +1 -0
  276. package/dist/src/tiles/grid/account-tile.d.ts +18 -0
  277. package/dist/src/tiles/grid/account-tile.js +111 -0
  278. package/dist/src/tiles/grid/account-tile.js.map +1 -0
  279. package/dist/src/tiles/grid/collection-tile.d.ts +15 -0
  280. package/dist/src/tiles/grid/collection-tile.js +160 -0
  281. package/dist/src/tiles/grid/collection-tile.js.map +1 -0
  282. package/dist/src/tiles/grid/item-tile.d.ts +41 -0
  283. package/dist/src/tiles/grid/item-tile.js +321 -0
  284. package/dist/src/tiles/grid/item-tile.js.map +1 -0
  285. package/dist/src/tiles/grid/search-tile.d.ts +10 -0
  286. package/dist/src/tiles/grid/search-tile.js +95 -0
  287. package/dist/src/tiles/grid/search-tile.js.map +1 -0
  288. package/dist/src/tiles/grid/styles/tile-grid-shared-styles.d.ts +1 -0
  289. package/dist/src/tiles/grid/styles/tile-grid-shared-styles.js +128 -0
  290. package/dist/src/tiles/grid/styles/tile-grid-shared-styles.js.map +1 -0
  291. package/dist/src/tiles/grid/tile-stats.d.ts +58 -0
  292. package/dist/src/tiles/grid/tile-stats.js +204 -0
  293. package/dist/src/tiles/grid/tile-stats.js.map +1 -0
  294. package/dist/src/tiles/hover/hover-pane-controller.d.ts +228 -0
  295. package/dist/src/tiles/hover/hover-pane-controller.js +446 -0
  296. package/dist/src/tiles/hover/hover-pane-controller.js.map +1 -0
  297. package/dist/src/tiles/hover/tile-hover-pane.d.ts +20 -0
  298. package/dist/src/tiles/hover/tile-hover-pane.js +185 -0
  299. package/dist/src/tiles/hover/tile-hover-pane.js.map +1 -0
  300. package/dist/src/tiles/image-block.d.ts +19 -0
  301. package/dist/src/tiles/image-block.js +175 -0
  302. package/dist/src/tiles/image-block.js.map +1 -0
  303. package/dist/src/tiles/item-image.d.ts +40 -0
  304. package/dist/src/tiles/item-image.js +191 -0
  305. package/dist/src/tiles/item-image.js.map +1 -0
  306. package/dist/src/tiles/list/tile-list-compact-header.d.ts +6 -0
  307. package/dist/src/tiles/list/tile-list-compact-header.js +85 -0
  308. package/dist/src/tiles/list/tile-list-compact-header.js.map +1 -0
  309. package/dist/src/tiles/list/tile-list-compact.d.ts +19 -0
  310. package/dist/src/tiles/list/tile-list-compact.js +223 -0
  311. package/dist/src/tiles/list/tile-list-compact.js.map +1 -0
  312. package/dist/src/tiles/list/tile-list.d.ts +54 -0
  313. package/dist/src/tiles/list/tile-list.js +621 -0
  314. package/dist/src/tiles/list/tile-list.js.map +1 -0
  315. package/dist/src/tiles/models.d.ts +1 -0
  316. package/dist/src/tiles/models.js +2 -0
  317. package/dist/src/tiles/models.js.map +1 -0
  318. package/dist/src/tiles/overlay/icon-overlay.d.ts +8 -0
  319. package/dist/src/tiles/overlay/icon-overlay.js +55 -0
  320. package/dist/src/tiles/overlay/icon-overlay.js.map +1 -0
  321. package/dist/src/tiles/overlay/text-overlay.d.ts +9 -0
  322. package/dist/src/tiles/overlay/text-overlay.js +74 -0
  323. package/dist/src/tiles/overlay/text-overlay.js.map +1 -0
  324. package/dist/src/tiles/review-block.d.ts +12 -0
  325. package/dist/src/tiles/review-block.js +134 -0
  326. package/dist/src/tiles/review-block.js.map +1 -0
  327. package/dist/src/tiles/text-snippet-block.d.ts +27 -0
  328. package/dist/src/tiles/text-snippet-block.js +132 -0
  329. package/dist/src/tiles/text-snippet-block.js.map +1 -0
  330. package/dist/src/tiles/tile-dispatcher.d.ts +71 -0
  331. package/dist/src/tiles/tile-dispatcher.js +472 -0
  332. package/dist/src/tiles/tile-dispatcher.js.map +1 -0
  333. package/dist/src/tiles/tile-display-value-provider.d.ts +47 -0
  334. package/dist/src/tiles/tile-display-value-provider.js +95 -0
  335. package/dist/src/tiles/tile-display-value-provider.js.map +1 -0
  336. package/dist/src/tiles/tile-mediatype-icon.d.ts +27 -0
  337. package/dist/src/tiles/tile-mediatype-icon.js +130 -0
  338. package/dist/src/tiles/tile-mediatype-icon.js.map +1 -0
  339. package/dist/src/utils/analytics-events.d.ts +28 -0
  340. package/dist/src/utils/analytics-events.js +31 -0
  341. package/dist/src/utils/analytics-events.js.map +1 -0
  342. package/dist/src/utils/array-equals.d.ts +4 -0
  343. package/dist/src/utils/array-equals.js +11 -0
  344. package/dist/src/utils/array-equals.js.map +1 -0
  345. package/dist/src/utils/collapse-repeated-quotes.d.ts +11 -0
  346. package/dist/src/utils/collapse-repeated-quotes.js +14 -0
  347. package/dist/src/utils/collapse-repeated-quotes.js.map +1 -0
  348. package/dist/src/utils/facet-utils.d.ts +83 -0
  349. package/dist/src/utils/facet-utils.js +152 -0
  350. package/dist/src/utils/facet-utils.js.map +1 -0
  351. package/dist/src/utils/format-count.d.ts +7 -0
  352. package/dist/src/utils/format-count.js +76 -0
  353. package/dist/src/utils/format-count.js.map +1 -0
  354. package/dist/src/utils/format-date.d.ts +16 -0
  355. package/dist/src/utils/format-date.js +33 -0
  356. package/dist/src/utils/format-date.js.map +1 -0
  357. package/dist/src/utils/format-unit-size.d.ts +2 -0
  358. package/dist/src/utils/format-unit-size.js +34 -0
  359. package/dist/src/utils/format-unit-size.js.map +1 -0
  360. package/dist/src/utils/local-date-from-utc.d.ts +9 -0
  361. package/dist/src/utils/local-date-from-utc.js +16 -0
  362. package/dist/src/utils/local-date-from-utc.js.map +1 -0
  363. package/dist/src/utils/log.d.ts +7 -0
  364. package/dist/src/utils/log.js +14 -0
  365. package/dist/src/utils/log.js.map +1 -0
  366. package/dist/src/utils/resolve-mediatype.d.ts +8 -0
  367. package/dist/src/utils/resolve-mediatype.js +24 -0
  368. package/dist/src/utils/resolve-mediatype.js.map +1 -0
  369. package/dist/src/utils/sha1.d.ts +2 -0
  370. package/dist/src/utils/sha1.js +9 -0
  371. package/dist/src/utils/sha1.js.map +1 -0
  372. package/dist/test/collection-browser.test.d.ts +1 -0
  373. package/dist/test/collection-browser.test.js +1721 -0
  374. package/dist/test/collection-browser.test.js.map +1 -0
  375. package/dist/test/collection-facets/facet-row.test.d.ts +1 -0
  376. package/dist/test/collection-facets/facet-row.test.js +274 -0
  377. package/dist/test/collection-facets/facet-row.test.js.map +1 -0
  378. package/dist/test/collection-facets/facets-template.test.d.ts +1 -0
  379. package/dist/test/collection-facets/facets-template.test.js +101 -0
  380. package/dist/test/collection-facets/facets-template.test.js.map +1 -0
  381. package/dist/test/collection-facets/more-facets-content.test.d.ts +1 -0
  382. package/dist/test/collection-facets/more-facets-content.test.js +169 -0
  383. package/dist/test/collection-facets/more-facets-content.test.js.map +1 -0
  384. package/dist/test/collection-facets/more-facets-pagination.test.d.ts +1 -0
  385. package/dist/test/collection-facets/more-facets-pagination.test.js +132 -0
  386. package/dist/test/collection-facets/more-facets-pagination.test.js.map +1 -0
  387. package/dist/test/collection-facets/toggle-switch.test.d.ts +1 -0
  388. package/dist/test/collection-facets/toggle-switch.test.js +96 -0
  389. package/dist/test/collection-facets/toggle-switch.test.js.map +1 -0
  390. package/dist/test/collection-facets.test.d.ts +2 -0
  391. package/dist/test/collection-facets.test.js +745 -0
  392. package/dist/test/collection-facets.test.js.map +1 -0
  393. package/dist/test/data-source/collection-browser-data-source.test.d.ts +1 -0
  394. package/dist/test/data-source/collection-browser-data-source.test.js +102 -0
  395. package/dist/test/data-source/collection-browser-data-source.test.js.map +1 -0
  396. package/dist/test/empty-placeholder.test.d.ts +1 -0
  397. package/dist/test/empty-placeholder.test.js +63 -0
  398. package/dist/test/empty-placeholder.test.js.map +1 -0
  399. package/dist/test/expanded-date-picker.test.d.ts +1 -0
  400. package/dist/test/expanded-date-picker.test.js +130 -0
  401. package/dist/test/expanded-date-picker.test.js.map +1 -0
  402. package/dist/test/icon-overlay.test.d.ts +1 -0
  403. package/dist/test/icon-overlay.test.js +30 -0
  404. package/dist/test/icon-overlay.test.js.map +1 -0
  405. package/dist/test/image-block.test.d.ts +1 -0
  406. package/dist/test/image-block.test.js +218 -0
  407. package/dist/test/image-block.test.js.map +1 -0
  408. package/dist/test/item-image.test.d.ts +1 -0
  409. package/dist/test/item-image.test.js +196 -0
  410. package/dist/test/item-image.test.js.map +1 -0
  411. package/dist/test/manage/manage-bar.test.d.ts +2 -0
  412. package/dist/test/manage/manage-bar.test.js +106 -0
  413. package/dist/test/manage/manage-bar.test.js.map +1 -0
  414. package/dist/test/manage/remove-items-modal-content.test.d.ts +1 -0
  415. package/dist/test/manage/remove-items-modal-content.test.js +65 -0
  416. package/dist/test/manage/remove-items-modal-content.test.js.map +1 -0
  417. package/dist/test/mediatype-config.test.d.ts +1 -0
  418. package/dist/test/mediatype-config.test.js +11 -0
  419. package/dist/test/mediatype-config.test.js.map +1 -0
  420. package/dist/test/mocks/mock-analytics-handler.d.ts +10 -0
  421. package/dist/test/mocks/mock-analytics-handler.js +16 -0
  422. package/dist/test/mocks/mock-analytics-handler.js.map +1 -0
  423. package/dist/test/mocks/mock-search-responses.d.ts +31 -0
  424. package/dist/test/mocks/mock-search-responses.js +1241 -0
  425. package/dist/test/mocks/mock-search-responses.js.map +1 -0
  426. package/dist/test/mocks/mock-search-service.d.ts +16 -0
  427. package/dist/test/mocks/mock-search-service.js +65 -0
  428. package/dist/test/mocks/mock-search-service.js.map +1 -0
  429. package/dist/test/restoration-state-handler.test.d.ts +1 -0
  430. package/dist/test/restoration-state-handler.test.js +343 -0
  431. package/dist/test/restoration-state-handler.test.js.map +1 -0
  432. package/dist/test/review-block.test.d.ts +1 -0
  433. package/dist/test/review-block.test.js +48 -0
  434. package/dist/test/review-block.test.js.map +1 -0
  435. package/dist/test/sort-filter-bar/alpha-bar-tooltip.test.d.ts +1 -0
  436. package/dist/test/sort-filter-bar/alpha-bar-tooltip.test.js +13 -0
  437. package/dist/test/sort-filter-bar/alpha-bar-tooltip.test.js.map +1 -0
  438. package/dist/test/sort-filter-bar/alpha-bar.test.d.ts +1 -0
  439. package/dist/test/sort-filter-bar/alpha-bar.test.js +74 -0
  440. package/dist/test/sort-filter-bar/alpha-bar.test.js.map +1 -0
  441. package/dist/test/sort-filter-bar/sort-filter-bar.test.d.ts +1 -0
  442. package/dist/test/sort-filter-bar/sort-filter-bar.test.js +609 -0
  443. package/dist/test/sort-filter-bar/sort-filter-bar.test.js.map +1 -0
  444. package/dist/test/text-overlay.test.d.ts +1 -0
  445. package/dist/test/text-overlay.test.js +42 -0
  446. package/dist/test/text-overlay.test.js.map +1 -0
  447. package/dist/test/text-snippet-block.test.d.ts +1 -0
  448. package/dist/test/text-snippet-block.test.js +63 -0
  449. package/dist/test/text-snippet-block.test.js.map +1 -0
  450. package/dist/test/tile-stats.test.d.ts +1 -0
  451. package/dist/test/tile-stats.test.js +128 -0
  452. package/dist/test/tile-stats.test.js.map +1 -0
  453. package/dist/test/tiles/grid/account-tile.test.d.ts +1 -0
  454. package/dist/test/tiles/grid/account-tile.test.js +96 -0
  455. package/dist/test/tiles/grid/account-tile.test.js.map +1 -0
  456. package/dist/test/tiles/grid/collection-tile.test.d.ts +1 -0
  457. package/dist/test/tiles/grid/collection-tile.test.js +96 -0
  458. package/dist/test/tiles/grid/collection-tile.test.js.map +1 -0
  459. package/dist/test/tiles/grid/item-tile.test.d.ts +1 -0
  460. package/dist/test/tiles/grid/item-tile.test.js +427 -0
  461. package/dist/test/tiles/grid/item-tile.test.js.map +1 -0
  462. package/dist/test/tiles/grid/search-tile.test.d.ts +1 -0
  463. package/dist/test/tiles/grid/search-tile.test.js +66 -0
  464. package/dist/test/tiles/grid/search-tile.test.js.map +1 -0
  465. package/dist/test/tiles/hover/hover-pane-controller.test.d.ts +1 -0
  466. package/dist/test/tiles/hover/hover-pane-controller.test.js +329 -0
  467. package/dist/test/tiles/hover/hover-pane-controller.test.js.map +1 -0
  468. package/dist/test/tiles/hover/tile-hover-pane.test.d.ts +1 -0
  469. package/dist/test/tiles/hover/tile-hover-pane.test.js +57 -0
  470. package/dist/test/tiles/hover/tile-hover-pane.test.js.map +1 -0
  471. package/dist/test/tiles/list/tile-list-compact.test.d.ts +1 -0
  472. package/dist/test/tiles/list/tile-list-compact.test.js +251 -0
  473. package/dist/test/tiles/list/tile-list-compact.test.js.map +1 -0
  474. package/dist/test/tiles/list/tile-list.test.d.ts +1 -0
  475. package/dist/test/tiles/list/tile-list.test.js +469 -0
  476. package/dist/test/tiles/list/tile-list.test.js.map +1 -0
  477. package/dist/test/tiles/tile-dispatcher.test.d.ts +1 -0
  478. package/dist/test/tiles/tile-dispatcher.test.js +231 -0
  479. package/dist/test/tiles/tile-dispatcher.test.js.map +1 -0
  480. package/dist/test/tiles/tile-display-value-provider.test.d.ts +1 -0
  481. package/dist/test/tiles/tile-display-value-provider.test.js +142 -0
  482. package/dist/test/tiles/tile-display-value-provider.test.js.map +1 -0
  483. package/dist/test/tiles/tile-mediatype-icon.test.d.ts +1 -0
  484. package/dist/test/tiles/tile-mediatype-icon.test.js +145 -0
  485. package/dist/test/tiles/tile-mediatype-icon.test.js.map +1 -0
  486. package/dist/test/utils/array-equals.test.d.ts +1 -0
  487. package/dist/test/utils/array-equals.test.js +27 -0
  488. package/dist/test/utils/array-equals.test.js.map +1 -0
  489. package/dist/test/utils/format-count.test.d.ts +1 -0
  490. package/dist/test/utils/format-count.test.js +24 -0
  491. package/dist/test/utils/format-count.test.js.map +1 -0
  492. package/dist/test/utils/format-date.test.d.ts +1 -0
  493. package/dist/test/utils/format-date.test.js +61 -0
  494. package/dist/test/utils/format-date.test.js.map +1 -0
  495. package/dist/test/utils/format-unit-size.test.d.ts +1 -0
  496. package/dist/test/utils/format-unit-size.test.js +18 -0
  497. package/dist/test/utils/format-unit-size.test.js.map +1 -0
  498. package/dist/test/utils/local-date-from-utc.test.d.ts +1 -0
  499. package/dist/test/utils/local-date-from-utc.test.js +27 -0
  500. package/dist/test/utils/local-date-from-utc.test.js.map +1 -0
  501. package/eslint.config.mjs +53 -53
  502. package/index.html +24 -24
  503. package/local.archive.org.cert +86 -86
  504. package/local.archive.org.key +27 -27
  505. package/package.json +118 -117
  506. package/renovate.json +6 -6
  507. package/src/collection-browser.ts +2954 -2829
  508. package/src/collection-facets/facet-row.ts +299 -296
  509. package/src/collection-facets/models.ts +10 -10
  510. package/src/collection-facets/more-facets-content.ts +639 -639
  511. package/src/collection-facets.ts +1005 -995
  512. package/src/data-source/collection-browser-data-source-interface.ts +345 -333
  513. package/src/data-source/collection-browser-data-source.ts +1441 -1401
  514. package/src/data-source/collection-browser-query-state.ts +59 -65
  515. package/src/data-source/models.ts +56 -43
  516. package/src/manage/manage-bar.ts +247 -247
  517. package/src/models.ts +866 -870
  518. package/src/restoration-state-handler.ts +542 -544
  519. package/src/tiles/base-tile-component.ts +65 -65
  520. package/src/tiles/grid/account-tile.ts +113 -113
  521. package/src/tiles/grid/collection-tile.ts +163 -163
  522. package/src/tiles/grid/item-tile.ts +340 -340
  523. package/src/tiles/grid/search-tile.ts +90 -90
  524. package/src/tiles/grid/styles/tile-grid-shared-styles.ts +130 -130
  525. package/src/tiles/hover/hover-pane-controller.ts +613 -517
  526. package/src/tiles/hover/tile-hover-pane.ts +184 -180
  527. package/src/tiles/list/tile-list-compact.ts +239 -239
  528. package/src/tiles/list/tile-list.ts +700 -700
  529. package/src/tiles/tile-dispatcher.ts +517 -490
  530. package/src/utils/format-date.ts +62 -62
  531. package/test/collection-browser.test.ts +2413 -2403
  532. package/test/collection-facets/facet-row.test.ts +375 -375
  533. package/test/collection-facets.test.ts +928 -928
  534. package/test/restoration-state-handler.test.ts +480 -510
  535. package/test/tiles/grid/item-tile.test.ts +520 -520
  536. package/test/tiles/hover/hover-pane-controller.test.ts +418 -353
  537. package/test/tiles/list/tile-list-compact.test.ts +282 -282
  538. package/test/tiles/list/tile-list.test.ts +552 -552
  539. package/test/tiles/tile-dispatcher.test.ts +283 -187
  540. package/test/utils/format-date.test.ts +89 -89
  541. package/tsconfig.json +20 -20
  542. package/web-dev-server.config.mjs +30 -30
  543. package/web-test-runner.config.mjs +41 -41
@@ -1,995 +1,1005 @@
1
- import {
2
- css,
3
- html,
4
- LitElement,
5
- PropertyValues,
6
- nothing,
7
- TemplateResult,
8
- } from 'lit';
9
- import { customElement, property, state } from 'lit/decorators.js';
10
- import { map } from 'lit/directives/map.js';
11
- import { ref } from 'lit/directives/ref.js';
12
- import { msg } from '@lit/localize';
13
- import { classMap } from 'lit/directives/class-map.js';
14
- import {
15
- Aggregation,
16
- AggregationSortType,
17
- Bucket,
18
- FilterMap,
19
- SearchServiceInterface,
20
- SearchType,
21
- } from '@internetarchive/search-service';
22
- import '@internetarchive/histogram-date-range';
23
- import '@internetarchive/feature-feedback';
24
- import {
25
- ModalConfig,
26
- ModalManagerInterface,
27
- } from '@internetarchive/modal-manager';
28
- import type { FeatureFeedbackServiceInterface } from '@internetarchive/feature-feedback';
29
- import type { RecaptchaManagerInterface } from '@internetarchive/recaptcha-manager';
30
- import type { AnalyticsManagerInterface } from '@internetarchive/analytics-manager';
31
- import type { SharedResizeObserverInterface } from '@internetarchive/shared-resize-observer';
32
- import type {
33
- BarScalingOption,
34
- BinSnappingInterval,
35
- } from '@internetarchive/histogram-date-range';
36
- import chevronIcon from './assets/img/icons/chevron';
37
- import expandIcon from './assets/img/icons/expand';
38
- import {
39
- FacetOption,
40
- SelectedFacets,
41
- FacetGroup,
42
- FacetBucket,
43
- defaultFacetDisplayOrder,
44
- facetTitles,
45
- lendingFacetDisplayNames,
46
- lendingFacetKeysVisibility,
47
- LendingFacetKey,
48
- suppressedCollections,
49
- defaultFacetSort,
50
- FacetEventDetails,
51
- } from './models';
52
- import type {
53
- CollectionTitles,
54
- PageSpecifierParams,
55
- TVChannelAliases,
56
- } from './data-source/models';
57
- import {
58
- analyticsActions,
59
- analyticsCategories,
60
- } from './utils/analytics-events';
61
- import { srOnlyStyle } from './styles/sr-only';
62
- import { ExpandedDatePicker } from './expanded-date-picker';
63
- import {
64
- sortBucketsBySelectionState,
65
- updateSelectedFacetBucket,
66
- } from './utils/facet-utils';
67
-
68
- import '@internetarchive/histogram-date-range';
69
- import './collection-facets/more-facets-content';
70
- import './collection-facets/facets-template';
71
- import './collection-facets/facet-tombstone-row';
72
- import './expanded-date-picker';
73
-
74
- @customElement('collection-facets')
75
- export class CollectionFacets extends LitElement {
76
- @property({ type: Object }) searchService?: SearchServiceInterface;
77
-
78
- @property({ type: Number }) searchType?: SearchType;
79
-
80
- @property({ type: Object }) aggregations?: Record<string, Aggregation>;
81
-
82
- @property({ type: Object }) histogramAggregation?: Aggregation;
83
-
84
- @property({ type: String }) minSelectedDate?: string;
85
-
86
- @property({ type: String }) maxSelectedDate?: string;
87
-
88
- @property({ type: Boolean }) moreLinksVisible = true;
89
-
90
- @property({ type: Boolean }) facetsLoading = false;
91
-
92
- @property({ type: Boolean }) histogramAggregationLoading = false;
93
-
94
- @property({ type: Object }) selectedFacets?: SelectedFacets;
95
-
96
- @property({ type: Boolean }) collapsableFacets = false;
97
-
98
- @property({ type: Number }) contentWidth?: number;
99
-
100
- @property({ type: Boolean }) showHistogramDatePicker = false;
101
-
102
- @property({ type: Boolean }) allowExpandingDatePicker = false;
103
-
104
- @property({ type: Boolean }) suppressMediatypeFacets = false;
105
-
106
- @property({ type: String }) query?: string;
107
-
108
- @property({ type: Array }) identifiers?: string[];
109
-
110
- @property({ type: Object }) pageSpecifierParams?: PageSpecifierParams;
111
-
112
- @property({ type: Array }) parentCollections: string[] = [];
113
-
114
- @property({ type: Object }) filterMap?: FilterMap;
115
-
116
- @property({ type: String }) baseNavigationUrl?: string;
117
-
118
- @property({ type: String }) collectionPagePath: string = '/details/';
119
-
120
- @property({ type: Boolean }) isManageView = false;
121
-
122
- @property({ type: Boolean }) isTvSearch = false;
123
-
124
- @property({ type: Array }) facetDisplayOrder: FacetOption[] =
125
- defaultFacetDisplayOrder;
126
-
127
- @property({ type: Object, attribute: false })
128
- modalManager?: ModalManagerInterface;
129
-
130
- @property({ type: Object, attribute: false })
131
- resizeObserver?: SharedResizeObserverInterface;
132
-
133
- @property({ type: Object, attribute: false })
134
- featureFeedbackService?: FeatureFeedbackServiceInterface;
135
-
136
- @property({ type: Object, attribute: false })
137
- recaptchaManager?: RecaptchaManagerInterface;
138
-
139
- @property({ type: Object, attribute: false })
140
- analyticsHandler?: AnalyticsManagerInterface;
141
-
142
- @property({ type: Object, attribute: false })
143
- collectionTitles?: CollectionTitles;
144
-
145
- @property({ type: Object, attribute: false })
146
- tvChannelAliases?: TVChannelAliases;
147
-
148
- @state() openFacets: Record<FacetOption, boolean> = {
149
- subject: false,
150
- lending: false,
151
- mediatype: false,
152
- language: false,
153
- creator: false,
154
- collection: false,
155
- year: false,
156
- program: false,
157
- person: false,
158
- sponsor: false,
159
- };
160
-
161
- /**
162
- * Maximum # of facet buckets to render per facet group
163
- */
164
- private allowedFacetCount = 6;
165
-
166
- render() {
167
- const containerClasses = classMap({
168
- loading: this.facetsLoading,
169
- managing: this.isManageView,
170
- });
171
-
172
- // Added data-testid for Playwright testing
173
- // Using facet-group class and aria-labels is not ideal for Playwright locator
174
- const datePickerLabelId = 'date-picker-label';
175
- return html`
176
- <div id="container" class=${containerClasses}>
177
- ${this.showHistogramDatePicker &&
178
- (this.histogramAggregation || this.histogramAggregationLoading)
179
- ? html`
180
- <section
181
- class="facet-group"
182
- aria-labelledby=${datePickerLabelId}
183
- data-testid="facet-group-header-label-date-picker"
184
- >
185
- <h3 id=${datePickerLabelId}>
186
- Year Published <span class="sr-only">range filter</span>
187
- ${this.expandDatePickerBtnTemplate}
188
- </h3>
189
- ${this.histogramTemplate}
190
- </section>
191
- `
192
- : nothing}
193
- ${this.collectionPartOfTemplate}
194
- ${this.mergedFacets.map(facetGroup =>
195
- this.getFacetGroupTemplate(facetGroup),
196
- )}
197
- </div>
198
- `;
199
- }
200
-
201
- private get collectionPartOfTemplate(): TemplateResult | typeof nothing {
202
- // We only display the "Part Of" section on collection pages
203
- if (!this.parentCollections?.length) return nothing;
204
-
205
- // Added data-testid for Playwright testing
206
- // Using className and aria-labels is not ideal for Playwright locator
207
- const headingId = 'partof-heading';
208
- return html`
209
- <section
210
- class="facet-group partof-collections"
211
- aria-labelledby=${headingId}
212
- data-testid="facet-group-partof-collections"
213
- >
214
- <div class="facet-group-header">
215
- <h3 id=${headingId}>${msg('Part Of')}</h3>
216
- </div>
217
- <ul>
218
- ${map(this.parentCollections, collxn => {
219
- const collectionURL = `${this.baseNavigationUrl}${this.collectionPagePath}${collxn}`;
220
-
221
- return html` <li>
222
- <a
223
- href=${collectionURL}
224
- data-id=${collxn}
225
- @click=${this.partOfCollectionClicked}
226
- >
227
- ${this.collectionTitles?.get(collxn) ?? collxn}
228
- </a>
229
- </li>`;
230
- })}
231
- </ul>
232
- </section>
233
- `;
234
- }
235
-
236
- private partOfCollectionClicked(e: Event): void {
237
- this.analyticsHandler?.sendEvent({
238
- category: analyticsCategories.default,
239
- action: analyticsActions.partOfCollectionClicked,
240
- label: (e.target as HTMLElement).dataset.id,
241
- });
242
- }
243
-
244
- /**
245
- * Properties to pass into the date-picker histogram component
246
- */
247
- private get histogramProps() {
248
- const { histogramAggregation: aggregation } = this;
249
- if (!aggregation) return undefined;
250
-
251
- // Normalize some properties from the raw aggregation
252
- const firstYear =
253
- aggregation.first_bucket_year ?? aggregation.first_bucket_key;
254
- const lastYear =
255
- aggregation.last_bucket_year ?? aggregation.last_bucket_key;
256
- if (firstYear == null || lastYear == null) return undefined; // We at least need a start/end year defined
257
-
258
- const firstMonth = aggregation.first_bucket_month ?? 1;
259
- const lastMonth = aggregation.last_bucket_month ?? 12;
260
-
261
- const yearInterval = aggregation.interval ?? 1;
262
- const monthInterval = aggregation.interval_in_months ?? 12;
263
-
264
- const zeroPadMonth = (month: number) => month.toString().padStart(2, '0');
265
-
266
- // The date picker is configured differently for TV search, allowing month-level resolution
267
- if (this.isTvSearch) {
268
- // Whether the bucket interval is less than a year
269
- // (i.e., requires individual months to be handled & labeled)
270
- const mustHandleMonths = monthInterval < 12;
271
-
272
- return {
273
- buckets: aggregation.buckets as number[],
274
- dateFormat: 'YYYY-MM',
275
- tooltipDateFormat: mustHandleMonths ? 'MMM YYYY' : 'YYYY',
276
- tooltipLabel: 'broadcast',
277
- binSnapping: (mustHandleMonths
278
- ? 'month'
279
- : 'year') as BinSnappingInterval,
280
- barScaling: 'linear' as BarScalingOption,
281
- minDate: `${firstYear}-${zeroPadMonth(firstMonth)}`,
282
- maxDate: `${lastYear}-${zeroPadMonth(lastMonth + monthInterval - 1)}`,
283
- };
284
- }
285
-
286
- // All other search types use the same configuration
287
- return {
288
- buckets: aggregation.buckets as number[],
289
- dateFormat: 'YYYY',
290
- tooltipDateFormat: 'YYYY',
291
- tooltipLabel: 'item',
292
- binSnapping: 'year' as BinSnappingInterval,
293
- barScaling: 'logarithmic' as BarScalingOption,
294
- minDate: `${firstYear}`,
295
- maxDate: `${lastYear + yearInterval - 1}`,
296
- };
297
- }
298
-
299
- /**
300
- * Opens a modal dialog containing an enlarged version of the date picker.
301
- */
302
- private showDatePickerModal(): void {
303
- const { histogramProps } = this;
304
- if (!histogramProps) return;
305
-
306
- const {
307
- buckets,
308
- dateFormat,
309
- tooltipDateFormat,
310
- tooltipLabel,
311
- binSnapping,
312
- barScaling,
313
- minDate,
314
- maxDate,
315
- } = histogramProps;
316
-
317
- // Because the modal manager does not clear its DOM content after being closed,
318
- // it may try to render the exact same date picker template when it is reopened.
319
- // And because it isn't actually a descendent of this collection-facets component,
320
- // changes to the template defined here may not trigger a reactive update to the date
321
- // picker, resulting in it displaying a stale date range.
322
- // This ref callback ensures that every time the date picker modal is opened, it will
323
- // always propagate the most recent date range into the date picker regardless of
324
- // whether Lit thinks the update is necessary.
325
- const expandedDatePickerChanged = (elmt?: Element) => {
326
- if (elmt && elmt instanceof ExpandedDatePicker) {
327
- const expandedDatePicker = elmt as ExpandedDatePicker;
328
- expandedDatePicker.minSelectedDate = this.minSelectedDate;
329
- expandedDatePicker.maxSelectedDate = this.maxSelectedDate;
330
- }
331
- };
332
-
333
- const customModalContent = html`
334
- <expanded-date-picker
335
- ${ref(expandedDatePickerChanged)}
336
- .minDate=${minDate}
337
- .maxDate=${maxDate}
338
- .minSelectedDate=${this.minSelectedDate}
339
- .maxSelectedDate=${this.maxSelectedDate}
340
- .customDateFormat=${dateFormat}
341
- .customTooltipDateFormat=${tooltipDateFormat}
342
- .customTooltipLabel=${tooltipLabel}
343
- .binSnapping=${binSnapping}
344
- .barScaling=${barScaling}
345
- .buckets=${buckets}
346
- .modalManager=${this.modalManager}
347
- .analyticsHandler=${this.analyticsHandler}
348
- @histogramDateRangeApplied=${this.histogramDateRangeUpdated}
349
- @modalClosed=${this.handleExpandedDatePickerClosed}
350
- ></expanded-date-picker>
351
- `;
352
-
353
- const config = new ModalConfig({
354
- bodyColor: '#fff',
355
- headerColor: '#194880',
356
- showHeaderLogo: false,
357
- closeOnBackdropClick: true, // TODO: want to fire analytics
358
- title: html`${msg('Select a date range')}`,
359
- });
360
-
361
- this.modalManager?.classList.add('expanded-date-picker');
362
- this.modalManager?.showModal({
363
- config,
364
- customModalContent,
365
- userClosedModalCallback: this.handleExpandedDatePickerClosed,
366
- });
367
-
368
- this.analyticsHandler?.sendEvent({
369
- category: analyticsCategories.default,
370
- action: analyticsActions.histogramExpanded,
371
- label: window.location.href,
372
- });
373
- }
374
-
375
- private handleExpandedDatePickerClosed = (): void => {
376
- this.modalManager?.classList.remove('expanded-date-picker');
377
- };
378
-
379
- updated(changed: PropertyValues) {
380
- if (changed.has('selectedFacets')) {
381
- this.dispatchFacetsChangedEvent();
382
- }
383
- }
384
-
385
- // TODO: want to fire analytics?
386
- private dispatchFacetsChangedEvent() {
387
- const event = new CustomEvent<SelectedFacets>('facetsChanged', {
388
- detail: this.selectedFacets,
389
- });
390
- this.dispatchEvent(event);
391
- }
392
-
393
- /**
394
- * Template for the "Expand" button to show the date picker modal, or
395
- * `nothing` if that button should currently not be shown.
396
- */
397
- private get expandDatePickerBtnTemplate(): TemplateResult | typeof nothing {
398
- return this.allowExpandingDatePicker && !this.facetsLoading
399
- ? html`<button
400
- class="expand-date-picker-btn"
401
- aria-hidden="true"
402
- tabindex="-1"
403
- @click=${this.showDatePickerModal}
404
- >
405
- ${expandIcon}
406
- </button>`
407
- : nothing;
408
- }
409
-
410
- private get histogramTemplate(): TemplateResult | typeof nothing {
411
- if (this.histogramAggregationLoading) {
412
- return html` <div class="histogram-loading-indicator">&hellip;</div> `;
413
- }
414
-
415
- const { histogramProps } = this;
416
- if (!histogramProps) return nothing;
417
-
418
- const {
419
- buckets,
420
- dateFormat,
421
- tooltipDateFormat,
422
- tooltipLabel,
423
- binSnapping,
424
- barScaling,
425
- minDate,
426
- maxDate,
427
- } = histogramProps;
428
-
429
- return html`
430
- <histogram-date-range
431
- class=${this.isTvSearch ? 'wide-inputs' : ''}
432
- .minDate=${minDate}
433
- .maxDate=${maxDate}
434
- .minSelectedDate=${this.minSelectedDate ?? minDate}
435
- .maxSelectedDate=${this.maxSelectedDate ?? maxDate}
436
- .updateDelay=${100}
437
- .dateFormat=${dateFormat}
438
- .tooltipDateFormat=${tooltipDateFormat}
439
- .tooltipLabel=${tooltipLabel}
440
- .binSnapping=${binSnapping}
441
- .barScaling=${barScaling}
442
- .bins=${buckets}
443
- missingDataMessage="..."
444
- .width=${this.collapsableFacets && this.contentWidth
445
- ? this.contentWidth
446
- : 180}
447
- @histogramDateRangeUpdated=${this.histogramDateRangeUpdated}
448
- ></histogram-date-range>
449
- `;
450
- }
451
-
452
- /**
453
- * Dispatches a `histogramDateRangeUpdated` event with the date range copied from the
454
- * input event.
455
- *
456
- * Arrow function to ensure `this` is always bound to the current component.
457
- */
458
- private histogramDateRangeUpdated = (
459
- e: CustomEvent<{
460
- minDate: string;
461
- maxDate: string;
462
- }>,
463
- ): void => {
464
- const { minDate, maxDate } = e.detail;
465
- const event = new CustomEvent('histogramDateRangeUpdated', {
466
- detail: { minDate, maxDate },
467
- });
468
- this.dispatchEvent(event);
469
- };
470
-
471
- /**
472
- * Combines the selected facets with the aggregations to create a single list of facets
473
- */
474
- private get mergedFacets(): FacetGroup[] {
475
- const facetGroups: FacetGroup[] = [];
476
-
477
- this.facetDisplayOrder.forEach(facetKey => {
478
- if (facetKey === 'mediatype' && this.suppressMediatypeFacets) return;
479
-
480
- const selectedFacetGroup = this.selectedFacetGroups.find(
481
- group => group.key === facetKey,
482
- );
483
- const aggregateFacetGroup = this.aggregationFacetGroups.find(
484
- group => group.key === facetKey,
485
- );
486
-
487
- // if the user selected a facet, but it's not in the aggregation, we add it as-is
488
- if (selectedFacetGroup && !aggregateFacetGroup) {
489
- facetGroups.push(selectedFacetGroup);
490
- return;
491
- }
492
-
493
- // if we don't have an aggregate facet group, don't add this to the list
494
- if (!aggregateFacetGroup) return;
495
-
496
- // start with either the selected group if we have one, or the aggregate group
497
- const facetGroup = selectedFacetGroup ?? aggregateFacetGroup;
498
-
499
- // attach the counts to the selected buckets
500
- let bucketsWithCount =
501
- selectedFacetGroup?.buckets.map(bucket => {
502
- const selectedBucket = aggregateFacetGroup.buckets.find(
503
- b => b.key === bucket.key,
504
- );
505
- return selectedBucket
506
- ? {
507
- ...bucket,
508
- count: selectedBucket.count,
509
- }
510
- : bucket;
511
- }) ?? [];
512
-
513
- // append any additional buckets that were not selected
514
- aggregateFacetGroup.buckets.forEach(bucket => {
515
- const existingBucket = bucketsWithCount.find(b => b.key === bucket.key);
516
- if (existingBucket) return;
517
- bucketsWithCount.push(bucket);
518
- });
519
-
520
- /**
521
- * render limited facet items on page facet area
522
- *
523
- * - by-default we are showing 6 items
524
- * - additionally want to show all items (selected/suppressed) in page facet area
525
- */
526
- let allowedFacetCount = Object.keys(
527
- (selectedFacetGroup?.buckets as []) || [],
528
- )?.length;
529
- if (allowedFacetCount < this.allowedFacetCount) {
530
- allowedFacetCount = this.allowedFacetCount; // splice start index from 0th
531
- }
532
-
533
- // For lending facets, only include a specific subset of buckets
534
- if (facetKey === 'lending') {
535
- bucketsWithCount = bucketsWithCount.filter(
536
- bucket => lendingFacetKeysVisibility[bucket.key as LendingFacetKey],
537
- );
538
- }
539
-
540
- // Sort the FacetBuckets so that selected and hidden buckets come before the rest
541
- sortBucketsBySelectionState(bucketsWithCount, defaultFacetSort[facetKey]);
542
-
543
- // For mediatype facets, ensure the collection bucket is always shown if present
544
- if (facetKey === 'mediatype') {
545
- const collectionIndex = bucketsWithCount.findIndex(
546
- bucket => bucket.key === 'collection',
547
- );
548
-
549
- if (collectionIndex >= allowedFacetCount) {
550
- const [collectionBucket] = bucketsWithCount.splice(
551
- collectionIndex,
552
- 1,
553
- );
554
-
555
- // If we're showing lots of selected facets, ensure we're not cutting off the last one
556
- if (allowedFacetCount > this.allowedFacetCount) {
557
- allowedFacetCount += 1;
558
- }
559
-
560
- bucketsWithCount.splice(allowedFacetCount - 1, 0, collectionBucket);
561
- }
562
- }
563
-
564
- // For TV creator facets, uppercase the display text
565
- if (facetKey === 'creator' && this.isTvSearch) {
566
- bucketsWithCount.forEach(b => {
567
- b.displayText = (b.displayText ?? b.key)?.toLocaleUpperCase();
568
-
569
- const channelLabel = this.tvChannelAliases?.get(b.displayText);
570
- if (channelLabel && channelLabel !== b.displayText) {
571
- b.extraNote = `(${channelLabel})`;
572
- }
573
- });
574
- }
575
-
576
- // slice off how many items we want to show in page facet area
577
- facetGroup.buckets = bucketsWithCount.slice(0, allowedFacetCount);
578
-
579
- facetGroups.push(facetGroup);
580
- });
581
-
582
- return facetGroups;
583
- }
584
-
585
- /**
586
- * Converts the selected facets to a `FacetGroup` array,
587
- * which is easier to work with
588
- */
589
- private get selectedFacetGroups(): FacetGroup[] {
590
- if (!this.selectedFacets) return [];
591
-
592
- const facetGroups: FacetGroup[] = Object.entries(this.selectedFacets).map(
593
- ([key, selectedFacets]) => {
594
- const option = key as FacetOption;
595
- const title = facetTitles[option];
596
-
597
- const buckets: FacetBucket[] = Object.entries(selectedFacets).map(
598
- ([value, facetData]) => {
599
- let displayText: string = value;
600
- // for lending facets, convert the key to a readable format
601
- if (option === 'lending') {
602
- displayText =
603
- lendingFacetDisplayNames[value as LendingFacetKey] ?? value;
604
- }
605
- return {
606
- displayText,
607
- key: value,
608
- count: facetData.count,
609
- state: facetData.state,
610
- };
611
- },
612
- );
613
-
614
- return {
615
- title,
616
- key: option,
617
- buckets,
618
- };
619
- },
620
- );
621
-
622
- return facetGroups;
623
- }
624
-
625
- /**
626
- * Converts the raw `aggregations` to `FacetGroups`, which are easier to use
627
- */
628
- private get aggregationFacetGroups(): FacetGroup[] {
629
- const facetGroups: FacetGroup[] = [];
630
- Object.entries(this.aggregations ?? []).forEach(([key, aggregation]) => {
631
- // the year_histogram and date_histogram data is in a different format so can't be handled here
632
- if (['year_histogram', 'date_histogram'].includes(key)) return;
633
-
634
- const option = key as FacetOption;
635
- const title = facetTitles[option];
636
- if (!title) return;
637
-
638
- let castedBuckets = aggregation.getSortedBuckets(
639
- defaultFacetSort[option],
640
- ) as Bucket[];
641
-
642
- if (option === 'collection') {
643
- // we are not showing fav- collections or certain deemphasized collections in facets
644
- castedBuckets = castedBuckets?.filter(bucket => {
645
- const bucketKey = bucket?.key?.toString();
646
- return (
647
- !suppressedCollections[bucketKey] && !bucketKey?.startsWith('fav-')
648
- );
649
- });
650
- }
651
-
652
- const facetBuckets: FacetBucket[] = castedBuckets.map(bucket => {
653
- const bucketKey = bucket.key;
654
- let displayText = `${bucket.key}`;
655
- // for lending facets, convert the bucket key to a readable format
656
- if (option === 'lending') {
657
- displayText =
658
- lendingFacetDisplayNames[bucket.key as LendingFacetKey] ??
659
- `${bucket.key}`;
660
- }
661
- return {
662
- displayText,
663
- key: `${bucketKey}`,
664
- count: bucket.doc_count,
665
- state: 'none',
666
- };
667
- });
668
- const group: FacetGroup = {
669
- title,
670
- key: option,
671
- buckets: facetBuckets,
672
- };
673
- facetGroups.push(group);
674
- });
675
- return facetGroups;
676
- }
677
-
678
- /**
679
- * Generate the template for a facet group with a header and the collapsible
680
- * chevron for the mobile view
681
- */
682
- private getFacetGroupTemplate(
683
- facetGroup: FacetGroup,
684
- ): TemplateResult | typeof nothing {
685
- if (!this.facetsLoading && facetGroup.buckets.length === 0) return nothing;
686
-
687
- const { key } = facetGroup;
688
- const isOpen = this.openFacets[key];
689
- const collapser = html`
690
- <span class="collapser ${isOpen ? 'open' : ''}"> ${chevronIcon} </span>
691
- `;
692
-
693
- const toggleCollapsed = () => {
694
- const newOpenFacets = { ...this.openFacets };
695
- newOpenFacets[key] = !isOpen;
696
- this.openFacets = newOpenFacets;
697
- };
698
-
699
- // Added data-testid for Playwright testing
700
- // Using className and aria-labels is not ideal for Playwright locator
701
- const headerId = `facet-group-header-label-${facetGroup.key}`;
702
- return html`
703
- <section
704
- class="facet-group ${this.collapsableFacets ? 'mobile' : ''}"
705
- aria-labelledby=${headerId}
706
- data-testid=${headerId}
707
- >
708
- <div class="facet-group-header">
709
- <h3
710
- id=${headerId}
711
- @click=${toggleCollapsed}
712
- @keyup=${toggleCollapsed}
713
- >
714
- ${this.collapsableFacets ? collapser : nothing} ${facetGroup.title}
715
- <span class="sr-only">filters</span>
716
- </h3>
717
- </div>
718
- <div
719
- class="facet-group-content ${isOpen ? 'open' : ''}"
720
- data-testid="facet-group-content-${facetGroup.key}"
721
- >
722
- ${this.facetsLoading
723
- ? this.getTombstoneFacetGroupTemplate()
724
- : html`
725
- ${this.getFacetTemplate(facetGroup)}
726
- ${this.searchMoreFacetsLink(facetGroup)}
727
- `}
728
- </div>
729
- </section>
730
- `;
731
- }
732
-
733
- private getTombstoneFacetGroupTemplate(): TemplateResult {
734
- // Render five tombstone rows
735
- return html`
736
- ${map(
737
- Array(5).fill(null),
738
- () => html`<facet-tombstone-row></facet-tombstone-row>`,
739
- )}
740
- `;
741
- }
742
-
743
- /**
744
- * Generate the More... link button just below the facets group
745
- *
746
- * TODO: want to fire analytics?
747
- */
748
- private searchMoreFacetsLink(
749
- facetGroup: FacetGroup,
750
- ): TemplateResult | typeof nothing {
751
- // Don't render More... links for FTS searches
752
- if (!this.moreLinksVisible) {
753
- return nothing;
754
- }
755
-
756
- // Don't render More... links for lending facets
757
- if (facetGroup.key === 'lending') {
758
- return nothing;
759
- }
760
-
761
- // Don't render More... link if the number of facets < this.allowedFacetCount
762
- if (Object.keys(facetGroup.buckets).length < this.allowedFacetCount) {
763
- return nothing;
764
- }
765
-
766
- // We sort years in numeric order by default, rather than bucket count
767
- const facetSort = defaultFacetSort[facetGroup.key];
768
-
769
- // Added data-testid for Playwright testing
770
- // Using the className is not ideal for Playwright locator
771
- return html`<button
772
- class="more-link"
773
- @click=${() => {
774
- this.showMoreFacetsModal(facetGroup, facetSort);
775
- this.analyticsHandler?.sendEvent({
776
- category: analyticsCategories.default,
777
- action: analyticsActions.showMoreFacetsModal,
778
- label: facetGroup.key,
779
- });
780
- this.dispatchEvent(
781
- new CustomEvent('showMoreFacets', { detail: facetGroup.key }),
782
- );
783
- }}
784
- data-testid="more-link-btn"
785
- >
786
- More...
787
- </button>`;
788
- }
789
-
790
- async showMoreFacetsModal(
791
- facetGroup: FacetGroup,
792
- sortedBy: AggregationSortType,
793
- ): Promise<void> {
794
- const customModalContent = html`
795
- <more-facets-content
796
- .analyticsHandler=${this.analyticsHandler}
797
- .facetKey=${facetGroup.key}
798
- .query=${this.query}
799
- .identifiers=${this.identifiers}
800
- .filterMap=${this.filterMap}
801
- .pageSpecifierParams=${this.pageSpecifierParams}
802
- .modalManager=${this.modalManager}
803
- .searchService=${this.searchService}
804
- .searchType=${this.searchType}
805
- .collectionTitles=${this.collectionTitles}
806
- .tvChannelAliases=${this.tvChannelAliases}
807
- .selectedFacets=${this.selectedFacets}
808
- .sortedBy=${sortedBy}
809
- .isTvSearch=${this.isTvSearch}
810
- @facetsChanged=${(e: CustomEvent) => {
811
- const event = new CustomEvent<SelectedFacets>('facetsChanged', {
812
- detail: e.detail,
813
- bubbles: true,
814
- composed: true,
815
- });
816
- this.dispatchEvent(event);
817
- }}
818
- >
819
- </more-facets-content>
820
- `;
821
-
822
- const config = new ModalConfig({
823
- bodyColor: '#fff',
824
- headerColor: '#194880',
825
- showHeaderLogo: false,
826
- closeOnBackdropClick: true, // TODO: want to fire analytics
827
- title: html`Select filters`,
828
- });
829
- this.modalManager?.classList.add('more-search-facets');
830
- this.modalManager?.showModal({
831
- config,
832
- customModalContent,
833
- userClosedModalCallback: () => {
834
- this.modalManager?.classList.remove('more-search-facets');
835
- },
836
- });
837
- }
838
-
839
- /**
840
- * Generate the list template for each bucket in a facet group
841
- */
842
- private getFacetTemplate(facetGroup: FacetGroup): TemplateResult {
843
- return html`
844
- <facets-template
845
- .collectionPagePath=${this.collectionPagePath}
846
- .facetGroup=${facetGroup}
847
- .selectedFacets=${this.selectedFacets}
848
- .collectionTitles=${this.collectionTitles}
849
- @facetClick=${(e: CustomEvent<FacetEventDetails>) => {
850
- this.selectedFacets = updateSelectedFacetBucket(
851
- this.selectedFacets,
852
- facetGroup.key,
853
- e.detail.bucket,
854
- true,
855
- );
856
-
857
- const event = new CustomEvent<SelectedFacets>('facetsChanged', {
858
- detail: this.selectedFacets,
859
- bubbles: true,
860
- composed: true,
861
- });
862
- this.dispatchEvent(event);
863
- }}
864
- ></facets-template>
865
- `;
866
- }
867
-
868
- static get styles() {
869
- return [
870
- srOnlyStyle,
871
- css`
872
- a:link {
873
- text-decoration: none;
874
- color: var(--ia-theme-link-color, #4b64ff);
875
- }
876
- a:link:hover {
877
- text-decoration: underline;
878
- }
879
-
880
- #container.loading {
881
- opacity: 0.5;
882
- }
883
-
884
- #container.managing {
885
- opacity: 0.3;
886
- }
887
-
888
- .histogram-loading-indicator {
889
- width: 100%;
890
- height: 2.25rem;
891
- margin-top: 1.75rem;
892
- font-size: 1.4rem;
893
- text-align: center;
894
- }
895
-
896
- .collapser {
897
- display: inline-block;
898
- cursor: pointer;
899
- width: 10px;
900
- height: 10px;
901
- }
902
-
903
- .collapser svg {
904
- transition: transform 0.2s ease-in-out;
905
- }
906
-
907
- .collapser.open svg {
908
- transform: rotate(90deg);
909
- }
910
-
911
- .facet-group:not(:last-child) {
912
- margin-bottom: 2rem;
913
- }
914
-
915
- .facet-group h3 {
916
- margin-bottom: 0.7rem;
917
- }
918
-
919
- .facet-group.mobile h3 {
920
- cursor: pointer;
921
- }
922
-
923
- .facet-group-header {
924
- display: flex;
925
- margin-bottom: 0.7rem;
926
- justify-content: space-between;
927
- border-bottom: 1px solid rgb(232, 232, 232);
928
- }
929
-
930
- .facet-group-content {
931
- transition: max-height 0.2s ease-in-out;
932
- }
933
-
934
- .facet-group.mobile .facet-group-content {
935
- max-height: 0;
936
- overflow: hidden;
937
- }
938
-
939
- .facet-group.mobile .facet-group-content.open {
940
- max-height: 2000px;
941
- }
942
-
943
- .partof-collections ul {
944
- list-style-type: none;
945
- padding: 0;
946
- font-size: 1.2rem;
947
- }
948
-
949
- h3 {
950
- font-size: 1.4rem;
951
- margin: 0;
952
- }
953
-
954
- .more-link {
955
- font-size: 1.2rem;
956
- text-decoration: none;
957
- padding: 0;
958
- margin-top: 0.25rem;
959
- background: inherit;
960
- border: 0;
961
- color: var(--ia-theme-link-color, #4b64ff);
962
- cursor: pointer;
963
- }
964
-
965
- #date-picker-label {
966
- display: flex;
967
- justify-content: space-between;
968
- }
969
-
970
- .expand-date-picker-btn {
971
- margin: 0;
972
- padding: 0;
973
- border: 0;
974
- appearance: none;
975
- background: none;
976
- cursor: pointer;
977
- }
978
-
979
- .expand-date-picker-btn > svg {
980
- width: 14px;
981
- height: 14px;
982
- }
983
-
984
- .sorting-icon {
985
- height: 15px;
986
- cursor: pointer;
987
- }
988
-
989
- histogram-date-range.wide-inputs {
990
- --histogramDateRangeInputWidth: 4.8rem;
991
- }
992
- `,
993
- ];
994
- }
995
- }
1
+ import {
2
+ css,
3
+ html,
4
+ LitElement,
5
+ PropertyValues,
6
+ nothing,
7
+ TemplateResult,
8
+ } from 'lit';
9
+ import { customElement, property, state } from 'lit/decorators.js';
10
+ import { map } from 'lit/directives/map.js';
11
+ import { ref } from 'lit/directives/ref.js';
12
+ import { msg } from '@lit/localize';
13
+ import { classMap } from 'lit/directives/class-map.js';
14
+ import {
15
+ Aggregation,
16
+ AggregationSortType,
17
+ Bucket,
18
+ FilterMap,
19
+ SearchServiceInterface,
20
+ SearchType,
21
+ } from '@internetarchive/search-service';
22
+ import '@internetarchive/histogram-date-range';
23
+ import '@internetarchive/feature-feedback';
24
+ import {
25
+ ModalConfig,
26
+ ModalManagerInterface,
27
+ } from '@internetarchive/modal-manager';
28
+ import type { FeatureFeedbackServiceInterface } from '@internetarchive/feature-feedback';
29
+ import type { RecaptchaManagerInterface } from '@internetarchive/recaptcha-manager';
30
+ import type { AnalyticsManagerInterface } from '@internetarchive/analytics-manager';
31
+ import type { SharedResizeObserverInterface } from '@internetarchive/shared-resize-observer';
32
+ import type {
33
+ BarScalingOption,
34
+ BinSnappingInterval,
35
+ } from '@internetarchive/histogram-date-range';
36
+ import chevronIcon from './assets/img/icons/chevron';
37
+ import expandIcon from './assets/img/icons/expand';
38
+ import {
39
+ FacetOption,
40
+ SelectedFacets,
41
+ FacetGroup,
42
+ FacetBucket,
43
+ defaultFacetDisplayOrder,
44
+ facetTitles,
45
+ lendingFacetDisplayNames,
46
+ lendingFacetKeysVisibility,
47
+ LendingFacetKey,
48
+ suppressedCollections,
49
+ defaultFacetSort,
50
+ FacetEventDetails,
51
+ } from './models';
52
+ import type {
53
+ CollectionTitles,
54
+ PageSpecifierParams,
55
+ TVChannelAliases,
56
+ } from './data-source/models';
57
+ import {
58
+ analyticsActions,
59
+ analyticsCategories,
60
+ } from './utils/analytics-events';
61
+ import { srOnlyStyle } from './styles/sr-only';
62
+ import { ExpandedDatePicker } from './expanded-date-picker';
63
+ import {
64
+ sortBucketsBySelectionState,
65
+ updateSelectedFacetBucket,
66
+ } from './utils/facet-utils';
67
+
68
+ import '@internetarchive/histogram-date-range';
69
+ import './collection-facets/more-facets-content';
70
+ import './collection-facets/facets-template';
71
+ import './collection-facets/facet-tombstone-row';
72
+ import './expanded-date-picker';
73
+
74
+ @customElement('collection-facets')
75
+ export class CollectionFacets extends LitElement {
76
+ @property({ type: Object }) searchService?: SearchServiceInterface;
77
+
78
+ @property({ type: Number }) searchType?: SearchType;
79
+
80
+ @property({ type: Object }) aggregations?: Record<string, Aggregation>;
81
+
82
+ @property({ type: Object }) histogramAggregation?: Aggregation;
83
+
84
+ @property({ type: String }) minSelectedDate?: string;
85
+
86
+ @property({ type: String }) maxSelectedDate?: string;
87
+
88
+ @property({ type: Boolean }) moreLinksVisible = true;
89
+
90
+ @property({ type: Boolean }) facetsLoading = false;
91
+
92
+ @property({ type: Boolean }) histogramAggregationLoading = false;
93
+
94
+ @property({ type: Object }) selectedFacets?: SelectedFacets;
95
+
96
+ @property({ type: Boolean }) collapsableFacets = false;
97
+
98
+ @property({ type: Number }) contentWidth?: number;
99
+
100
+ @property({ type: Boolean }) showHistogramDatePicker = false;
101
+
102
+ @property({ type: Boolean }) allowExpandingDatePicker = false;
103
+
104
+ @property({ type: Boolean }) suppressMediatypeFacets = false;
105
+
106
+ @property({ type: String }) query?: string;
107
+
108
+ @property({ type: Array }) identifiers?: string[];
109
+
110
+ @property({ type: Object }) pageSpecifierParams?: PageSpecifierParams;
111
+
112
+ @property({ type: Array }) parentCollections: string[] = [];
113
+
114
+ @property({ type: Object }) filterMap?: FilterMap;
115
+
116
+ @property({ type: String }) baseNavigationUrl?: string;
117
+
118
+ @property({ type: String }) collectionPagePath: string = '/details/';
119
+
120
+ @property({ type: Boolean }) isManageView = false;
121
+
122
+ @property({ type: Boolean }) isTvSearch = false;
123
+
124
+ @property({ type: Array }) facetDisplayOrder: FacetOption[] =
125
+ defaultFacetDisplayOrder;
126
+
127
+ @property({ type: Object, attribute: false })
128
+ modalManager?: ModalManagerInterface;
129
+
130
+ @property({ type: Object, attribute: false })
131
+ resizeObserver?: SharedResizeObserverInterface;
132
+
133
+ @property({ type: Object, attribute: false })
134
+ featureFeedbackService?: FeatureFeedbackServiceInterface;
135
+
136
+ @property({ type: Object, attribute: false })
137
+ recaptchaManager?: RecaptchaManagerInterface;
138
+
139
+ @property({ type: Object, attribute: false })
140
+ analyticsHandler?: AnalyticsManagerInterface;
141
+
142
+ @property({ type: Object, attribute: false })
143
+ collectionTitles?: CollectionTitles;
144
+
145
+ @property({ type: Object, attribute: false })
146
+ tvChannelAliases?: TVChannelAliases;
147
+
148
+ @state() openFacets: Record<FacetOption, boolean> = {
149
+ subject: false,
150
+ lending: false,
151
+ mediatype: false,
152
+ language: false,
153
+ creator: false,
154
+ collection: false,
155
+ year: false,
156
+ clip_type: false,
157
+ program: false,
158
+ person: false,
159
+ sponsor: false,
160
+ };
161
+
162
+ /**
163
+ * Maximum # of facet buckets to render per facet group
164
+ */
165
+ private allowedFacetCount = 6;
166
+
167
+ render() {
168
+ const containerClasses = classMap({
169
+ loading: this.facetsLoading,
170
+ managing: this.isManageView,
171
+ });
172
+
173
+ // Added data-testid for Playwright testing
174
+ // Using facet-group class and aria-labels is not ideal for Playwright locator
175
+ const datePickerLabelId = 'date-picker-label';
176
+ return html`
177
+ <div id="container" class=${containerClasses}>
178
+ ${this.showHistogramDatePicker &&
179
+ (this.histogramAggregation || this.histogramAggregationLoading)
180
+ ? html`
181
+ <section
182
+ class="facet-group"
183
+ aria-labelledby=${datePickerLabelId}
184
+ data-testid="facet-group-header-label-date-picker"
185
+ >
186
+ <h3 id=${datePickerLabelId}>
187
+ Year Published <span class="sr-only">range filter</span>
188
+ ${this.expandDatePickerBtnTemplate}
189
+ </h3>
190
+ ${this.histogramTemplate}
191
+ </section>
192
+ `
193
+ : nothing}
194
+ ${this.collectionPartOfTemplate}
195
+ <slot name="facets-top"></slot>
196
+ ${this.mergedFacets.map(facetGroup =>
197
+ this.getFacetGroupTemplate(facetGroup),
198
+ )}
199
+ </div>
200
+ `;
201
+ }
202
+
203
+ private get collectionPartOfTemplate(): TemplateResult | typeof nothing {
204
+ // We only display the "Part Of" section on collection pages
205
+ if (!this.parentCollections?.length) return nothing;
206
+
207
+ // Added data-testid for Playwright testing
208
+ // Using className and aria-labels is not ideal for Playwright locator
209
+ const headingId = 'partof-heading';
210
+ return html`
211
+ <section
212
+ class="facet-group partof-collections"
213
+ aria-labelledby=${headingId}
214
+ data-testid="facet-group-partof-collections"
215
+ >
216
+ <div class="facet-group-header">
217
+ <h3 id=${headingId}>${msg('Part Of')}</h3>
218
+ </div>
219
+ <ul>
220
+ ${map(this.parentCollections, collxn => {
221
+ const collectionURL = `${this.baseNavigationUrl}${this.collectionPagePath}${collxn}`;
222
+
223
+ return html` <li>
224
+ <a
225
+ href=${collectionURL}
226
+ data-id=${collxn}
227
+ @click=${this.partOfCollectionClicked}
228
+ >
229
+ ${this.collectionTitles?.get(collxn) ?? collxn}
230
+ </a>
231
+ </li>`;
232
+ })}
233
+ </ul>
234
+ </section>
235
+ `;
236
+ }
237
+
238
+ private partOfCollectionClicked(e: Event): void {
239
+ this.analyticsHandler?.sendEvent({
240
+ category: analyticsCategories.default,
241
+ action: analyticsActions.partOfCollectionClicked,
242
+ label: (e.target as HTMLElement).dataset.id,
243
+ });
244
+ }
245
+
246
+ /**
247
+ * Properties to pass into the date-picker histogram component
248
+ */
249
+ private get histogramProps() {
250
+ const { histogramAggregation: aggregation } = this;
251
+ if (!aggregation) return undefined;
252
+
253
+ // Normalize some properties from the raw aggregation
254
+ const firstYear =
255
+ aggregation.first_bucket_year ?? aggregation.first_bucket_key;
256
+ const lastYear =
257
+ aggregation.last_bucket_year ?? aggregation.last_bucket_key;
258
+ if (firstYear == null || lastYear == null) return undefined; // We at least need a start/end year defined
259
+
260
+ const firstMonth = aggregation.first_bucket_month ?? 1;
261
+ const lastMonth = aggregation.last_bucket_month ?? 12;
262
+
263
+ const yearInterval = aggregation.interval ?? 1;
264
+ const monthInterval = aggregation.interval_in_months ?? 12;
265
+
266
+ const zeroPadMonth = (month: number) => month.toString().padStart(2, '0');
267
+
268
+ // The date picker is configured differently for TV search, allowing month-level resolution
269
+ if (this.isTvSearch) {
270
+ // Whether the bucket interval is less than a year
271
+ // (i.e., requires individual months to be handled & labeled)
272
+ const mustHandleMonths = monthInterval < 12;
273
+
274
+ return {
275
+ buckets: aggregation.buckets as number[],
276
+ dateFormat: 'YYYY-MM',
277
+ tooltipDateFormat: mustHandleMonths ? 'MMM YYYY' : 'YYYY',
278
+ tooltipLabel: 'broadcast',
279
+ binSnapping: (mustHandleMonths
280
+ ? 'month'
281
+ : 'year') as BinSnappingInterval,
282
+ barScaling: 'linear' as BarScalingOption,
283
+ minDate: `${firstYear}-${zeroPadMonth(firstMonth)}`,
284
+ maxDate: `${lastYear}-${zeroPadMonth(lastMonth + monthInterval - 1)}`,
285
+ };
286
+ }
287
+
288
+ // All other search types use the same configuration
289
+ return {
290
+ buckets: aggregation.buckets as number[],
291
+ dateFormat: 'YYYY',
292
+ tooltipDateFormat: 'YYYY',
293
+ tooltipLabel: 'item',
294
+ binSnapping: 'year' as BinSnappingInterval,
295
+ barScaling: 'logarithmic' as BarScalingOption,
296
+ minDate: `${firstYear}`,
297
+ maxDate: `${lastYear + yearInterval - 1}`,
298
+ };
299
+ }
300
+
301
+ /**
302
+ * Opens a modal dialog containing an enlarged version of the date picker.
303
+ */
304
+ private showDatePickerModal(): void {
305
+ const { histogramProps } = this;
306
+ if (!histogramProps) return;
307
+
308
+ const {
309
+ buckets,
310
+ dateFormat,
311
+ tooltipDateFormat,
312
+ tooltipLabel,
313
+ binSnapping,
314
+ barScaling,
315
+ minDate,
316
+ maxDate,
317
+ } = histogramProps;
318
+
319
+ // Because the modal manager does not clear its DOM content after being closed,
320
+ // it may try to render the exact same date picker template when it is reopened.
321
+ // And because it isn't actually a descendent of this collection-facets component,
322
+ // changes to the template defined here may not trigger a reactive update to the date
323
+ // picker, resulting in it displaying a stale date range.
324
+ // This ref callback ensures that every time the date picker modal is opened, it will
325
+ // always propagate the most recent date range into the date picker regardless of
326
+ // whether Lit thinks the update is necessary.
327
+ const expandedDatePickerChanged = (elmt?: Element) => {
328
+ if (elmt && elmt instanceof ExpandedDatePicker) {
329
+ const expandedDatePicker = elmt as ExpandedDatePicker;
330
+ expandedDatePicker.minSelectedDate = this.minSelectedDate;
331
+ expandedDatePicker.maxSelectedDate = this.maxSelectedDate;
332
+ }
333
+ };
334
+
335
+ const customModalContent = html`
336
+ <expanded-date-picker
337
+ ${ref(expandedDatePickerChanged)}
338
+ .minDate=${minDate}
339
+ .maxDate=${maxDate}
340
+ .minSelectedDate=${this.minSelectedDate}
341
+ .maxSelectedDate=${this.maxSelectedDate}
342
+ .customDateFormat=${dateFormat}
343
+ .customTooltipDateFormat=${tooltipDateFormat}
344
+ .customTooltipLabel=${tooltipLabel}
345
+ .binSnapping=${binSnapping}
346
+ .barScaling=${barScaling}
347
+ .buckets=${buckets}
348
+ .modalManager=${this.modalManager}
349
+ .analyticsHandler=${this.analyticsHandler}
350
+ @histogramDateRangeApplied=${this.histogramDateRangeUpdated}
351
+ @modalClosed=${this.handleExpandedDatePickerClosed}
352
+ ></expanded-date-picker>
353
+ `;
354
+
355
+ const config = new ModalConfig({
356
+ bodyColor: '#fff',
357
+ headerColor: '#194880',
358
+ showHeaderLogo: false,
359
+ closeOnBackdropClick: true, // TODO: want to fire analytics
360
+ title: html`${msg('Select a date range')}`,
361
+ });
362
+
363
+ this.modalManager?.classList.add('expanded-date-picker');
364
+ this.modalManager?.showModal({
365
+ config,
366
+ customModalContent,
367
+ userClosedModalCallback: this.handleExpandedDatePickerClosed,
368
+ });
369
+
370
+ this.analyticsHandler?.sendEvent({
371
+ category: analyticsCategories.default,
372
+ action: analyticsActions.histogramExpanded,
373
+ label: window.location.href,
374
+ });
375
+ }
376
+
377
+ private handleExpandedDatePickerClosed = (): void => {
378
+ this.modalManager?.classList.remove('expanded-date-picker');
379
+ };
380
+
381
+ updated(changed: PropertyValues) {
382
+ if (changed.has('selectedFacets')) {
383
+ this.dispatchFacetsChangedEvent();
384
+ }
385
+ }
386
+
387
+ // TODO: want to fire analytics?
388
+ private dispatchFacetsChangedEvent() {
389
+ const event = new CustomEvent<SelectedFacets>('facetsChanged', {
390
+ detail: this.selectedFacets,
391
+ });
392
+ this.dispatchEvent(event);
393
+ }
394
+
395
+ /**
396
+ * Template for the "Expand" button to show the date picker modal, or
397
+ * `nothing` if that button should currently not be shown.
398
+ */
399
+ private get expandDatePickerBtnTemplate(): TemplateResult | typeof nothing {
400
+ return this.allowExpandingDatePicker && !this.facetsLoading
401
+ ? html`<button
402
+ class="expand-date-picker-btn"
403
+ aria-haspopup="dialog"
404
+ @click=${this.showDatePickerModal}
405
+ >
406
+ ${expandIcon}
407
+ </button>`
408
+ : nothing;
409
+ }
410
+
411
+ private get histogramTemplate(): TemplateResult | typeof nothing {
412
+ if (this.histogramAggregationLoading) {
413
+ return html` <div class="histogram-loading-indicator">&hellip;</div> `;
414
+ }
415
+
416
+ const { histogramProps } = this;
417
+ if (!histogramProps) return nothing;
418
+
419
+ const {
420
+ buckets,
421
+ dateFormat,
422
+ tooltipDateFormat,
423
+ tooltipLabel,
424
+ binSnapping,
425
+ barScaling,
426
+ minDate,
427
+ maxDate,
428
+ } = histogramProps;
429
+
430
+ return html`
431
+ <histogram-date-range
432
+ class=${this.isTvSearch ? 'wide-inputs' : ''}
433
+ .minDate=${minDate}
434
+ .maxDate=${maxDate}
435
+ .minSelectedDate=${this.minSelectedDate ?? minDate}
436
+ .maxSelectedDate=${this.maxSelectedDate ?? maxDate}
437
+ .updateDelay=${100}
438
+ .dateFormat=${dateFormat}
439
+ .tooltipDateFormat=${tooltipDateFormat}
440
+ .tooltipLabel=${tooltipLabel}
441
+ .binSnapping=${binSnapping}
442
+ .barScaling=${barScaling}
443
+ .bins=${buckets}
444
+ missingDataMessage="..."
445
+ .width=${this.collapsableFacets && this.contentWidth
446
+ ? this.contentWidth
447
+ : 180}
448
+ @histogramDateRangeUpdated=${this.histogramDateRangeUpdated}
449
+ ></histogram-date-range>
450
+ `;
451
+ }
452
+
453
+ /**
454
+ * Dispatches a `histogramDateRangeUpdated` event with the date range copied from the
455
+ * input event.
456
+ *
457
+ * Arrow function to ensure `this` is always bound to the current component.
458
+ */
459
+ private histogramDateRangeUpdated = (
460
+ e: CustomEvent<{
461
+ minDate: string;
462
+ maxDate: string;
463
+ }>,
464
+ ): void => {
465
+ const { minDate, maxDate } = e.detail;
466
+ const event = new CustomEvent('histogramDateRangeUpdated', {
467
+ detail: { minDate, maxDate },
468
+ });
469
+ this.dispatchEvent(event);
470
+ };
471
+
472
+ /**
473
+ * Combines the selected facets with the aggregations to create a single list of facets
474
+ */
475
+ private get mergedFacets(): FacetGroup[] {
476
+ const facetGroups: FacetGroup[] = [];
477
+
478
+ this.facetDisplayOrder.forEach(facetKey => {
479
+ if (facetKey === 'mediatype' && this.suppressMediatypeFacets) return;
480
+
481
+ const selectedFacetGroup = this.selectedFacetGroups.find(
482
+ group => group.key === facetKey,
483
+ );
484
+ const aggregateFacetGroup = this.aggregationFacetGroups.find(
485
+ group => group.key === facetKey,
486
+ );
487
+
488
+ // if the user selected a facet, but it's not in the aggregation, we add it as-is
489
+ if (selectedFacetGroup && !aggregateFacetGroup) {
490
+ facetGroups.push(selectedFacetGroup);
491
+ return;
492
+ }
493
+
494
+ // if we don't have an aggregate facet group, don't add this to the list
495
+ if (!aggregateFacetGroup) return;
496
+
497
+ // start with either the selected group if we have one, or the aggregate group
498
+ const facetGroup = selectedFacetGroup ?? aggregateFacetGroup;
499
+
500
+ // attach the counts to the selected buckets
501
+ let bucketsWithCount =
502
+ selectedFacetGroup?.buckets.map(bucket => {
503
+ const selectedBucket = aggregateFacetGroup.buckets.find(
504
+ b => b.key === bucket.key,
505
+ );
506
+ return selectedBucket
507
+ ? {
508
+ ...bucket,
509
+ count: selectedBucket.count,
510
+ }
511
+ : bucket;
512
+ }) ?? [];
513
+
514
+ // append any additional buckets that were not selected
515
+ aggregateFacetGroup.buckets.forEach(bucket => {
516
+ const existingBucket = bucketsWithCount.find(b => b.key === bucket.key);
517
+ if (existingBucket) return;
518
+ bucketsWithCount.push(bucket);
519
+ });
520
+
521
+ /**
522
+ * render limited facet items on page facet area
523
+ *
524
+ * - by-default we are showing 6 items
525
+ * - additionally want to show all items (selected/suppressed) in page facet area
526
+ */
527
+ let allowedFacetCount = Object.keys(
528
+ (selectedFacetGroup?.buckets as []) || [],
529
+ )?.length;
530
+ if (allowedFacetCount < this.allowedFacetCount) {
531
+ allowedFacetCount = this.allowedFacetCount; // splice start index from 0th
532
+ }
533
+
534
+ // For lending facets, only include a specific subset of buckets
535
+ if (facetKey === 'lending') {
536
+ bucketsWithCount = bucketsWithCount.filter(
537
+ bucket => lendingFacetKeysVisibility[bucket.key as LendingFacetKey],
538
+ );
539
+ }
540
+
541
+ // Sort the FacetBuckets so that selected and hidden buckets come before the rest
542
+ sortBucketsBySelectionState(bucketsWithCount, defaultFacetSort[facetKey]);
543
+
544
+ // For mediatype facets, ensure the collection bucket is always shown if present
545
+ if (facetKey === 'mediatype') {
546
+ const collectionIndex = bucketsWithCount.findIndex(
547
+ bucket => bucket.key === 'collection',
548
+ );
549
+
550
+ if (collectionIndex >= allowedFacetCount) {
551
+ const [collectionBucket] = bucketsWithCount.splice(
552
+ collectionIndex,
553
+ 1,
554
+ );
555
+
556
+ // If we're showing lots of selected facets, ensure we're not cutting off the last one
557
+ if (allowedFacetCount > this.allowedFacetCount) {
558
+ allowedFacetCount += 1;
559
+ }
560
+
561
+ bucketsWithCount.splice(allowedFacetCount - 1, 0, collectionBucket);
562
+ }
563
+ }
564
+
565
+ // For TV creator facets, uppercase the display text
566
+ if (facetKey === 'creator' && this.isTvSearch) {
567
+ bucketsWithCount.forEach(b => {
568
+ b.displayText = (b.displayText ?? b.key)?.toLocaleUpperCase();
569
+
570
+ const channelLabel = this.tvChannelAliases?.get(b.displayText);
571
+ if (channelLabel && channelLabel !== b.displayText) {
572
+ b.extraNote = `(${channelLabel})`;
573
+ }
574
+ });
575
+ }
576
+ // For TV clip_type facets, capitalize the display text
577
+ if (facetKey === 'clip_type') {
578
+ bucketsWithCount.forEach(b => {
579
+ b.displayText ??= b.key;
580
+ b.displayText =
581
+ b.displayText.charAt(0).toLocaleUpperCase() +
582
+ b.displayText.slice(1);
583
+ });
584
+ }
585
+
586
+ // slice off how many items we want to show in page facet area
587
+ facetGroup.buckets = bucketsWithCount.slice(0, allowedFacetCount);
588
+
589
+ facetGroups.push(facetGroup);
590
+ });
591
+
592
+ return facetGroups;
593
+ }
594
+
595
+ /**
596
+ * Converts the selected facets to a `FacetGroup` array,
597
+ * which is easier to work with
598
+ */
599
+ private get selectedFacetGroups(): FacetGroup[] {
600
+ if (!this.selectedFacets) return [];
601
+
602
+ const facetGroups: FacetGroup[] = Object.entries(this.selectedFacets).map(
603
+ ([key, selectedFacets]) => {
604
+ const option = key as FacetOption;
605
+ const title = facetTitles[option];
606
+
607
+ const buckets: FacetBucket[] = Object.entries(selectedFacets).map(
608
+ ([value, facetData]) => {
609
+ let displayText: string = value;
610
+ // for lending facets, convert the key to a readable format
611
+ if (option === 'lending') {
612
+ displayText =
613
+ lendingFacetDisplayNames[value as LendingFacetKey] ?? value;
614
+ }
615
+ return {
616
+ displayText,
617
+ key: value,
618
+ count: facetData.count,
619
+ state: facetData.state,
620
+ };
621
+ },
622
+ );
623
+
624
+ return {
625
+ title,
626
+ key: option,
627
+ buckets,
628
+ };
629
+ },
630
+ );
631
+
632
+ return facetGroups;
633
+ }
634
+
635
+ /**
636
+ * Converts the raw `aggregations` to `FacetGroups`, which are easier to use
637
+ */
638
+ private get aggregationFacetGroups(): FacetGroup[] {
639
+ const facetGroups: FacetGroup[] = [];
640
+ Object.entries(this.aggregations ?? []).forEach(([key, aggregation]) => {
641
+ // the year_histogram and date_histogram data is in a different format so can't be handled here
642
+ if (['year_histogram', 'date_histogram'].includes(key)) return;
643
+
644
+ const option = key as FacetOption;
645
+ const title = facetTitles[option];
646
+ if (!title) return;
647
+
648
+ let castedBuckets = aggregation.getSortedBuckets(
649
+ defaultFacetSort[option],
650
+ ) as Bucket[];
651
+
652
+ if (option === 'collection') {
653
+ // we are not showing fav- collections or certain deemphasized collections in facets
654
+ castedBuckets = castedBuckets?.filter(bucket => {
655
+ const bucketKey = bucket?.key?.toString();
656
+ return (
657
+ !suppressedCollections[bucketKey] && !bucketKey?.startsWith('fav-')
658
+ );
659
+ });
660
+ }
661
+
662
+ const facetBuckets: FacetBucket[] = castedBuckets.map(bucket => {
663
+ const bucketKey = bucket.key;
664
+ let displayText = `${bucket.key}`;
665
+ // for lending facets, convert the bucket key to a readable format
666
+ if (option === 'lending') {
667
+ displayText =
668
+ lendingFacetDisplayNames[bucket.key as LendingFacetKey] ??
669
+ `${bucket.key}`;
670
+ }
671
+ return {
672
+ displayText,
673
+ key: `${bucketKey}`,
674
+ count: bucket.doc_count,
675
+ state: 'none',
676
+ };
677
+ });
678
+ const group: FacetGroup = {
679
+ title,
680
+ key: option,
681
+ buckets: facetBuckets,
682
+ };
683
+ facetGroups.push(group);
684
+ });
685
+ return facetGroups;
686
+ }
687
+
688
+ /**
689
+ * Generate the template for a facet group with a header and the collapsible
690
+ * chevron for the mobile view
691
+ */
692
+ private getFacetGroupTemplate(
693
+ facetGroup: FacetGroup,
694
+ ): TemplateResult | typeof nothing {
695
+ if (!this.facetsLoading && facetGroup.buckets.length === 0) return nothing;
696
+
697
+ const { key } = facetGroup;
698
+ const isOpen = this.openFacets[key];
699
+ const collapser = html`
700
+ <span class="collapser ${isOpen ? 'open' : ''}"> ${chevronIcon} </span>
701
+ `;
702
+
703
+ const toggleCollapsed = () => {
704
+ const newOpenFacets = { ...this.openFacets };
705
+ newOpenFacets[key] = !isOpen;
706
+ this.openFacets = newOpenFacets;
707
+ };
708
+
709
+ // Added data-testid for Playwright testing
710
+ // Using className and aria-labels is not ideal for Playwright locator
711
+ const headerId = `facet-group-header-label-${facetGroup.key}`;
712
+ return html`
713
+ <section
714
+ class="facet-group ${this.collapsableFacets ? 'mobile' : ''}"
715
+ aria-labelledby=${headerId}
716
+ data-testid=${headerId}
717
+ >
718
+ <div class="facet-group-header">
719
+ <h3
720
+ id=${headerId}
721
+ @click=${toggleCollapsed}
722
+ @keyup=${toggleCollapsed}
723
+ >
724
+ ${this.collapsableFacets ? collapser : nothing} ${facetGroup.title}
725
+ <span class="sr-only">filters</span>
726
+ </h3>
727
+ </div>
728
+ <div
729
+ class="facet-group-content ${isOpen ? 'open' : ''}"
730
+ data-testid="facet-group-content-${facetGroup.key}"
731
+ >
732
+ ${this.facetsLoading
733
+ ? this.getTombstoneFacetGroupTemplate()
734
+ : html`
735
+ ${this.getFacetTemplate(facetGroup)}
736
+ ${this.searchMoreFacetsLink(facetGroup)}
737
+ `}
738
+ </div>
739
+ </section>
740
+ `;
741
+ }
742
+
743
+ private getTombstoneFacetGroupTemplate(): TemplateResult {
744
+ // Render five tombstone rows
745
+ return html`
746
+ ${map(
747
+ Array(5).fill(null),
748
+ () => html`<facet-tombstone-row></facet-tombstone-row>`,
749
+ )}
750
+ `;
751
+ }
752
+
753
+ /**
754
+ * Generate the More... link button just below the facets group
755
+ *
756
+ * TODO: want to fire analytics?
757
+ */
758
+ private searchMoreFacetsLink(
759
+ facetGroup: FacetGroup,
760
+ ): TemplateResult | typeof nothing {
761
+ // Don't render More... links for FTS searches
762
+ if (!this.moreLinksVisible) {
763
+ return nothing;
764
+ }
765
+
766
+ // Don't render More... links for lending facets
767
+ if (facetGroup.key === 'lending') {
768
+ return nothing;
769
+ }
770
+
771
+ // Don't render More... link if the number of facets < this.allowedFacetCount
772
+ if (Object.keys(facetGroup.buckets).length < this.allowedFacetCount) {
773
+ return nothing;
774
+ }
775
+
776
+ // We sort years in numeric order by default, rather than bucket count
777
+ const facetSort = defaultFacetSort[facetGroup.key];
778
+
779
+ // Added data-testid for Playwright testing
780
+ // Using the className is not ideal for Playwright locator
781
+ return html`<button
782
+ class="more-link"
783
+ @click=${() => {
784
+ this.showMoreFacetsModal(facetGroup, facetSort);
785
+ this.analyticsHandler?.sendEvent({
786
+ category: analyticsCategories.default,
787
+ action: analyticsActions.showMoreFacetsModal,
788
+ label: facetGroup.key,
789
+ });
790
+ this.dispatchEvent(
791
+ new CustomEvent('showMoreFacets', { detail: facetGroup.key }),
792
+ );
793
+ }}
794
+ data-testid="more-link-btn"
795
+ >
796
+ More...
797
+ </button>`;
798
+ }
799
+
800
+ async showMoreFacetsModal(
801
+ facetGroup: FacetGroup,
802
+ sortedBy: AggregationSortType,
803
+ ): Promise<void> {
804
+ const customModalContent = html`
805
+ <more-facets-content
806
+ .analyticsHandler=${this.analyticsHandler}
807
+ .facetKey=${facetGroup.key}
808
+ .query=${this.query}
809
+ .identifiers=${this.identifiers}
810
+ .filterMap=${this.filterMap}
811
+ .pageSpecifierParams=${this.pageSpecifierParams}
812
+ .modalManager=${this.modalManager}
813
+ .searchService=${this.searchService}
814
+ .searchType=${this.searchType}
815
+ .collectionTitles=${this.collectionTitles}
816
+ .tvChannelAliases=${this.tvChannelAliases}
817
+ .selectedFacets=${this.selectedFacets}
818
+ .sortedBy=${sortedBy}
819
+ .isTvSearch=${this.isTvSearch}
820
+ @facetsChanged=${(e: CustomEvent) => {
821
+ const event = new CustomEvent<SelectedFacets>('facetsChanged', {
822
+ detail: e.detail,
823
+ bubbles: true,
824
+ composed: true,
825
+ });
826
+ this.dispatchEvent(event);
827
+ }}
828
+ >
829
+ </more-facets-content>
830
+ `;
831
+
832
+ const config = new ModalConfig({
833
+ bodyColor: '#fff',
834
+ headerColor: '#194880',
835
+ showHeaderLogo: false,
836
+ closeOnBackdropClick: true, // TODO: want to fire analytics
837
+ title: html`Select filters`,
838
+ });
839
+ this.modalManager?.classList.add('more-search-facets');
840
+ this.modalManager?.showModal({
841
+ config,
842
+ customModalContent,
843
+ userClosedModalCallback: () => {
844
+ this.modalManager?.classList.remove('more-search-facets');
845
+ },
846
+ });
847
+ }
848
+
849
+ /**
850
+ * Generate the list template for each bucket in a facet group
851
+ */
852
+ private getFacetTemplate(facetGroup: FacetGroup): TemplateResult {
853
+ return html`
854
+ <facets-template
855
+ .collectionPagePath=${this.collectionPagePath}
856
+ .facetGroup=${facetGroup}
857
+ .selectedFacets=${this.selectedFacets}
858
+ .collectionTitles=${this.collectionTitles}
859
+ @facetClick=${(e: CustomEvent<FacetEventDetails>) => {
860
+ this.selectedFacets = updateSelectedFacetBucket(
861
+ this.selectedFacets,
862
+ facetGroup.key,
863
+ e.detail.bucket,
864
+ true,
865
+ );
866
+
867
+ const event = new CustomEvent<SelectedFacets>('facetsChanged', {
868
+ detail: this.selectedFacets,
869
+ bubbles: true,
870
+ composed: true,
871
+ });
872
+ this.dispatchEvent(event);
873
+ }}
874
+ ></facets-template>
875
+ `;
876
+ }
877
+
878
+ static get styles() {
879
+ return [
880
+ srOnlyStyle,
881
+ css`
882
+ a:link {
883
+ text-decoration: none;
884
+ color: var(--ia-theme-link-color, #4b64ff);
885
+ }
886
+ a:link:hover {
887
+ text-decoration: underline;
888
+ }
889
+
890
+ #container.loading {
891
+ opacity: 0.5;
892
+ }
893
+
894
+ #container.managing {
895
+ opacity: 0.3;
896
+ }
897
+
898
+ .histogram-loading-indicator {
899
+ width: 100%;
900
+ height: 2.25rem;
901
+ margin-top: 1.75rem;
902
+ font-size: 1.4rem;
903
+ text-align: center;
904
+ }
905
+
906
+ .collapser {
907
+ display: inline-block;
908
+ cursor: pointer;
909
+ width: 10px;
910
+ height: 10px;
911
+ }
912
+
913
+ .collapser svg {
914
+ transition: transform 0.2s ease-in-out;
915
+ }
916
+
917
+ .collapser.open svg {
918
+ transform: rotate(90deg);
919
+ }
920
+
921
+ .facet-group:not(:last-child) {
922
+ margin-bottom: 2rem;
923
+ }
924
+
925
+ .facet-group h3 {
926
+ margin-bottom: 0.7rem;
927
+ }
928
+
929
+ .facet-group.mobile h3 {
930
+ cursor: pointer;
931
+ }
932
+
933
+ .facet-group-header {
934
+ display: flex;
935
+ margin-bottom: 0.7rem;
936
+ justify-content: space-between;
937
+ border-bottom: 1px solid rgb(232, 232, 232);
938
+ }
939
+
940
+ .facet-group-content {
941
+ transition: max-height 0.2s ease-in-out;
942
+ }
943
+
944
+ .facet-group.mobile .facet-group-content {
945
+ max-height: 0;
946
+ overflow: hidden;
947
+ }
948
+
949
+ .facet-group.mobile .facet-group-content.open {
950
+ max-height: 2000px;
951
+ }
952
+
953
+ .partof-collections ul {
954
+ list-style-type: none;
955
+ padding: 0;
956
+ font-size: 1.2rem;
957
+ }
958
+
959
+ h3 {
960
+ font-size: 1.4rem;
961
+ margin: 0;
962
+ }
963
+
964
+ .more-link {
965
+ font-size: 1.2rem;
966
+ text-decoration: none;
967
+ padding: 0;
968
+ margin-top: 0.25rem;
969
+ background: inherit;
970
+ border: 0;
971
+ color: var(--ia-theme-link-color, #4b64ff);
972
+ cursor: pointer;
973
+ }
974
+
975
+ #date-picker-label {
976
+ display: flex;
977
+ justify-content: space-between;
978
+ }
979
+
980
+ .expand-date-picker-btn {
981
+ margin: 0;
982
+ padding: 0;
983
+ border: 0;
984
+ appearance: none;
985
+ background: none;
986
+ cursor: pointer;
987
+ }
988
+
989
+ .expand-date-picker-btn > svg {
990
+ width: 14px;
991
+ height: 14px;
992
+ }
993
+
994
+ .sorting-icon {
995
+ height: 15px;
996
+ cursor: pointer;
997
+ }
998
+
999
+ histogram-date-range.wide-inputs {
1000
+ --histogramDateRangeInputWidth: 4.8rem;
1001
+ }
1002
+ `,
1003
+ ];
1004
+ }
1005
+ }