@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
@@ -0,0 +1,1721 @@
1
+ import { aTimeout, expect, fixture } from '@open-wc/testing';
2
+ import { html } from 'lit';
3
+ import sinon from 'sinon';
4
+ import { FilterConstraint, SearchType } from '@internetarchive/search-service';
5
+ import '../src/collection-browser';
6
+ import { getDefaultSelectedFacets, SortField, } from '../src/models';
7
+ import { MockSearchService } from './mocks/mock-search-service';
8
+ import { MockAnalyticsHandler } from './mocks/mock-analytics-handler';
9
+ import { analyticsActions, analyticsCategories, } from '../src/utils/analytics-events';
10
+ /**
11
+ * Wait for the next tick of the event loop.
12
+ *
13
+ * This is necessary in some of the tests because certain collection browser
14
+ * updates take more than one tick to render (e.g., date picker & query changes).
15
+ * These delays are non-ideal and should eventually be investigated and fixed,
16
+ * but they are minor enough that waiting for the next tick is a reasonable
17
+ * testing solution for now.
18
+ */
19
+ const nextTick = () => aTimeout(0);
20
+ describe('Collection Browser', () => {
21
+ beforeEach(async () => {
22
+ // Apparently query params set by one test can bleed into other tests.
23
+ // Since collection browser restores its state from certain query params, we need
24
+ // to clear these before each test to ensure they run in isolation from one another.
25
+ const url = new URL(window.location.href);
26
+ const { searchParams } = url;
27
+ searchParams.delete('sin');
28
+ searchParams.delete('sort');
29
+ searchParams.delete('query');
30
+ searchParams.delete('page');
31
+ searchParams.delete('and[]');
32
+ searchParams.delete('not[]');
33
+ window.history.replaceState({}, '', url);
34
+ });
35
+ it('clears selected facets when requested', async () => {
36
+ const selectedFacets = getDefaultSelectedFacets();
37
+ selectedFacets.creator.foo = { count: 1, key: 'foo', state: 'selected' };
38
+ const el = await fixture(html `<collection-browser></collection-browser>`);
39
+ el.selectedFacets = selectedFacets;
40
+ await el.updateComplete;
41
+ el.clearFilters(); // By default, sort is not cleared
42
+ expect(el.selectedFacets).to.deep.equal(getDefaultSelectedFacets());
43
+ });
44
+ it('clears existing filters but not sort by default', async () => {
45
+ const el = await fixture(html `<collection-browser></collection-browser>`);
46
+ el.selectedSort = 'title';
47
+ el.sortDirection = 'asc';
48
+ await el.updateComplete;
49
+ el.clearFilters(); // By default, sort is not cleared
50
+ expect(el.selectedFacets).to.deep.equal(getDefaultSelectedFacets());
51
+ expect(el.selectedSort).to.equal('title');
52
+ expect(el.sortDirection).to.equal('asc');
53
+ expect(el.sortParam).to.deep.equal({
54
+ field: 'titleSorter',
55
+ direction: 'asc',
56
+ });
57
+ expect(el.selectedCreatorFilter).to.be.null;
58
+ expect(el.selectedTitleFilter).to.be.null;
59
+ });
60
+ it('clears existing filters for facets & sort via option', async () => {
61
+ const el = await fixture(html `<collection-browser></collection-browser>`);
62
+ el.selectedSort = 'title';
63
+ await el.updateComplete;
64
+ el.clearFilters({ sort: true }); // Sort is reset too due to the option
65
+ expect(el.selectedFacets).to.deep.equal(getDefaultSelectedFacets());
66
+ expect(el.selectedSort).to.equal(SortField.default);
67
+ expect(el.sortDirection).to.be.null;
68
+ expect(el.sortParam).to.be.null;
69
+ expect(el.selectedCreatorFilter).to.be.null;
70
+ expect(el.selectedTitleFilter).to.be.null;
71
+ });
72
+ it('filterBy creator with analytics', async () => {
73
+ const mockAnalyticsHandler = new MockAnalyticsHandler();
74
+ const el = await fixture(html `<collection-browser .analyticsHandler=${mockAnalyticsHandler}>
75
+ </collection-browser>`);
76
+ el.searchContext = 'betaSearchService';
77
+ el.selectedSort = 'creator';
78
+ el.sortDirection = 'asc';
79
+ el.selectedCreatorFilter = 'A';
80
+ await el.updateComplete;
81
+ expect(mockAnalyticsHandler.callCategory).to.equal('betaSearchService');
82
+ expect(mockAnalyticsHandler.callAction).to.equal('filterByCreator');
83
+ expect(mockAnalyticsHandler.callLabel).to.equal('start-A');
84
+ el.clearFilters();
85
+ await el.updateComplete;
86
+ expect(el.selectedTitleFilter).to.be.null;
87
+ expect(mockAnalyticsHandler.callCategory).to.equal('betaSearchService');
88
+ expect(mockAnalyticsHandler.callAction).to.equal('filterByCreator');
89
+ expect(mockAnalyticsHandler.callLabel).to.equal('clear-A');
90
+ });
91
+ it('filterBy title with analytics', async () => {
92
+ const mockAnalyticsHandler = new MockAnalyticsHandler();
93
+ const el = await fixture(html `<collection-browser .analyticsHandler=${mockAnalyticsHandler}>
94
+ </collection-browser>`);
95
+ el.searchContext = 'beta-search-service';
96
+ el.selectedSort = 'title';
97
+ el.sortDirection = 'asc';
98
+ el.selectedTitleFilter = 'A';
99
+ await el.updateComplete;
100
+ expect(mockAnalyticsHandler.callCategory).to.equal('beta-search-service');
101
+ expect(mockAnalyticsHandler.callAction).to.equal('filterByTitle');
102
+ expect(mockAnalyticsHandler.callLabel).to.equal('start-A');
103
+ el.clearFilters();
104
+ await el.updateComplete;
105
+ expect(el.selectedTitleFilter).to.be.null;
106
+ expect(mockAnalyticsHandler.callCategory).to.equal('beta-search-service');
107
+ expect(mockAnalyticsHandler.callAction).to.equal('filterByTitle');
108
+ expect(mockAnalyticsHandler.callLabel).to.equal('clear-A');
109
+ });
110
+ it('selected facets with analytics - not negative facets', async () => {
111
+ const mockAnalyticsHandler = new MockAnalyticsHandler();
112
+ const mediaTypeBucket = { count: 123, state: 'selected' };
113
+ const mockedSelectedFacets = {
114
+ subject: {},
115
+ lending: {},
116
+ mediatype: { data: mediaTypeBucket },
117
+ language: {},
118
+ creator: {},
119
+ collection: {},
120
+ year: {},
121
+ };
122
+ const el = await fixture(html `<collection-browser .analyticsHandler=${mockAnalyticsHandler}>
123
+ </collection-browser>`);
124
+ el.searchContext = 'search-service';
125
+ el.selectedFacets = mockedSelectedFacets;
126
+ await el.updateComplete;
127
+ el.facetClickHandler(new CustomEvent('facetClick', {
128
+ detail: {
129
+ facetType: 'mediatype',
130
+ bucket: {
131
+ key: '',
132
+ state: 'selected',
133
+ count: 123,
134
+ },
135
+ negative: false,
136
+ },
137
+ }));
138
+ expect(mockAnalyticsHandler.callCategory).to.equal('search-service');
139
+ expect(mockAnalyticsHandler.callAction).to.equal('facetSelected');
140
+ expect(mockAnalyticsHandler.callLabel).to.equal('mediatype');
141
+ el.facetClickHandler(new CustomEvent('facetClick', {
142
+ detail: {
143
+ facetType: 'mediatype',
144
+ bucket: {
145
+ key: '',
146
+ state: 'none',
147
+ count: 123,
148
+ },
149
+ negative: false,
150
+ },
151
+ }));
152
+ expect(el.selectedFacets).to.equal(mockedSelectedFacets);
153
+ expect(mockAnalyticsHandler.callCategory).to.equal('search-service');
154
+ expect(mockAnalyticsHandler.callAction).to.equal('facetDeselected');
155
+ expect(mockAnalyticsHandler.callLabel).to.equal('mediatype');
156
+ });
157
+ it('selected facets with analytics - negative facets', async () => {
158
+ const mockAnalyticsHandler = new MockAnalyticsHandler();
159
+ const mediaTypeBucket = { count: 123, state: 'selected' };
160
+ const mockedSelectedFacets = {
161
+ subject: {},
162
+ lending: {},
163
+ mediatype: { data: mediaTypeBucket },
164
+ language: {},
165
+ creator: {},
166
+ collection: {},
167
+ year: {},
168
+ };
169
+ const el = await fixture(html `<collection-browser .analyticsHandler=${mockAnalyticsHandler}>
170
+ </collection-browser>`);
171
+ el.searchContext = 'beta-search-service';
172
+ el.selectedFacets = mockedSelectedFacets;
173
+ await el.updateComplete;
174
+ el.facetClickHandler(new CustomEvent('facetClick', {
175
+ detail: {
176
+ facetType: 'mediatype',
177
+ bucket: {
178
+ key: '',
179
+ state: 'hidden',
180
+ count: 123,
181
+ },
182
+ negative: true,
183
+ },
184
+ }));
185
+ expect(mockAnalyticsHandler.callCategory).to.equal('beta-search-service');
186
+ expect(mockAnalyticsHandler.callAction).to.equal('facetNegativeSelected');
187
+ expect(mockAnalyticsHandler.callLabel).to.equal('mediatype');
188
+ el.facetClickHandler(new CustomEvent('facetClick', {
189
+ detail: {
190
+ facetType: 'mediatype',
191
+ bucket: {
192
+ key: '',
193
+ state: 'none',
194
+ count: 123,
195
+ },
196
+ negative: true,
197
+ },
198
+ }));
199
+ expect(el.selectedFacets).to.equal(mockedSelectedFacets);
200
+ expect(mockAnalyticsHandler.callCategory).to.equal('beta-search-service');
201
+ expect(mockAnalyticsHandler.callAction).to.equal('facetNegativeDeselected');
202
+ expect(mockAnalyticsHandler.callLabel).to.equal('mediatype');
203
+ });
204
+ it('should render with a sort bar, facets, and infinite scroller', async () => {
205
+ var _a, _b, _c;
206
+ const searchService = new MockSearchService();
207
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
208
+ </collection-browser>`);
209
+ el.baseQuery = 'hello';
210
+ await el.updateComplete;
211
+ await nextTick();
212
+ const facets = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('collection-facets');
213
+ const sortBar = (_b = el.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('sort-filter-bar');
214
+ const infiniteScroller = (_c = el.shadowRoot) === null || _c === void 0 ? void 0 : _c.querySelector('infinite-scroller');
215
+ expect(facets, 'facets').to.exist;
216
+ expect(sortBar, 'sort bar').to.exist;
217
+ expect(infiniteScroller, 'infinite scroller').to.exist;
218
+ });
219
+ it('queries the search service when given a base query', async () => {
220
+ var _a, _b, _c;
221
+ const searchService = new MockSearchService();
222
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
223
+ </collection-browser>`);
224
+ el.baseQuery = 'collection:foo';
225
+ await el.updateComplete;
226
+ await el.initialSearchComplete;
227
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('collection:foo');
228
+ expect((_c = (_b = el.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('#big-results-label')) === null || _c === void 0 ? void 0 : _c.textContent).to.contains('Results');
229
+ });
230
+ it('queries the search service when given a list of identifiers and no query', async () => {
231
+ var _a, _b, _c;
232
+ const searchService = new MockSearchService();
233
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
234
+ </collection-browser>`);
235
+ el.identifiers = ['foo', 'bar'];
236
+ await el.updateComplete;
237
+ await el.initialSearchComplete;
238
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.identifiers).to.deep.equal([
239
+ 'foo',
240
+ 'bar',
241
+ ]);
242
+ expect((_c = (_b = el.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('#big-results-label')) === null || _c === void 0 ? void 0 : _c.textContent).to.contains('Results');
243
+ });
244
+ it('queries the search service when given a list of identifiers with a query', async () => {
245
+ var _a, _b, _c, _d;
246
+ const searchService = new MockSearchService();
247
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
248
+ </collection-browser>`);
249
+ el.baseQuery = 'collection:foo';
250
+ el.identifiers = ['foo', 'bar'];
251
+ await el.updateComplete;
252
+ await el.initialSearchComplete;
253
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('collection:foo');
254
+ expect((_b = searchService.searchParams) === null || _b === void 0 ? void 0 : _b.identifiers).to.deep.equal([
255
+ 'foo',
256
+ 'bar',
257
+ ]);
258
+ expect((_d = (_c = el.shadowRoot) === null || _c === void 0 ? void 0 : _c.querySelector('#big-results-label')) === null || _d === void 0 ? void 0 : _d.textContent).to.contains('Results');
259
+ });
260
+ it('queries the search service with a metadata search', async () => {
261
+ var _a, _b, _c;
262
+ const searchService = new MockSearchService();
263
+ const el = await fixture(html ` <collection-browser .searchService=${searchService}>
264
+ </collection-browser>`);
265
+ el.searchType = SearchType.METADATA;
266
+ await el.updateComplete;
267
+ el.baseQuery = 'collection:foo';
268
+ await el.updateComplete;
269
+ await el.initialSearchComplete;
270
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('collection:foo');
271
+ expect(searchService.searchType).to.equal(SearchType.METADATA);
272
+ expect((_c = (_b = el.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('#big-results-label')) === null || _c === void 0 ? void 0 : _c.textContent).to.contains('Results');
273
+ });
274
+ it('can change search type', async () => {
275
+ const searchService = new MockSearchService();
276
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
277
+ </collection-browser>`);
278
+ el.baseQuery = 'collection:foo';
279
+ await el.updateComplete;
280
+ el.searchType = SearchType.FULLTEXT;
281
+ await el.updateComplete;
282
+ await el.initialSearchComplete;
283
+ expect(searchService.searchType).to.equal(SearchType.FULLTEXT);
284
+ });
285
+ it('queries the search service with a fulltext search', async () => {
286
+ var _a, _b, _c;
287
+ const searchService = new MockSearchService();
288
+ const el = await fixture(html ` <collection-browser .searchService=${searchService}>
289
+ </collection-browser>`);
290
+ el.searchType = SearchType.FULLTEXT;
291
+ await el.updateComplete;
292
+ el.baseQuery = 'collection:foo';
293
+ await el.updateComplete;
294
+ await el.initialSearchComplete;
295
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('collection:foo');
296
+ expect(searchService.searchType).to.equal(SearchType.FULLTEXT);
297
+ expect((_c = (_b = el.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('#big-results-label')) === null || _c === void 0 ? void 0 : _c.textContent).to.contains('Results');
298
+ });
299
+ it('queries the search service with a radio search', async () => {
300
+ var _a, _b, _c;
301
+ const searchService = new MockSearchService();
302
+ const el = await fixture(html ` <collection-browser .searchService=${searchService}>
303
+ </collection-browser>`);
304
+ el.searchType = SearchType.RADIO;
305
+ await el.updateComplete;
306
+ el.baseQuery = 'collection:foo';
307
+ await el.updateComplete;
308
+ await el.initialSearchComplete;
309
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('collection:foo');
310
+ expect(searchService.searchType).to.equal(SearchType.RADIO);
311
+ expect((_c = (_b = el.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('#big-results-label')) === null || _c === void 0 ? void 0 : _c.textContent).to.contains('Results');
312
+ });
313
+ it('queries the search service with a TV search', async () => {
314
+ var _a, _b, _c;
315
+ const searchService = new MockSearchService();
316
+ const el = await fixture(html ` <collection-browser .searchService=${searchService}>
317
+ </collection-browser>`);
318
+ el.searchType = SearchType.TV;
319
+ await el.updateComplete;
320
+ el.baseQuery = 'collection:foo';
321
+ await el.updateComplete;
322
+ await el.initialSearchComplete;
323
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('collection:foo');
324
+ expect(searchService.searchType).to.equal(SearchType.TV);
325
+ expect((_c = (_b = el.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('#big-results-label')) === null || _c === void 0 ? void 0 : _c.textContent).to.contains('Results');
326
+ });
327
+ it('queries the search service with facets selected/negated', async () => {
328
+ var _a, _b;
329
+ const searchService = new MockSearchService();
330
+ const selectedFacets = {
331
+ subject: {
332
+ foo: {
333
+ key: 'foo',
334
+ count: 1,
335
+ state: 'selected',
336
+ },
337
+ bar: {
338
+ key: 'bar',
339
+ count: 2,
340
+ state: 'hidden',
341
+ },
342
+ },
343
+ lending: {},
344
+ mediatype: {},
345
+ language: {
346
+ en: {
347
+ key: 'en',
348
+ count: 1,
349
+ state: 'selected',
350
+ },
351
+ },
352
+ creator: {},
353
+ collection: {},
354
+ year: {},
355
+ };
356
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
357
+ </collection-browser>`);
358
+ el.baseQuery = 'collection:foo';
359
+ el.selectedFacets = selectedFacets;
360
+ await el.updateComplete;
361
+ await el.initialSearchComplete;
362
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('collection:foo');
363
+ expect((_b = searchService.searchParams) === null || _b === void 0 ? void 0 : _b.filters).to.deep.equal({
364
+ subject: {
365
+ foo: 'inc',
366
+ bar: 'exc',
367
+ },
368
+ language: {
369
+ en: 'inc',
370
+ },
371
+ });
372
+ });
373
+ it('fails gracefully if no search service provided', async () => {
374
+ var _a;
375
+ const el = await fixture(html `<collection-browser></collection-browser>`);
376
+ el.baseQuery = 'collection:foo';
377
+ await el.updateComplete;
378
+ // This shouldn't throw an error
379
+ expect(el.dataSource.fetchPage(3)).to.exist;
380
+ // Should continue showing the empty placeholder
381
+ expect((_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('empty-placeholder')).to.exist;
382
+ });
383
+ it('restores search type from URL param', async () => {
384
+ // Add a sin=TXT param to the URL
385
+ const url = new URL(window.location.href);
386
+ url.searchParams.append('sin', 'TXT');
387
+ window.history.replaceState({}, '', url);
388
+ const searchService = new MockSearchService();
389
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
390
+ </collection-browser>`);
391
+ expect(el.searchType).to.equal(SearchType.FULLTEXT);
392
+ });
393
+ it('does not persist or restore search type from URL param if suppressed', async () => {
394
+ // Add a sin=TXT param to the URL
395
+ let url = new URL(window.location.href);
396
+ url.searchParams.append('sin', 'TXT');
397
+ window.history.replaceState({}, '', url);
398
+ const searchService = new MockSearchService();
399
+ const el = await fixture(html `<collection-browser
400
+ .searchService=${searchService}
401
+ suppressURLSinParam
402
+ >
403
+ </collection-browser>`);
404
+ url = new URL(window.location.href);
405
+ expect(el.searchType).to.equal(SearchType.DEFAULT);
406
+ expect(url.searchParams.has('sin')).to.be.false; // Removes existing sin param
407
+ el.searchType = SearchType.RADIO;
408
+ await el.updateComplete;
409
+ url = new URL(window.location.href);
410
+ expect(url.searchParams.has('sin')).to.be.false; // Doesn't add sin param
411
+ });
412
+ it('can construct tile models with many fields present', async () => {
413
+ const searchService = new MockSearchService();
414
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
415
+ </collection-browser>`);
416
+ el.baseQuery = 'many-fields';
417
+ await el.updateComplete;
418
+ await el.initialSearchComplete;
419
+ const cellTemplate = el.cellForIndex(0);
420
+ expect(cellTemplate).to.exist;
421
+ const cell = await fixture(cellTemplate);
422
+ expect(cell).to.exist;
423
+ });
424
+ it('emits empty results event when search fetches no results', async () => {
425
+ const searchService = new MockSearchService();
426
+ const emptyResultsSpy = sinon.spy();
427
+ const el = await fixture(html `<collection-browser
428
+ .searchService=${searchService}
429
+ @emptyResults=${emptyResultsSpy}
430
+ >
431
+ </collection-browser>`);
432
+ el.baseQuery = 'no-results';
433
+ await el.updateComplete;
434
+ await el.initialSearchComplete;
435
+ expect(emptyResultsSpy.callCount).to.equal(1);
436
+ });
437
+ it('emits searchError event when search results in an error', async () => {
438
+ const searchService = new MockSearchService();
439
+ const searchErrorSpy = sinon.spy();
440
+ const el = await fixture(html `<collection-browser
441
+ .searchService=${searchService}
442
+ @searchError=${searchErrorSpy}
443
+ >
444
+ </collection-browser>`);
445
+ el.baseQuery = 'error';
446
+ await el.updateComplete;
447
+ await el.initialSearchComplete;
448
+ expect(searchErrorSpy.callCount).to.equal(1);
449
+ });
450
+ it('applies loggedin flag to tile models if needed', async () => {
451
+ var _a;
452
+ const searchService = new MockSearchService();
453
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
454
+ </collection-browser>`);
455
+ el.baseQuery = 'loggedin';
456
+ await el.updateComplete;
457
+ await el.initialSearchComplete;
458
+ const cellTemplate = el.cellForIndex(0);
459
+ expect(cellTemplate).to.exist;
460
+ const cell = await fixture(cellTemplate);
461
+ expect(cell).to.exist;
462
+ expect((_a = cell.model) === null || _a === void 0 ? void 0 : _a.loginRequired).to.be.true;
463
+ });
464
+ it('applies no-preview flag to tile models if needed', async () => {
465
+ var _a;
466
+ const searchService = new MockSearchService();
467
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
468
+ </collection-browser>`);
469
+ el.baseQuery = 'no-preview';
470
+ await el.updateComplete;
471
+ await el.initialSearchComplete;
472
+ const cellTemplate = el.cellForIndex(0);
473
+ expect(cellTemplate).to.exist;
474
+ const cell = await fixture(cellTemplate);
475
+ expect(cell).to.exist;
476
+ expect((_a = cell.model) === null || _a === void 0 ? void 0 : _a.contentWarning).to.be.true;
477
+ });
478
+ it('both loggedin and no-preview flags can be set simultaneously', async () => {
479
+ var _a, _b;
480
+ const searchService = new MockSearchService();
481
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
482
+ </collection-browser>`);
483
+ el.baseQuery = 'loggedin-no-preview';
484
+ await el.updateComplete;
485
+ await el.initialSearchComplete;
486
+ const cellTemplate = el.cellForIndex(0);
487
+ expect(cellTemplate).to.exist;
488
+ const cell = await fixture(cellTemplate);
489
+ expect(cell).to.exist;
490
+ expect((_a = cell.model) === null || _a === void 0 ? void 0 : _a.loginRequired).to.be.true;
491
+ expect((_b = cell.model) === null || _b === void 0 ? void 0 : _b.contentWarning).to.be.true;
492
+ });
493
+ it('joins full description array into a single string with line breaks', async () => {
494
+ var _a;
495
+ const searchService = new MockSearchService();
496
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
497
+ </collection-browser>`);
498
+ // This query receives an array description like ['line1', 'line2']
499
+ el.baseQuery = 'multi-line-description';
500
+ await el.updateComplete;
501
+ await el.initialSearchComplete;
502
+ const cellTemplate = el.cellForIndex(0);
503
+ expect(cellTemplate).to.exist;
504
+ const cell = await fixture(cellTemplate);
505
+ expect(cell).to.exist;
506
+ // Actual model description should be joined
507
+ expect((_a = cell.model) === null || _a === void 0 ? void 0 : _a.description).to.equal('line1\nline2');
508
+ });
509
+ it('can change search type', async () => {
510
+ var _a;
511
+ const searchService = new MockSearchService();
512
+ const el = await fixture(html `<collection-browser
513
+ .searchService=${searchService}
514
+ .searchType=${SearchType.METADATA}
515
+ ></collection-browser>`);
516
+ el.baseQuery = 'collection:foo';
517
+ el.searchType = SearchType.FULLTEXT;
518
+ await el.updateComplete;
519
+ await el.initialSearchComplete;
520
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('collection:foo');
521
+ expect(searchService.searchType).to.equal(SearchType.FULLTEXT);
522
+ });
523
+ it('trims queries of leading/trailing whitespace', async () => {
524
+ var _a;
525
+ const searchService = new MockSearchService();
526
+ const el = await fixture(html `<collection-browser
527
+ .searchService=${searchService}
528
+ ></collection-browser>`);
529
+ el.baseQuery = ' collection:foo ';
530
+ await el.updateComplete;
531
+ await el.initialSearchComplete;
532
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('collection:foo');
533
+ });
534
+ it('shows error message when error response received', async () => {
535
+ var _a, _b;
536
+ const searchService = new MockSearchService();
537
+ const el = await fixture(html `<collection-browser
538
+ .searchService=${searchService}
539
+ ></collection-browser>`);
540
+ el.baseQuery = 'error';
541
+ await el.updateComplete;
542
+ await el.initialSearchComplete;
543
+ const errorPlaceholder = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('empty-placeholder');
544
+ const errorDetails = (_b = errorPlaceholder === null || errorPlaceholder === void 0 ? void 0 : errorPlaceholder.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('.error-details');
545
+ expect(errorDetails).to.exist;
546
+ expect(errorDetails.textContent).to.contain('foo');
547
+ });
548
+ it('shows error message when error response received for a collection', async () => {
549
+ var _a, _b;
550
+ const searchService = new MockSearchService();
551
+ const el = await fixture(html `<collection-browser
552
+ .searchService=${searchService}
553
+ ></collection-browser>`);
554
+ el.withinCollection = 'error';
555
+ await el.updateComplete;
556
+ await el.initialSearchComplete;
557
+ const errorPlaceholder = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('empty-placeholder');
558
+ const errorDetails = (_b = errorPlaceholder === null || errorPlaceholder === void 0 ? void 0 : errorPlaceholder.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('.error-details');
559
+ expect(errorDetails).to.exist;
560
+ expect(errorDetails.textContent).to.contain('foo');
561
+ });
562
+ it('reports malformed response errors to Sentry', async () => {
563
+ const sentrySpy = sinon.spy();
564
+ // eslint-disable-next-line @typescript-eslint/no-explicit-any
565
+ window.Sentry = { captureMessage: sentrySpy };
566
+ const searchService = new MockSearchService();
567
+ const el = await fixture(html `<collection-browser
568
+ .searchService=${searchService}
569
+ ></collection-browser>`);
570
+ el.baseQuery = 'malformed';
571
+ await el.updateComplete;
572
+ await el.initialSearchComplete;
573
+ expect(sentrySpy.callCount).to.be.greaterThanOrEqual(1);
574
+ });
575
+ it('adds collection names to cache when present on response', async () => {
576
+ const searchService = new MockSearchService();
577
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
578
+ </collection-browser>`);
579
+ el.baseQuery = 'collection-titles';
580
+ await el.updateComplete;
581
+ await el.initialSearchComplete;
582
+ expect(el.dataSource.collectionTitles.get('foo')).to.equal('Foo Collection');
583
+ expect(el.dataSource.collectionTitles.get('bar')).to.equal('Bar Collection');
584
+ expect(el.dataSource.collectionTitles.get('baz')).to.equal('Baz Collection');
585
+ expect(el.dataSource.collectionTitles.get('boop')).to.equal('Boop Collection');
586
+ });
587
+ it('adds tv channel aliases to cache when present on response', async () => {
588
+ const searchService = new MockSearchService();
589
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
590
+ </collection-browser>`);
591
+ el.baseQuery = 'channel-aliases';
592
+ await el.updateComplete;
593
+ await el.initialSearchComplete;
594
+ expect(el.dataSource.tvChannelAliases.get('foo')).to.equal('Foo Network');
595
+ expect(el.dataSource.tvChannelAliases.get('bar')).to.equal('Bar Network');
596
+ });
597
+ it('keeps search results from fetch if no change to query or sort param', async () => {
598
+ const resultsSpy = sinon.spy();
599
+ const searchService = new MockSearchService({
600
+ asyncResponse: true,
601
+ resultsSpy,
602
+ });
603
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
604
+ </collection-browser>`);
605
+ el.baseQuery = 'with-sort';
606
+ el.selectedSort = SortField.date;
607
+ el.sortDirection = 'asc';
608
+ await el.updateComplete;
609
+ await el.dataSource.fetchPage(3);
610
+ // If there is no change to the query or sort param during the fetch, the results
611
+ // should be read.
612
+ expect(resultsSpy.callCount).to.be.greaterThanOrEqual(1);
613
+ });
614
+ it('discards obsolete search results if sort params changed before arrival', async () => {
615
+ const resultsSpy = sinon.spy();
616
+ const searchService = new MockSearchService({
617
+ asyncResponse: true,
618
+ resultsSpy,
619
+ });
620
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
621
+ </collection-browser>`);
622
+ el.baseQuery = 'with-sort';
623
+ el.selectedSort = SortField.date;
624
+ el.sortDirection = 'asc';
625
+ await el.updateComplete;
626
+ // We want to spy exclusively on the first set of results, not the second
627
+ searchService.asyncResponse = false;
628
+ searchService.resultsSpy = () => { };
629
+ el.sortDirection = 'desc';
630
+ await el.updateComplete;
631
+ await el.initialSearchComplete;
632
+ // If the different sort param causes the results to be discarded,
633
+ // the first results array should never be read.
634
+ expect(resultsSpy.callCount).to.equal(0);
635
+ });
636
+ it('discards obsolete search results if sort param added before arrival', async () => {
637
+ const resultsSpy = sinon.spy();
638
+ const searchService = new MockSearchService({
639
+ asyncResponse: true,
640
+ resultsSpy,
641
+ });
642
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
643
+ </collection-browser>`);
644
+ el.baseQuery = 'single-result';
645
+ await el.updateComplete;
646
+ // We want to spy exclusively on the first set of results, not the second
647
+ searchService.asyncResponse = false;
648
+ searchService.resultsSpy = () => { };
649
+ el.selectedSort = SortField.date;
650
+ el.sortDirection = 'asc';
651
+ await el.updateComplete;
652
+ await el.initialSearchComplete;
653
+ // If the different sort param causes the results to be discarded,
654
+ // the first results array should never be read.
655
+ expect(resultsSpy.callCount).to.equal(0);
656
+ });
657
+ it('discards obsolete search results if sort param cleared before arrival', async () => {
658
+ const resultsSpy = sinon.spy();
659
+ const searchService = new MockSearchService({
660
+ asyncResponse: true,
661
+ resultsSpy,
662
+ });
663
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
664
+ </collection-browser>`);
665
+ el.baseQuery = 'with-sort';
666
+ el.selectedSort = SortField.date;
667
+ el.sortDirection = 'asc';
668
+ await el.updateComplete;
669
+ // We want to spy exclusively on the first set of results, not the second
670
+ searchService.asyncResponse = false;
671
+ searchService.resultsSpy = () => { };
672
+ el.selectedSort = SortField.default;
673
+ await el.updateComplete;
674
+ await el.initialSearchComplete;
675
+ // If the different sort param causes the results to be discarded,
676
+ // the first results array should never be read.
677
+ expect(resultsSpy.callCount).to.equal(0);
678
+ });
679
+ it('sets sort properties when user changes sort', async () => {
680
+ var _a, _b, _c, _d, _e, _f;
681
+ const searchService = new MockSearchService();
682
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
683
+ </collection-browser>`);
684
+ expect(el.selectedSort).to.equal(SortField.default);
685
+ el.baseQuery = 'foo';
686
+ await el.updateComplete;
687
+ await nextTick();
688
+ const sortBar = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('#content-container sort-filter-bar');
689
+ const sortSelector = (_b = sortBar === null || sortBar === void 0 ? void 0 : sortBar.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('#desktop-sort-selector');
690
+ expect(sortSelector, 'sort bar').to.exist;
691
+ // Click the title sorter
692
+ (_d = (_c = Array.from(sortSelector.children)
693
+ .find(child => { var _a; return ((_a = child.textContent) === null || _a === void 0 ? void 0 : _a.trim()) === 'Title'; })) === null || _c === void 0 ? void 0 : _c.querySelector('button')) === null || _d === void 0 ? void 0 : _d.click();
694
+ await el.updateComplete;
695
+ expect(el.selectedSort).to.equal(SortField.title);
696
+ // Click the creator sorter
697
+ (_f = (_e = Array.from(sortSelector.children)
698
+ .find(child => { var _a; return ((_a = child.textContent) === null || _a === void 0 ? void 0 : _a.trim()) === 'Creator'; })) === null || _e === void 0 ? void 0 : _e.querySelector('button')) === null || _f === void 0 ? void 0 : _f.click();
699
+ await el.updateComplete;
700
+ expect(el.selectedSort).to.equal(SortField.creator);
701
+ });
702
+ it('sets sort filter properties when user selects title filter', async () => {
703
+ var _a, _b, _c, _d;
704
+ const searchService = new MockSearchService();
705
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
706
+ </collection-browser>`);
707
+ el.baseQuery = 'first-title';
708
+ el.selectedSort = 'title';
709
+ el.sortDirection = 'asc';
710
+ el.selectedTitleFilter = 'X';
711
+ await el.updateComplete;
712
+ await el.initialSearchComplete;
713
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('first-title');
714
+ expect((_d = (_c = (_b = searchService.searchParams) === null || _b === void 0 ? void 0 : _b.filters) === null || _c === void 0 ? void 0 : _c.firstTitle) === null || _d === void 0 ? void 0 : _d.X).to.equal(FilterConstraint.INCLUDE);
715
+ });
716
+ it('sets sort filter properties when user selects creator filter', async () => {
717
+ var _a, _b, _c, _d;
718
+ const searchService = new MockSearchService();
719
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
720
+ </collection-browser>`);
721
+ el.baseQuery = 'first-creator';
722
+ el.selectedSort = 'creator';
723
+ el.sortDirection = 'asc';
724
+ el.selectedCreatorFilter = 'X';
725
+ await el.updateComplete;
726
+ await el.initialSearchComplete;
727
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('first-creator');
728
+ expect((_d = (_c = (_b = searchService.searchParams) === null || _b === void 0 ? void 0 : _b.filters) === null || _c === void 0 ? void 0 : _c.firstCreator) === null || _d === void 0 ? void 0 : _d.X).to.equal(FilterConstraint.INCLUDE);
729
+ });
730
+ it('sets sort filter properties simultaneous with facets and date range', async () => {
731
+ var _a, _b;
732
+ const searchService = new MockSearchService();
733
+ const selectedFacets = {
734
+ collection: { foo: { key: 'foo', state: 'selected', count: 1 } },
735
+ creator: {},
736
+ language: {},
737
+ lending: {},
738
+ mediatype: {},
739
+ subject: {},
740
+ year: {},
741
+ };
742
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
743
+ </collection-browser>`);
744
+ el.baseQuery = 'first-creator';
745
+ el.selectedSort = 'creator';
746
+ el.selectedFacets = selectedFacets;
747
+ el.minSelectedDate = '1950';
748
+ el.maxSelectedDate = '1970';
749
+ el.sortDirection = 'asc';
750
+ el.selectedCreatorFilter = 'X';
751
+ await el.updateComplete;
752
+ await el.initialSearchComplete;
753
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('first-creator');
754
+ expect((_b = searchService.searchParams) === null || _b === void 0 ? void 0 : _b.filters).to.deep.equal({
755
+ collection: {
756
+ foo: 'inc',
757
+ },
758
+ year: {
759
+ '1950': 'gte',
760
+ '1970': 'lte',
761
+ },
762
+ firstCreator: {
763
+ X: 'inc',
764
+ },
765
+ });
766
+ });
767
+ it('applies correct TV search filter for commercials', async () => {
768
+ var _a, _b, _c;
769
+ const searchService = new MockSearchService();
770
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
771
+ </collection-browser>`);
772
+ el.baseQuery = 'tv-fields';
773
+ el.searchType = SearchType.TV;
774
+ el.selectedFacets = {
775
+ clip_type: {
776
+ commercial: { key: 'commercial', count: 1, state: 'selected' },
777
+ },
778
+ };
779
+ await el.updateComplete;
780
+ await el.initialSearchComplete;
781
+ expect((_c = (_b = (_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.filters) === null || _b === void 0 ? void 0 : _b.clip_type) === null || _c === void 0 ? void 0 : _c.commercial).to.equal(FilterConstraint.INCLUDE);
782
+ });
783
+ it('applies correct TV search filter for fact checks', async () => {
784
+ var _a, _b, _c;
785
+ const searchService = new MockSearchService();
786
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
787
+ </collection-browser>`);
788
+ el.baseQuery = 'tv-fields';
789
+ el.searchType = SearchType.TV;
790
+ el.selectedFacets = {
791
+ clip_type: {
792
+ 'fact check': { key: 'fact check', count: 1, state: 'selected' },
793
+ },
794
+ };
795
+ await el.updateComplete;
796
+ await el.initialSearchComplete;
797
+ expect((_c = (_b = (_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.filters) === null || _b === void 0 ? void 0 : _b.clip_type) === null || _c === void 0 ? void 0 : _c['fact check']).to.equal(FilterConstraint.INCLUDE);
798
+ });
799
+ it('applies correct TV search filter for quotes', async () => {
800
+ var _a, _b, _c;
801
+ const searchService = new MockSearchService();
802
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
803
+ </collection-browser>`);
804
+ el.baseQuery = 'tv-fields';
805
+ el.searchType = SearchType.TV;
806
+ el.selectedFacets = {
807
+ clip_type: { quote: { key: 'quote', count: 1, state: 'selected' } },
808
+ };
809
+ await el.updateComplete;
810
+ await el.initialSearchComplete;
811
+ expect((_c = (_b = (_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.filters) === null || _b === void 0 ? void 0 : _b.clip_type) === null || _c === void 0 ? void 0 : _c.quote).to.equal(FilterConstraint.INCLUDE);
812
+ });
813
+ it('resets letter filters when query changes', async () => {
814
+ var _a, _b, _c, _d, _e, _f, _g;
815
+ const searchService = new MockSearchService();
816
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
817
+ </collection-browser>`);
818
+ el.baseQuery = 'first-creator';
819
+ el.selectedSort = 'creator';
820
+ el.sortDirection = 'asc';
821
+ el.selectedCreatorFilter = 'X';
822
+ await el.updateComplete;
823
+ await el.initialSearchComplete;
824
+ await nextTick();
825
+ expect((_a = searchService.searchParams) === null || _a === void 0 ? void 0 : _a.query).to.equal('first-creator');
826
+ expect((_d = (_c = (_b = searchService.searchParams) === null || _b === void 0 ? void 0 : _b.filters) === null || _c === void 0 ? void 0 : _c.firstCreator) === null || _d === void 0 ? void 0 : _d.X).to.equal(FilterConstraint.INCLUDE);
827
+ el.baseQuery = 'collection:foo';
828
+ await el.updateComplete;
829
+ await nextTick();
830
+ expect((_e = searchService.searchParams) === null || _e === void 0 ? void 0 : _e.query).to.equal('collection:foo');
831
+ expect((_g = (_f = searchService.searchParams) === null || _f === void 0 ? void 0 : _f.filters) === null || _g === void 0 ? void 0 : _g.firstCreator).not.to.exist;
832
+ });
833
+ it('sets date range query when date picker selection changed', async () => {
834
+ var _a, _b, _c;
835
+ const searchService = new MockSearchService();
836
+ const mockAnalyticsHandler = new MockAnalyticsHandler();
837
+ const el = await fixture(html `<collection-browser
838
+ .searchService=${searchService}
839
+ .analyticsHandler=${mockAnalyticsHandler}
840
+ .suppressPlaceholders=${true}
841
+ >
842
+ </collection-browser>`);
843
+ el.baseQuery = 'years'; // Includes year_histogram aggregation in response
844
+ el.showHistogramDatePicker = true;
845
+ await el.updateComplete;
846
+ const facets = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('collection-facets');
847
+ await (facets === null || facets === void 0 ? void 0 : facets.updateComplete);
848
+ // Wait for the date picker to be rendered (which may take until the next tick)
849
+ await nextTick();
850
+ const histogram = (_b = facets === null || facets === void 0 ? void 0 : facets.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('histogram-date-range');
851
+ expect(histogram, 'histogram exists').to.exist;
852
+ // Enter a new min date into the date picker
853
+ const minDateInput = (_c = histogram.shadowRoot) === null || _c === void 0 ? void 0 : _c.querySelector('#date-min');
854
+ const pressEnterEvent = new KeyboardEvent('keyup', {
855
+ key: 'Enter',
856
+ });
857
+ minDateInput.value = '1960';
858
+ minDateInput.dispatchEvent(pressEnterEvent);
859
+ // Wait for the histogram's update delay
860
+ await aTimeout(histogram.updateDelay + 50);
861
+ // Ensure that the histogram change propagated to the collection browser's
862
+ // date query correctly.
863
+ await el.updateComplete;
864
+ expect(el.minSelectedDate).to.equal('1960');
865
+ expect(el.maxSelectedDate).to.equal('2009');
866
+ });
867
+ it('sets date range query when monthly date picker selection changed', async () => {
868
+ var _a, _b, _c;
869
+ const searchService = new MockSearchService();
870
+ const el = await fixture(html `<collection-browser
871
+ .searchService=${searchService}
872
+ .suppressPlaceholders=${true}
873
+ >
874
+ </collection-browser>`);
875
+ el.baseQuery = 'months'; // Includes date_histogram aggregation in response
876
+ el.searchType = SearchType.TV;
877
+ el.showHistogramDatePicker = true;
878
+ await el.updateComplete;
879
+ const facets = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('collection-facets');
880
+ await (facets === null || facets === void 0 ? void 0 : facets.updateComplete);
881
+ // Wait for the date picker to be rendered (which may take until the next tick)
882
+ await nextTick();
883
+ const histogram = (_b = facets === null || facets === void 0 ? void 0 : facets.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('histogram-date-range');
884
+ expect(histogram, 'histogram exists').to.exist;
885
+ // Enter a new min date into the date picker
886
+ const minDateInput = (_c = histogram.shadowRoot) === null || _c === void 0 ? void 0 : _c.querySelector('#date-min');
887
+ const pressEnterEvent = new KeyboardEvent('keyup', {
888
+ key: 'Enter',
889
+ });
890
+ minDateInput.value = '2001-02';
891
+ minDateInput.dispatchEvent(pressEnterEvent);
892
+ // Wait for the histogram's update delay
893
+ await aTimeout(histogram.updateDelay + 50);
894
+ // Ensure that the histogram change propagated to the collection browser's
895
+ // date query correctly.
896
+ await el.updateComplete;
897
+ expect(el.minSelectedDate).to.equal('2001-02');
898
+ expect(el.maxSelectedDate).to.equal('2002-12');
899
+ });
900
+ it('emits event when results start and end loading', async () => {
901
+ var _a, _b, _c, _d;
902
+ const spy = sinon.spy();
903
+ const searchService = new MockSearchService();
904
+ const el = await fixture(html `<collection-browser
905
+ .searchService=${searchService}
906
+ @searchResultsLoadingChanged=${spy}
907
+ ></collection-browser>`);
908
+ spy.resetHistory();
909
+ el.baseQuery = 'collection:foo';
910
+ await el.updateComplete;
911
+ await el.initialSearchComplete;
912
+ // Should initially emit loading=true, then later emit loading=false
913
+ expect(spy.callCount).to.equal(2);
914
+ expect((_b = (_a = spy.firstCall.firstArg) === null || _a === void 0 ? void 0 : _a.detail) === null || _b === void 0 ? void 0 : _b.loading).to.equal(true);
915
+ expect((_d = (_c = spy.secondCall.firstArg) === null || _c === void 0 ? void 0 : _c.detail) === null || _d === void 0 ? void 0 : _d.loading).to.equal(false);
916
+ });
917
+ it('collapses extra set of quotes around href field', async () => {
918
+ var _a;
919
+ const searchService = new MockSearchService();
920
+ const el = await fixture(html `<collection-browser
921
+ .searchService=${searchService}
922
+ .baseNavigationUrl=${''}
923
+ ></collection-browser>`);
924
+ el.baseQuery = 'extra-quoted-href';
925
+ await el.updateComplete;
926
+ await el.initialSearchComplete;
927
+ await el.updateComplete;
928
+ await aTimeout(50);
929
+ // Original href q param starts/ends with %22%22, but should be collapsed to %22 before render
930
+ expect((_a = el.dataSource.getTileModelAt(0)) === null || _a === void 0 ? void 0 : _a.href).to.equal('/details/foo?q=%22quoted+query%22');
931
+ });
932
+ it('sets default sort from collection metadata', async () => {
933
+ var _a;
934
+ const searchService = new MockSearchService();
935
+ const el = await fixture(html `<collection-browser
936
+ .searchService=${searchService}
937
+ .baseNavigationUrl=${''}
938
+ ></collection-browser>`);
939
+ el.withinCollection = 'default-sort';
940
+ await el.updateComplete;
941
+ await el.initialSearchComplete;
942
+ await el.updateComplete;
943
+ await aTimeout(50);
944
+ const sortBar = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('sort-filter-bar');
945
+ expect(sortBar).to.exist;
946
+ expect(sortBar.defaultSortField).to.equal(SortField.title);
947
+ expect(sortBar.defaultSortDirection).to.equal('asc');
948
+ expect(sortBar.selectedSort).to.equal(SortField.default);
949
+ expect(sortBar.sortDirection).to.be.null;
950
+ });
951
+ it('sets default sort from collection metadata in "-field" format', async () => {
952
+ var _a;
953
+ const searchService = new MockSearchService();
954
+ const el = await fixture(html `<collection-browser
955
+ .searchService=${searchService}
956
+ .baseNavigationUrl=${''}
957
+ ></collection-browser>`);
958
+ el.withinCollection = 'default-sort-concise';
959
+ await el.updateComplete;
960
+ await el.initialSearchComplete;
961
+ await el.updateComplete;
962
+ await aTimeout(50);
963
+ const sortBar = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('sort-filter-bar');
964
+ expect(sortBar).to.exist;
965
+ expect(sortBar.defaultSortField).to.equal(SortField.dateadded);
966
+ expect(sortBar.defaultSortDirection).to.equal('desc');
967
+ expect(sortBar.selectedSort).to.equal(SortField.default);
968
+ expect(sortBar.sortDirection).to.be.null;
969
+ });
970
+ it('falls back to weekly views default sorting on profiles when tab not set', async () => {
971
+ const el = await fixture(html `<collection-browser
972
+ .withinProfile=${'@foobar'}
973
+ ></collection-browser>`);
974
+ el.applyDefaultProfileSort();
975
+ expect(el.defaultSortParam).to.deep.equal({
976
+ field: 'week',
977
+ direction: 'desc',
978
+ });
979
+ });
980
+ it('uses relevance sort as default when a query is set', async () => {
981
+ var _a;
982
+ const searchService = new MockSearchService();
983
+ const el = await fixture(html `<collection-browser
984
+ .searchService=${searchService}
985
+ .baseNavigationUrl=${''}
986
+ ></collection-browser>`);
987
+ el.withinCollection = 'default-sort';
988
+ el.baseQuery = 'default-sort';
989
+ await el.updateComplete;
990
+ await el.initialSearchComplete;
991
+ await el.updateComplete;
992
+ await aTimeout(50);
993
+ const sortBar = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('sort-filter-bar');
994
+ expect(sortBar).to.exist;
995
+ expect(sortBar.defaultSortField).to.equal(SortField.relevance);
996
+ expect(sortBar.defaultSortDirection).to.be.null;
997
+ expect(sortBar.selectedSort).to.equal(SortField.default);
998
+ expect(sortBar.sortDirection).to.be.null;
999
+ });
1000
+ it('uses date favorited sort as default when targeting fav- collection', async () => {
1001
+ var _a;
1002
+ const searchService = new MockSearchService();
1003
+ const el = await fixture(html `<collection-browser
1004
+ .searchService=${searchService}
1005
+ .baseNavigationUrl=${''}
1006
+ ></collection-browser>`);
1007
+ el.withinCollection = 'fav-sort';
1008
+ await el.updateComplete;
1009
+ await el.initialSearchComplete;
1010
+ await el.updateComplete;
1011
+ await aTimeout(50);
1012
+ const sortBar = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('sort-filter-bar');
1013
+ expect(sortBar).to.exist;
1014
+ expect(sortBar.defaultSortField).to.equal(SortField.datefavorited);
1015
+ expect(sortBar.defaultSortDirection).to.equal('desc');
1016
+ expect(sortBar.selectedSort).to.equal(SortField.default);
1017
+ expect(sortBar.sortDirection).to.be.null;
1018
+ });
1019
+ it('scrolls to page', async () => {
1020
+ var _a;
1021
+ const searchService = new MockSearchService();
1022
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
1023
+ </collection-browser>`);
1024
+ // Infinite scroller won't exist unless there's a base query.
1025
+ // First ensure that we don't throw errors when it doesn't exist.
1026
+ await el.goToPage(1);
1027
+ // And make sure it correctly calls scrollToCell when the
1028
+ // infinite scroller does exist.
1029
+ el.baseQuery = 'collection:foo';
1030
+ await el.updateComplete;
1031
+ await nextTick();
1032
+ const infiniteScroller = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('infinite-scroller');
1033
+ expect(infiniteScroller).to.exist;
1034
+ const oldScrollToCell = infiniteScroller.scrollToCell;
1035
+ const spy = sinon.spy();
1036
+ infiniteScroller.scrollToCell = spy;
1037
+ await el.goToPage(1);
1038
+ expect(spy.callCount, 'scroll to page fires once').to.equal(1);
1039
+ infiniteScroller.scrollToCell = oldScrollToCell;
1040
+ });
1041
+ it('shows mobile facets in mobile view', async () => {
1042
+ var _a, _b;
1043
+ const searchService = new MockSearchService();
1044
+ const el = await fixture(html `<collection-browser
1045
+ .searchService=${searchService}
1046
+ .mobileBreakpoint=${9999}
1047
+ ></collection-browser>`);
1048
+ el.baseQuery = 'collection:foo';
1049
+ await el.updateComplete;
1050
+ const contentContainer = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('#content-container');
1051
+ el.handleResize({
1052
+ target: contentContainer,
1053
+ contentRect: contentContainer.getBoundingClientRect(),
1054
+ borderBoxSize: [],
1055
+ contentBoxSize: [],
1056
+ devicePixelContentBoxSize: [],
1057
+ });
1058
+ await el.updateComplete;
1059
+ const mobileFacets = (_b = el.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('#mobile-filter-collapse');
1060
+ expect(mobileFacets).to.exist;
1061
+ });
1062
+ it('fires analytics when mobile facets toggled', async () => {
1063
+ var _a, _b;
1064
+ const searchService = new MockSearchService();
1065
+ const analyticsHandler = new MockAnalyticsHandler();
1066
+ const el = await fixture(html `<collection-browser
1067
+ .searchService=${searchService}
1068
+ .analyticsHandler=${analyticsHandler}
1069
+ .searchContext=${'foobar-context'}
1070
+ .mobileBreakpoint=${9999}
1071
+ ></collection-browser>`);
1072
+ el.baseQuery = 'collection:foo';
1073
+ await el.updateComplete;
1074
+ const contentContainer = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('#content-container');
1075
+ el.handleResize({
1076
+ target: contentContainer,
1077
+ contentRect: contentContainer.getBoundingClientRect(),
1078
+ borderBoxSize: [],
1079
+ contentBoxSize: [],
1080
+ devicePixelContentBoxSize: [],
1081
+ });
1082
+ await el.updateComplete;
1083
+ const mobileFacets = (_b = el.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('#mobile-filter-collapse');
1084
+ expect(mobileFacets).to.exist;
1085
+ // We set up a Promise to wait for the 'toggle' event on the collapser,
1086
+ // which is what triggers the analytics.
1087
+ let facetsToggled = new Promise(resolve => {
1088
+ mobileFacets.addEventListener('toggle', resolve);
1089
+ });
1090
+ // Open the mobile facets accordion & check analytics
1091
+ const mobileFacetsHeader = mobileFacets.querySelector('summary');
1092
+ expect(mobileFacetsHeader).to.exist;
1093
+ mobileFacetsHeader.click();
1094
+ await facetsToggled;
1095
+ expect(analyticsHandler.callCategory).to.equal('foobar-context');
1096
+ expect(analyticsHandler.callAction).to.equal(analyticsActions.mobileFacetsToggled);
1097
+ expect(analyticsHandler.callLabel).to.equal('open');
1098
+ // Close the mobile facets accordion & check analytics
1099
+ facetsToggled = new Promise(resolve => {
1100
+ mobileFacets.addEventListener('toggle', resolve);
1101
+ });
1102
+ mobileFacetsHeader.click();
1103
+ await facetsToggled;
1104
+ expect(analyticsHandler.callCategory).to.equal('foobar-context');
1105
+ expect(analyticsHandler.callAction).to.equal(analyticsActions.mobileFacetsToggled);
1106
+ expect(analyticsHandler.callLabel).to.equal('closed');
1107
+ });
1108
+ it('sets parent collections to prop when searching a collection', async () => {
1109
+ const searchService = new MockSearchService();
1110
+ const el = await fixture(html `<collection-browser
1111
+ .searchService=${searchService}
1112
+ .withinCollection=${'fake'}
1113
+ ></collection-browser>`);
1114
+ el.baseQuery = 'parent-collections';
1115
+ await el.updateComplete;
1116
+ await el.initialSearchComplete;
1117
+ await aTimeout(0);
1118
+ expect(el.dataSource.parentCollections).to.deep.equal(['foo', 'bar']);
1119
+ });
1120
+ it('recognizes TV collections', async () => {
1121
+ const searchService = new MockSearchService();
1122
+ const el = await fixture(html `<collection-browser
1123
+ .searchService=${searchService}
1124
+ .withinCollection=${'TV-FOO'}
1125
+ ></collection-browser>`);
1126
+ el.baseQuery = 'tv-collection';
1127
+ await el.updateComplete;
1128
+ await el.initialSearchComplete;
1129
+ await aTimeout(0);
1130
+ expect(el.isTVCollection).to.be.true;
1131
+ });
1132
+ it('refreshes when certain properties change - with some analytics event sampling', async () => {
1133
+ var _a;
1134
+ const mockAnalyticsHandler = new MockAnalyticsHandler();
1135
+ const searchService = new MockSearchService();
1136
+ const el = await fixture(html `<collection-browser
1137
+ .analyticsHandler=${mockAnalyticsHandler}
1138
+ .searchService=${searchService}
1139
+ ></collection-browser>`);
1140
+ const infiniteScrollerRefreshSpy = sinon.spy();
1141
+ // Infinite scroller won't exist unless there's a base query
1142
+ el.baseQuery = 'collection:foo';
1143
+ await el.updateComplete;
1144
+ await nextTick();
1145
+ const infiniteScroller = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('infinite-scroller');
1146
+ infiniteScroller.reload = infiniteScrollerRefreshSpy;
1147
+ expect(infiniteScrollerRefreshSpy.called).to.be.false;
1148
+ expect(infiniteScrollerRefreshSpy.callCount).to.equal(0);
1149
+ // testing: `loggedIn`
1150
+ el.loggedIn = true;
1151
+ await el.updateComplete;
1152
+ expect(infiniteScrollerRefreshSpy.called, 'Infinite Scroller Refresh').to.be
1153
+ .true;
1154
+ expect(infiniteScrollerRefreshSpy.callCount, 'Infinite Scroller Refresh call count').to.equal(1);
1155
+ el.loggedIn = false;
1156
+ await el.updateComplete;
1157
+ expect(infiniteScrollerRefreshSpy.callCount, '2nd Infinite Scroller Refresh').to.equal(2);
1158
+ // testing: `displayMode`
1159
+ el.displayMode = 'list-compact';
1160
+ el.searchContext = 'beta-search';
1161
+ await el.updateComplete;
1162
+ expect(infiniteScrollerRefreshSpy.callCount, '3rd Infinite Scroller Refresh').to.equal(3);
1163
+ expect(mockAnalyticsHandler.callCategory).to.equal('beta-search');
1164
+ expect(mockAnalyticsHandler.callAction).to.equal('displayMode');
1165
+ expect(mockAnalyticsHandler.callLabel).to.equal('list-compact');
1166
+ el.displayMode = 'list-detail';
1167
+ await el.updateComplete;
1168
+ expect(infiniteScrollerRefreshSpy.callCount, '4th Infinite Scroller Refresh').to.equal(4);
1169
+ expect(mockAnalyticsHandler.callCategory).to.equal('beta-search');
1170
+ expect(mockAnalyticsHandler.callAction).to.equal('displayMode');
1171
+ expect(mockAnalyticsHandler.callLabel).to.equal('list-detail');
1172
+ // testing: `baseNavigationUrl`
1173
+ el.baseNavigationUrl = 'https://funtestsite.com';
1174
+ await el.updateComplete;
1175
+ expect(infiniteScrollerRefreshSpy.callCount, '5th Infinite Scroller Refresh').to.equal(5);
1176
+ // testing: `baseImageUrl`
1177
+ el.baseImageUrl = 'https://funtestsiteforimages.com';
1178
+ await el.updateComplete;
1179
+ expect(infiniteScrollerRefreshSpy.callCount, '6th Infinite Scroller Refresh').to.equal(6);
1180
+ });
1181
+ it('query the search service for single result', async () => {
1182
+ var _a, _b;
1183
+ const searchService = new MockSearchService();
1184
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
1185
+ </collection-browser>`);
1186
+ el.baseQuery = 'single-result';
1187
+ await el.updateComplete;
1188
+ await el.initialSearchComplete;
1189
+ expect((_b = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('#big-results-label')) === null || _b === void 0 ? void 0 : _b.textContent).to.contains('Result');
1190
+ });
1191
+ it('`searchContext` prop helps describe where component is being used', async () => {
1192
+ const el = await fixture(html `<collection-browser></collection-browser>`);
1193
+ expect(el.searchContext).to.equal(analyticsCategories.default);
1194
+ el.searchContext = 'unicorn-search';
1195
+ await el.updateComplete;
1196
+ expect(el.searchContext).to.equal('unicorn-search');
1197
+ // property is reflected as attribute
1198
+ expect(el.getAttribute('searchcontext')).to.equal('unicorn-search');
1199
+ });
1200
+ it('respects the initial set of URL parameters for a general search', async () => {
1201
+ var _a, _b, _c, _d, _e, _f;
1202
+ const url = new URL(window.location.href);
1203
+ const { searchParams } = url;
1204
+ searchParams.set('query', 'foo');
1205
+ searchParams.set('sin', 'TXT');
1206
+ searchParams.set('sort', 'title');
1207
+ searchParams.append('not[]', 'mediatype:"data"');
1208
+ searchParams.append('and[]', 'subject:"baz"');
1209
+ searchParams.append('and[]', 'firstTitle:X');
1210
+ searchParams.append('and[]', 'year:[2000 TO 2010]');
1211
+ window.history.replaceState({}, '', url);
1212
+ const searchService = new MockSearchService();
1213
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
1214
+ </collection-browser>`);
1215
+ await el.initialSearchComplete;
1216
+ await el.updateComplete;
1217
+ expect(el.baseQuery).to.equal('foo');
1218
+ expect(el.searchType).to.equal(SearchType.FULLTEXT);
1219
+ expect(el.selectedSort).to.equal(SortField.title);
1220
+ expect((_c = (_b = (_a = el.selectedFacets) === null || _a === void 0 ? void 0 : _a.mediatype) === null || _b === void 0 ? void 0 : _b.data) === null || _c === void 0 ? void 0 : _c.state).to.equal('hidden');
1221
+ expect((_f = (_e = (_d = el.selectedFacets) === null || _d === void 0 ? void 0 : _d.subject) === null || _e === void 0 ? void 0 : _e.baz) === null || _f === void 0 ? void 0 : _f.state).to.equal('selected');
1222
+ expect(el.selectedTitleFilter).to.equal('X');
1223
+ expect(el.minSelectedDate).to.equal('2000');
1224
+ expect(el.maxSelectedDate).to.equal('2010');
1225
+ });
1226
+ it('respects the initial set of URL parameters within a collection', async () => {
1227
+ var _a, _b, _c, _d, _e, _f;
1228
+ const url = new URL(window.location.href);
1229
+ const { searchParams } = url;
1230
+ searchParams.set('query', 'foo');
1231
+ searchParams.set('sin', 'TXT');
1232
+ searchParams.set('sort', 'title');
1233
+ searchParams.append('not[]', 'mediatype:"data"');
1234
+ searchParams.append('and[]', 'subject:"baz"');
1235
+ searchParams.append('and[]', 'firstTitle:X');
1236
+ searchParams.append('and[]', 'year:[2000 TO 2010]');
1237
+ window.history.replaceState({}, '', url);
1238
+ const searchService = new MockSearchService();
1239
+ const el = await fixture(html `<collection-browser
1240
+ .searchService=${searchService}
1241
+ .withinCollection=${'foobar'}
1242
+ >
1243
+ </collection-browser>`);
1244
+ await el.initialSearchComplete;
1245
+ await el.updateComplete;
1246
+ expect(el.withinCollection).to.equal('foobar');
1247
+ expect(el.baseQuery).to.equal('foo');
1248
+ expect(el.searchType).to.equal(SearchType.FULLTEXT);
1249
+ expect(el.selectedSort).to.equal(SortField.title);
1250
+ expect((_c = (_b = (_a = el.selectedFacets) === null || _a === void 0 ? void 0 : _a.mediatype) === null || _b === void 0 ? void 0 : _b.data) === null || _c === void 0 ? void 0 : _c.state).to.equal('hidden');
1251
+ expect((_f = (_e = (_d = el.selectedFacets) === null || _d === void 0 ? void 0 : _d.subject) === null || _e === void 0 ? void 0 : _e.baz) === null || _f === void 0 ? void 0 : _f.state).to.equal('selected');
1252
+ expect(el.selectedTitleFilter).to.equal('X');
1253
+ expect(el.minSelectedDate).to.equal('2000');
1254
+ expect(el.maxSelectedDate).to.equal('2010');
1255
+ });
1256
+ it('respects the initial set of URL parameters within a profile page', async () => {
1257
+ var _a, _b, _c, _d, _e, _f;
1258
+ const url = new URL(window.location.href);
1259
+ const { searchParams } = url;
1260
+ searchParams.set('query', 'foo');
1261
+ searchParams.append('not[]', 'mediatype:"data"');
1262
+ searchParams.append('and[]', 'subject:"baz"');
1263
+ searchParams.append('and[]', 'firstTitle:X');
1264
+ searchParams.append('and[]', 'year:[2000 TO 2010]');
1265
+ window.history.replaceState({}, '', url);
1266
+ const searchService = new MockSearchService();
1267
+ const el = await fixture(html `<collection-browser
1268
+ .searchService=${searchService}
1269
+ .withinProfile=${'@foobar'}
1270
+ .profileElement=${'uploads'}
1271
+ >
1272
+ </collection-browser>`);
1273
+ await el.initialSearchComplete;
1274
+ await el.updateComplete;
1275
+ expect(el.withinProfile).to.equal('@foobar');
1276
+ expect(el.profileElement).to.equal('uploads');
1277
+ expect(el.baseQuery).to.equal('foo');
1278
+ expect(el.searchType).to.equal(SearchType.DEFAULT);
1279
+ expect((_c = (_b = (_a = el.selectedFacets) === null || _a === void 0 ? void 0 : _a.mediatype) === null || _b === void 0 ? void 0 : _b.data) === null || _c === void 0 ? void 0 : _c.state).to.equal('hidden');
1280
+ expect((_f = (_e = (_d = el.selectedFacets) === null || _d === void 0 ? void 0 : _d.subject) === null || _e === void 0 ? void 0 : _e.baz) === null || _f === void 0 ? void 0 : _f.state).to.equal('selected');
1281
+ expect(el.selectedTitleFilter).to.equal('X');
1282
+ expect(el.minSelectedDate).to.equal('2000');
1283
+ expect(el.maxSelectedDate).to.equal('2010');
1284
+ });
1285
+ it('clears filters except sort when query changes for a general search', async () => {
1286
+ var _a, _b, _c, _d;
1287
+ const url = new URL(window.location.href);
1288
+ const { searchParams } = url;
1289
+ searchParams.set('query', 'foo');
1290
+ searchParams.set('sin', 'TXT');
1291
+ searchParams.set('sort', 'title');
1292
+ searchParams.append('not[]', 'mediatype:"data"');
1293
+ searchParams.append('and[]', 'subject:"baz"');
1294
+ searchParams.append('and[]', 'firstTitle:X');
1295
+ searchParams.append('and[]', 'year:[2000 TO 2010]');
1296
+ window.history.replaceState({}, '', url);
1297
+ const searchService = new MockSearchService();
1298
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
1299
+ </collection-browser>`);
1300
+ await el.initialSearchComplete;
1301
+ await el.updateComplete;
1302
+ el.baseQuery = 'bar';
1303
+ await el.updateComplete;
1304
+ expect(el.baseQuery).to.equal('bar');
1305
+ expect(el.searchType).to.equal(SearchType.FULLTEXT);
1306
+ expect(el.selectedSort).to.equal(SortField.title);
1307
+ expect((_b = (_a = el.selectedFacets) === null || _a === void 0 ? void 0 : _a.mediatype) === null || _b === void 0 ? void 0 : _b.data).not.to.exist;
1308
+ expect((_d = (_c = el.selectedFacets) === null || _c === void 0 ? void 0 : _c.subject) === null || _d === void 0 ? void 0 : _d.baz).not.to.exist;
1309
+ expect(el.selectedTitleFilter).not.to.exist;
1310
+ expect(el.minSelectedDate).not.to.exist;
1311
+ expect(el.maxSelectedDate).not.to.exist;
1312
+ });
1313
+ it('clears filters except sort when query changes within a collection', async () => {
1314
+ var _a, _b, _c, _d;
1315
+ const url = new URL(window.location.href);
1316
+ const { searchParams } = url;
1317
+ searchParams.set('query', 'foo');
1318
+ searchParams.set('sin', 'TXT');
1319
+ searchParams.set('sort', 'title');
1320
+ searchParams.append('not[]', 'mediatype:"data"');
1321
+ searchParams.append('and[]', 'subject:"baz"');
1322
+ searchParams.append('and[]', 'firstTitle:X');
1323
+ searchParams.append('and[]', 'year:[2000 TO 2010]');
1324
+ window.history.replaceState({}, '', url);
1325
+ const searchService = new MockSearchService();
1326
+ const el = await fixture(html `<collection-browser
1327
+ .searchService=${searchService}
1328
+ .withinCollection=${'foobar'}
1329
+ >
1330
+ </collection-browser>`);
1331
+ el.baseQuery = 'bar';
1332
+ await el.updateComplete;
1333
+ expect(el.withinCollection).to.equal('foobar');
1334
+ expect(el.baseQuery).to.equal('bar');
1335
+ expect(el.searchType).to.equal(SearchType.FULLTEXT);
1336
+ expect(el.selectedSort).to.equal(SortField.title);
1337
+ expect((_b = (_a = el.selectedFacets) === null || _a === void 0 ? void 0 : _a.mediatype) === null || _b === void 0 ? void 0 : _b.data).not.to.exist;
1338
+ expect((_d = (_c = el.selectedFacets) === null || _c === void 0 ? void 0 : _c.subject) === null || _d === void 0 ? void 0 : _d.baz).not.to.exist;
1339
+ expect(el.selectedTitleFilter).not.to.exist;
1340
+ expect(el.minSelectedDate).not.to.exist;
1341
+ expect(el.maxSelectedDate).not.to.exist;
1342
+ });
1343
+ it('clears filters *including* sort when target collection changes', async () => {
1344
+ var _a, _b, _c, _d;
1345
+ const url = new URL(window.location.href);
1346
+ const { searchParams } = url;
1347
+ searchParams.set('query', 'foo');
1348
+ searchParams.set('sin', 'TXT');
1349
+ searchParams.set('sort', 'title');
1350
+ searchParams.append('not[]', 'mediatype:"data"');
1351
+ searchParams.append('and[]', 'subject:"baz"');
1352
+ searchParams.append('and[]', 'firstTitle:X');
1353
+ searchParams.append('and[]', 'year:[2000 TO 2010]');
1354
+ window.history.replaceState({}, '', url);
1355
+ const searchService = new MockSearchService();
1356
+ const el = await fixture(html `<collection-browser
1357
+ .searchService=${searchService}
1358
+ .withinCollection=${'foobar'}
1359
+ >
1360
+ </collection-browser>`);
1361
+ el.withinCollection = 'bar';
1362
+ await el.updateComplete;
1363
+ expect(el.withinCollection).to.equal('bar');
1364
+ expect(el.baseQuery).to.equal('foo');
1365
+ expect(el.searchType).to.equal(SearchType.FULLTEXT);
1366
+ expect(el.selectedSort).to.equal(SortField.default);
1367
+ expect((_b = (_a = el.selectedFacets) === null || _a === void 0 ? void 0 : _a.mediatype) === null || _b === void 0 ? void 0 : _b.data).not.to.exist;
1368
+ expect((_d = (_c = el.selectedFacets) === null || _c === void 0 ? void 0 : _c.subject) === null || _d === void 0 ? void 0 : _d.baz).not.to.exist;
1369
+ expect(el.selectedTitleFilter).not.to.exist;
1370
+ expect(el.minSelectedDate).not.to.exist;
1371
+ expect(el.maxSelectedDate).not.to.exist;
1372
+ });
1373
+ it('correctly retrieves web archive hits', async () => {
1374
+ var _a, _b, _c;
1375
+ const searchService = new MockSearchService();
1376
+ const el = await fixture(html `<collection-browser
1377
+ .searchService=${searchService}
1378
+ .withinProfile=${'@foo'}
1379
+ .profileElement=${'web_archives'}
1380
+ >
1381
+ </collection-browser>`);
1382
+ el.baseQuery = 'web-archive';
1383
+ await el.updateComplete;
1384
+ await el.initialSearchComplete;
1385
+ await nextTick();
1386
+ console.log('\n\n*****\n\n*****\n\n', el.dataSource.getAllPages(), '\n\n*****\n\n*****\n\n');
1387
+ expect(el.dataSource.totalResults, 'total results').to.equal(1);
1388
+ expect((_a = el.dataSource.getTileModelAt(0)) === null || _a === void 0 ? void 0 : _a.title).to.equal('https://example.com');
1389
+ expect((_c = (_b = el.dataSource.getTileModelAt(0)) === null || _b === void 0 ? void 0 : _b.captureDates) === null || _c === void 0 ? void 0 : _c.length, 'capture dates').to.equal(1);
1390
+ });
1391
+ it('shows dropdown accordion in facet sidebar when opt-in strategy is specified', async () => {
1392
+ var _a;
1393
+ const searchService = new MockSearchService();
1394
+ const el = await fixture(html `<collection-browser
1395
+ .searchService=${searchService}
1396
+ facetLoadStrategy=${'opt-in'}
1397
+ >
1398
+ </collection-browser>`);
1399
+ el.baseQuery = 'foo';
1400
+ await el.updateComplete;
1401
+ await el.initialSearchComplete;
1402
+ const facetsDropdown = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('.desktop-facets-dropdown');
1403
+ expect(facetsDropdown).to.exist;
1404
+ });
1405
+ it('shows temporarily unavailable message when facets suppressed', async () => {
1406
+ var _a, _b;
1407
+ const searchService = new MockSearchService();
1408
+ const el = await fixture(html `<collection-browser
1409
+ .searchService=${searchService}
1410
+ facetLoadStrategy=${'off'}
1411
+ >
1412
+ </collection-browser>`);
1413
+ el.baseQuery = 'foo';
1414
+ await el.updateComplete;
1415
+ await el.initialSearchComplete;
1416
+ const facetsMsg = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('.facets-message');
1417
+ expect(facetsMsg).to.exist;
1418
+ expect((_b = facetsMsg === null || facetsMsg === void 0 ? void 0 : facetsMsg.textContent) === null || _b === void 0 ? void 0 : _b.trim()).to.equal('Facets are temporarily unavailable.');
1419
+ });
1420
+ it('shows manage bar interface instead of sort bar when in manage view', async () => {
1421
+ var _a, _b, _c, _d;
1422
+ const searchService = new MockSearchService();
1423
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
1424
+ </collection-browser>`);
1425
+ el.baseQuery = 'foo';
1426
+ await el.updateComplete;
1427
+ await el.initialSearchComplete;
1428
+ el.isManageView = true;
1429
+ await el.updateComplete;
1430
+ expect((_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('manage-bar')).to.exist;
1431
+ expect((_b = el.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('sort-filter-bar')).not.to.exist;
1432
+ el.isManageView = false;
1433
+ await el.updateComplete;
1434
+ expect((_c = el.shadowRoot) === null || _c === void 0 ? void 0 : _c.querySelector('manage-bar')).not.to.exist;
1435
+ expect((_d = el.shadowRoot) === null || _d === void 0 ? void 0 : _d.querySelector('sort-filter-bar')).to.exist;
1436
+ });
1437
+ it('switches to grid display mode when manage view activated', async () => {
1438
+ const searchService = new MockSearchService();
1439
+ const el = await fixture(html `<collection-browser
1440
+ .searchService=${searchService}
1441
+ .baseQuery=${'foo'}
1442
+ .displayMode=${'list-detail'}
1443
+ >
1444
+ </collection-browser>`);
1445
+ el.isManageView = true;
1446
+ await el.updateComplete;
1447
+ expect(el.displayMode).to.equal('grid');
1448
+ });
1449
+ it('can remove all checked tiles', async () => {
1450
+ var _a, _b, _c, _d, _e, _f, _g, _h, _j;
1451
+ const searchService = new MockSearchService();
1452
+ const el = await fixture(html `<collection-browser
1453
+ .searchService=${searchService}
1454
+ .baseNavigationUrl=${''}
1455
+ >
1456
+ </collection-browser>`);
1457
+ el.baseQuery = 'foo';
1458
+ el.pageSize = 1; // To hit the edge case of a page break while offsetting tiles
1459
+ await el.updateComplete;
1460
+ await el.initialSearchComplete;
1461
+ el.isManageView = true;
1462
+ await el.updateComplete;
1463
+ const infiniteScroller = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('infinite-scroller');
1464
+ expect(infiniteScroller).to.exist;
1465
+ let tiles = (_b = infiniteScroller.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelectorAll('tile-dispatcher');
1466
+ expect(tiles).to.exist;
1467
+ expect(tiles === null || tiles === void 0 ? void 0 : tiles.length).to.equal(2);
1468
+ const firstTile = tiles[0];
1469
+ const firstTileLink = (_c = firstTile.shadowRoot) === null || _c === void 0 ? void 0 : _c.querySelector('a[href]');
1470
+ expect((_d = firstTile.model) === null || _d === void 0 ? void 0 : _d.identifier).to.equal('foo');
1471
+ expect(firstTileLink).to.exist;
1472
+ // No effect if no tiles checked
1473
+ el.removeCheckedTiles();
1474
+ await el.updateComplete;
1475
+ tiles = (_e = infiniteScroller.shadowRoot) === null || _e === void 0 ? void 0 : _e.querySelectorAll('tile-dispatcher');
1476
+ expect(tiles).to.exist;
1477
+ expect(tiles === null || tiles === void 0 ? void 0 : tiles.length).to.equal(2);
1478
+ // Check the first tile
1479
+ firstTileLink.click();
1480
+ expect((_f = firstTile.model) === null || _f === void 0 ? void 0 : _f.checked).to.be.true;
1481
+ // Remove checked tiles and verify that we only kept the second tile
1482
+ el.removeCheckedTiles();
1483
+ await el.updateComplete;
1484
+ expect((_g = el === null || el === void 0 ? void 0 : el.dataSource) === null || _g === void 0 ? void 0 : _g.size, 'data source count').to.equal(1);
1485
+ tiles = (_j = (_h = el.shadowRoot) === null || _h === void 0 ? void 0 : _h.querySelector('infinite-scroller').shadowRoot) === null || _j === void 0 ? void 0 : _j.querySelectorAll('tile-dispatcher');
1486
+ expect(tiles).to.exist;
1487
+ expect(tiles.length, 'tile count after `el.removeCheckedTiles()`').to.equal(1);
1488
+ });
1489
+ it('can check/uncheck all tiles', async () => {
1490
+ const searchService = new MockSearchService();
1491
+ const el = await fixture(html `<collection-browser
1492
+ .searchService=${searchService}
1493
+ .baseNavigationUrl=${''}
1494
+ >
1495
+ </collection-browser>`);
1496
+ el.baseQuery = 'foo';
1497
+ await el.updateComplete;
1498
+ await el.initialSearchComplete;
1499
+ el.isManageView = true;
1500
+ await el.updateComplete;
1501
+ expect(el.dataSource.checkedTileModels.length).to.equal(0);
1502
+ expect(el.dataSource.uncheckedTileModels.length).to.equal(2);
1503
+ el.dataSource.checkAllTiles();
1504
+ expect(el.dataSource.checkedTileModels.length).to.equal(2);
1505
+ expect(el.dataSource.uncheckedTileModels.length).to.equal(0);
1506
+ el.dataSource.uncheckAllTiles();
1507
+ expect(el.dataSource.checkedTileModels.length).to.equal(0);
1508
+ expect(el.dataSource.uncheckedTileModels.length).to.equal(2);
1509
+ });
1510
+ it('emits event when manage view state changes', async () => {
1511
+ var _a, _b;
1512
+ const spy = sinon.spy();
1513
+ const searchService = new MockSearchService();
1514
+ const el = await fixture(html `<collection-browser
1515
+ .searchService=${searchService}
1516
+ .baseNavigationUrl=${''}
1517
+ @manageModeChanged=${spy}
1518
+ ></collection-browser>`);
1519
+ el.isManageView = true;
1520
+ await el.updateComplete;
1521
+ expect(spy.callCount).to.equal(1);
1522
+ expect((_a = spy.args[0][0]) === null || _a === void 0 ? void 0 : _a.detail).to.be.true;
1523
+ el.isManageView = false;
1524
+ await el.updateComplete;
1525
+ expect(spy.callCount).to.equal(2);
1526
+ expect((_b = spy.args[1][0]) === null || _b === void 0 ? void 0 : _b.detail).to.be.false;
1527
+ });
1528
+ it('emits event when item removal requested', async () => {
1529
+ var _a, _b, _c, _d, _e;
1530
+ const spy = sinon.spy();
1531
+ const searchService = new MockSearchService();
1532
+ const el = await fixture(html `<collection-browser
1533
+ .searchService=${searchService}
1534
+ .baseNavigationUrl=${''}
1535
+ @itemRemovalRequested=${spy}
1536
+ >
1537
+ </collection-browser>`);
1538
+ el.baseQuery = 'foo';
1539
+ await el.updateComplete;
1540
+ await el.initialSearchComplete;
1541
+ el.isManageView = true;
1542
+ await el.updateComplete;
1543
+ const infiniteScroller = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('infinite-scroller');
1544
+ expect(infiniteScroller).to.exist;
1545
+ const tiles = (_b = infiniteScroller.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelectorAll('tile-dispatcher');
1546
+ expect(tiles).to.exist.and.have.length(2);
1547
+ const firstTile = tiles[0];
1548
+ const firstTileLink = (_c = firstTile.shadowRoot) === null || _c === void 0 ? void 0 : _c.querySelector('a[href]');
1549
+ expect((_d = firstTile.model) === null || _d === void 0 ? void 0 : _d.identifier).to.equal('foo');
1550
+ expect(firstTileLink).to.exist;
1551
+ // Check the first tile
1552
+ firstTileLink.click();
1553
+ await el.updateComplete;
1554
+ const manageBar = (_e = el.shadowRoot) === null || _e === void 0 ? void 0 : _e.querySelector('manage-bar');
1555
+ expect(manageBar).to.exist;
1556
+ // Emit remove event from manage bar
1557
+ manageBar.dispatchEvent(new CustomEvent('removeItems'));
1558
+ await el.updateComplete;
1559
+ expect(spy.callCount).to.equal(1);
1560
+ expect(spy.args[0].length).to.equal(1);
1561
+ expect(spy.args[0][0].detail.items[0]).to.equal('foo');
1562
+ });
1563
+ it('disables manage view when manage bar cancelled', async () => {
1564
+ var _a;
1565
+ const searchService = new MockSearchService();
1566
+ const el = await fixture(html `<collection-browser
1567
+ .searchService=${searchService}
1568
+ .baseNavigationUrl=${''}
1569
+ >
1570
+ </collection-browser>`);
1571
+ el.baseQuery = 'foo';
1572
+ await el.updateComplete;
1573
+ await el.initialSearchComplete;
1574
+ el.isManageView = true;
1575
+ await el.updateComplete;
1576
+ const manageBar = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('manage-bar');
1577
+ expect(manageBar).to.exist;
1578
+ // Emit remove event from manage bar
1579
+ manageBar.dispatchEvent(new CustomEvent('cancel'));
1580
+ await el.updateComplete;
1581
+ expect(el.isManageView).to.be.false;
1582
+ });
1583
+ it('enable/disable manage view delete button when you selectAll/unselectAll', async () => {
1584
+ var _a, _b, _c, _d;
1585
+ const searchService = new MockSearchService();
1586
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
1587
+ </collection-browser>`);
1588
+ el.baseQuery = 'foo';
1589
+ await el.updateComplete;
1590
+ await el.initialSearchComplete;
1591
+ el.isManageView = true;
1592
+ await el.updateComplete;
1593
+ const manageBar = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('manage-bar');
1594
+ expect(manageBar).to.exist;
1595
+ // disable button exists
1596
+ expect((_b = manageBar === null || manageBar === void 0 ? void 0 : manageBar.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('.danger:disabled')).to.be
1597
+ .exist;
1598
+ // Emit remove event from manage bar
1599
+ manageBar.dispatchEvent(new CustomEvent('selectAll'));
1600
+ await el.updateComplete;
1601
+ // disable button does not exists
1602
+ expect((_c = manageBar === null || manageBar === void 0 ? void 0 : manageBar.shadowRoot) === null || _c === void 0 ? void 0 : _c.querySelector('.danger:disabled')).to.be.not
1603
+ .exist;
1604
+ // Emit remove event from manage bar
1605
+ manageBar.dispatchEvent(new CustomEvent('unselectAll'));
1606
+ await el.updateComplete;
1607
+ // disable button exists again
1608
+ expect((_d = manageBar === null || manageBar === void 0 ? void 0 : manageBar.shadowRoot) === null || _d === void 0 ? void 0 : _d.querySelector('.danger:disabled')).to.be
1609
+ .exist;
1610
+ });
1611
+ it('shows Blurring checkbox for admin users', async () => {
1612
+ var _a;
1613
+ const searchService = new MockSearchService();
1614
+ const el = await fixture(html `<collection-browser
1615
+ .baseNavigationUrl=${''}
1616
+ .searchService=${searchService}
1617
+ >
1618
+ </collection-browser>`);
1619
+ el.baseQuery = 'archive-org-user-loggedin';
1620
+ await el.updateComplete;
1621
+ await el.initialSearchComplete;
1622
+ const blurringCheck = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('#tile-blur-check');
1623
+ expect(blurringCheck).to.exist;
1624
+ expect(blurringCheck.checked).to.be.true;
1625
+ });
1626
+ it('unchecks Blurring checkbox for admin users with blurring preference off', async () => {
1627
+ var _a;
1628
+ const searchService = new MockSearchService();
1629
+ const el = await fixture(html `<collection-browser
1630
+ .baseNavigationUrl=${''}
1631
+ .searchService=${searchService}
1632
+ >
1633
+ </collection-browser>`);
1634
+ el.baseQuery = 'archive-org-user-loggedin-noblur';
1635
+ await el.updateComplete;
1636
+ await el.initialSearchComplete;
1637
+ const blurringCheck = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('#tile-blur-check');
1638
+ expect(blurringCheck).to.exist;
1639
+ expect(blurringCheck.checked).to.be.false;
1640
+ });
1641
+ it('toggles blur state when Blurring checkbox is toggled', async () => {
1642
+ var _a, _b, _c;
1643
+ const searchService = new MockSearchService();
1644
+ const el = await fixture(html `<collection-browser
1645
+ .baseNavigationUrl=${''}
1646
+ .searchService=${searchService}
1647
+ >
1648
+ </collection-browser>`);
1649
+ el.baseQuery = 'archive-org-user-loggedin';
1650
+ await el.updateComplete;
1651
+ await el.initialSearchComplete;
1652
+ const blurringCheck = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('#tile-blur-check');
1653
+ expect(blurringCheck).to.exist;
1654
+ blurringCheck.dispatchEvent(new PointerEvent('click'));
1655
+ await el.updateComplete;
1656
+ const infiniteScroller = (_b = el.shadowRoot) === null || _b === void 0 ? void 0 : _b.querySelector('infinite-scroller');
1657
+ const firstTile = (_c = infiniteScroller === null || infiniteScroller === void 0 ? void 0 : infiniteScroller.shadowRoot) === null || _c === void 0 ? void 0 : _c.querySelector('tile-dispatcher');
1658
+ expect(firstTile.suppressBlurring).to.be.true;
1659
+ });
1660
+ it('applies loans tab properties to sort bar', async () => {
1661
+ var _a;
1662
+ const searchService = new MockSearchService();
1663
+ const el = await fixture(html `<collection-browser
1664
+ .baseNavigationUrl=${''}
1665
+ .searchService=${searchService}
1666
+ .enableSortOptionsSlot=${true}
1667
+ >
1668
+ </collection-browser>`);
1669
+ el.baseQuery = 'collection:foo';
1670
+ await el.updateComplete;
1671
+ await aTimeout(10);
1672
+ const sortBar = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('sort-filter-bar');
1673
+ expect(sortBar === null || sortBar === void 0 ? void 0 : sortBar.enableSortOptionsSlot, 'show loans in sort bar').to.be.true;
1674
+ expect(el.enableSortOptionsSlot, 'collection browser is loans tab').to.be
1675
+ .true;
1676
+ const loansTabSlot = sortBar.querySelector('slot[name="sort-options"]');
1677
+ expect(loansTabSlot).to.exist;
1678
+ });
1679
+ it('can suppress presence of result count', async () => {
1680
+ var _a;
1681
+ const searchService = new MockSearchService();
1682
+ const el = await fixture(html `<collection-browser
1683
+ .searchService=${searchService}
1684
+ suppressResultCount
1685
+ ></collection-browser>`);
1686
+ el.baseQuery = 'collection:foo';
1687
+ await el.updateComplete;
1688
+ await el.initialSearchComplete;
1689
+ const resultCount = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('#results-total');
1690
+ expect(resultCount).not.to.exist;
1691
+ });
1692
+ it('can suppress presence of result tiles', async () => {
1693
+ var _a;
1694
+ const searchService = new MockSearchService();
1695
+ const el = await fixture(html `<collection-browser
1696
+ .searchService=${searchService}
1697
+ suppressResultTiles
1698
+ ></collection-browser>`);
1699
+ el.baseQuery = 'collection:foo';
1700
+ await el.updateComplete;
1701
+ const infiniteScroller = (_a = el.shadowRoot) === null || _a === void 0 ? void 0 : _a.querySelector('infinite-scroller');
1702
+ expect(infiniteScroller).not.to.exist;
1703
+ });
1704
+ it('fetch larger result on search page for admin user to manage items', async () => {
1705
+ const resultsSpy = sinon.spy();
1706
+ const searchService = new MockSearchService({
1707
+ asyncResponse: true,
1708
+ resultsSpy,
1709
+ });
1710
+ const el = await fixture(html `<collection-browser .searchService=${searchService}>
1711
+ </collection-browser>`);
1712
+ const numberOfPages = 15;
1713
+ el.baseQuery = 'jack';
1714
+ el.isManageView = true;
1715
+ await el.dataSource.fetchPage(1, numberOfPages);
1716
+ await el.updateComplete;
1717
+ const initialResults = el.dataSource.getAllPages();
1718
+ expect(Object.keys(initialResults).length).to.deep.equal(numberOfPages);
1719
+ });
1720
+ });
1721
+ //# sourceMappingURL=collection-browser.test.js.map