@internetarchive/collection-browser 2.1.4-alpha.3 → 2.1.4-alpha2

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 (307) hide show
  1. package/.editorconfig +29 -29
  2. package/.husky/pre-commit +4 -4
  3. package/LICENSE +661 -661
  4. package/README.md +83 -83
  5. package/dist/index.d.ts +12 -12
  6. package/dist/index.js +12 -12
  7. package/dist/src/app-root.d.ts +84 -88
  8. package/dist/src/app-root.js +454 -487
  9. package/dist/src/app-root.js.map +1 -1
  10. package/dist/src/assets/img/icons/arrow-left.d.ts +2 -2
  11. package/dist/src/assets/img/icons/arrow-left.js +2 -2
  12. package/dist/src/assets/img/icons/arrow-right.d.ts +2 -2
  13. package/dist/src/assets/img/icons/arrow-right.js +2 -2
  14. package/dist/src/assets/img/icons/chevron.d.ts +2 -2
  15. package/dist/src/assets/img/icons/chevron.js +2 -2
  16. package/dist/src/assets/img/icons/contract.d.ts +2 -2
  17. package/dist/src/assets/img/icons/contract.js +2 -2
  18. package/dist/src/assets/img/icons/empty-query.d.ts +2 -2
  19. package/dist/src/assets/img/icons/empty-query.js +2 -2
  20. package/dist/src/assets/img/icons/expand.d.ts +2 -2
  21. package/dist/src/assets/img/icons/expand.js +2 -2
  22. package/dist/src/assets/img/icons/eye-closed.d.ts +2 -2
  23. package/dist/src/assets/img/icons/eye-closed.js +2 -2
  24. package/dist/src/assets/img/icons/eye.d.ts +2 -2
  25. package/dist/src/assets/img/icons/eye.js +2 -2
  26. package/dist/src/assets/img/icons/favorite-filled.d.ts +1 -1
  27. package/dist/src/assets/img/icons/favorite-filled.js +2 -2
  28. package/dist/src/assets/img/icons/favorite-unfilled.d.ts +1 -1
  29. package/dist/src/assets/img/icons/favorite-unfilled.js +2 -2
  30. package/dist/src/assets/img/icons/login-required.d.ts +1 -1
  31. package/dist/src/assets/img/icons/login-required.js +2 -2
  32. package/dist/src/assets/img/icons/mediatype/account.d.ts +1 -1
  33. package/dist/src/assets/img/icons/mediatype/account.js +2 -2
  34. package/dist/src/assets/img/icons/mediatype/audio.d.ts +1 -1
  35. package/dist/src/assets/img/icons/mediatype/audio.js +2 -2
  36. package/dist/src/assets/img/icons/mediatype/collection.d.ts +1 -1
  37. package/dist/src/assets/img/icons/mediatype/collection.js +2 -2
  38. package/dist/src/assets/img/icons/mediatype/data.d.ts +1 -1
  39. package/dist/src/assets/img/icons/mediatype/data.js +2 -2
  40. package/dist/src/assets/img/icons/mediatype/etree.d.ts +1 -1
  41. package/dist/src/assets/img/icons/mediatype/etree.js +2 -2
  42. package/dist/src/assets/img/icons/mediatype/film.d.ts +1 -1
  43. package/dist/src/assets/img/icons/mediatype/film.js +2 -2
  44. package/dist/src/assets/img/icons/mediatype/images.d.ts +1 -1
  45. package/dist/src/assets/img/icons/mediatype/images.js +2 -2
  46. package/dist/src/assets/img/icons/mediatype/radio.d.ts +1 -1
  47. package/dist/src/assets/img/icons/mediatype/radio.js +2 -2
  48. package/dist/src/assets/img/icons/mediatype/search.d.ts +1 -1
  49. package/dist/src/assets/img/icons/mediatype/search.js +2 -2
  50. package/dist/src/assets/img/icons/mediatype/software.d.ts +1 -1
  51. package/dist/src/assets/img/icons/mediatype/software.js +2 -2
  52. package/dist/src/assets/img/icons/mediatype/texts.d.ts +1 -1
  53. package/dist/src/assets/img/icons/mediatype/texts.js +2 -2
  54. package/dist/src/assets/img/icons/mediatype/tv.d.ts +1 -1
  55. package/dist/src/assets/img/icons/mediatype/tv.js +2 -2
  56. package/dist/src/assets/img/icons/mediatype/video.d.ts +1 -1
  57. package/dist/src/assets/img/icons/mediatype/video.js +2 -2
  58. package/dist/src/assets/img/icons/mediatype/web.d.ts +1 -1
  59. package/dist/src/assets/img/icons/mediatype/web.js +2 -2
  60. package/dist/src/assets/img/icons/null-result.d.ts +2 -2
  61. package/dist/src/assets/img/icons/null-result.js +2 -2
  62. package/dist/src/assets/img/icons/restricted.d.ts +1 -1
  63. package/dist/src/assets/img/icons/restricted.js +2 -2
  64. package/dist/src/assets/img/icons/reviews.d.ts +1 -1
  65. package/dist/src/assets/img/icons/reviews.js +2 -2
  66. package/dist/src/assets/img/icons/upload.d.ts +1 -1
  67. package/dist/src/assets/img/icons/upload.js +2 -2
  68. package/dist/src/assets/img/icons/views.d.ts +1 -1
  69. package/dist/src/assets/img/icons/views.js +2 -2
  70. package/dist/src/circular-activity-indicator.d.ts +5 -5
  71. package/dist/src/circular-activity-indicator.js +17 -17
  72. package/dist/src/collection-browser.d.ts +395 -395
  73. package/dist/src/collection-browser.js +1363 -1366
  74. package/dist/src/collection-browser.js.map +1 -1
  75. package/dist/src/collection-facets/facet-row.d.ts +30 -30
  76. package/dist/src/collection-facets/facet-row.js +114 -114
  77. package/dist/src/collection-facets/facet-tombstone-row.d.ts +5 -5
  78. package/dist/src/collection-facets/facet-tombstone-row.js +15 -15
  79. package/dist/src/collection-facets/facets-template.d.ts +17 -17
  80. package/dist/src/collection-facets/facets-template.js +114 -114
  81. package/dist/src/collection-facets/more-facets-content.d.ts +70 -70
  82. package/dist/src/collection-facets/more-facets-content.js +354 -354
  83. package/dist/src/collection-facets/more-facets-pagination.d.ts +36 -36
  84. package/dist/src/collection-facets/more-facets-pagination.js +196 -196
  85. package/dist/src/collection-facets/toggle-switch.d.ts +41 -41
  86. package/dist/src/collection-facets/toggle-switch.js +94 -94
  87. package/dist/src/collection-facets.d.ts +103 -103
  88. package/dist/src/collection-facets.js +509 -509
  89. package/dist/src/data-source/collection-browser-data-source-interface.d.ts +232 -232
  90. package/dist/src/data-source/collection-browser-data-source-interface.js +1 -1
  91. package/dist/src/data-source/collection-browser-data-source.d.ts +360 -360
  92. package/dist/src/data-source/collection-browser-data-source.js +930 -930
  93. package/dist/src/data-source/collection-browser-query-state.d.ts +43 -43
  94. package/dist/src/data-source/collection-browser-query-state.js +1 -1
  95. package/dist/src/data-source/models.d.ts +28 -28
  96. package/dist/src/data-source/models.js +8 -8
  97. package/dist/src/empty-placeholder.d.ts +23 -23
  98. package/dist/src/empty-placeholder.js +74 -74
  99. package/dist/src/expanded-date-picker.d.ts +43 -43
  100. package/dist/src/expanded-date-picker.js +109 -109
  101. package/dist/src/language-code-handler/language-code-handler.d.ts +37 -37
  102. package/dist/src/language-code-handler/language-code-handler.js +26 -26
  103. package/dist/src/language-code-handler/language-code-mapping.d.ts +1 -1
  104. package/dist/src/language-code-handler/language-code-mapping.js +562 -562
  105. package/dist/src/manage/manage-bar.d.ts +30 -26
  106. package/dist/src/manage/manage-bar.js +76 -85
  107. package/dist/src/manage/manage-bar.js.map +1 -1
  108. package/dist/src/mediatype/mediatype-config.d.ts +3 -3
  109. package/dist/src/mediatype/mediatype-config.js +91 -91
  110. package/dist/src/models.d.ts +198 -198
  111. package/dist/src/models.js +381 -381
  112. package/dist/src/restoration-state-handler.d.ts +70 -70
  113. package/dist/src/restoration-state-handler.js +357 -357
  114. package/dist/src/sort-filter-bar/alpha-bar-tooltip.d.ts +6 -6
  115. package/dist/src/sort-filter-bar/alpha-bar-tooltip.js +24 -24
  116. package/dist/src/sort-filter-bar/alpha-bar.d.ts +21 -21
  117. package/dist/src/sort-filter-bar/alpha-bar.js +128 -128
  118. package/dist/src/sort-filter-bar/img/compact.d.ts +1 -1
  119. package/dist/src/sort-filter-bar/img/compact.js +2 -2
  120. package/dist/src/sort-filter-bar/img/list.d.ts +1 -1
  121. package/dist/src/sort-filter-bar/img/list.js +2 -2
  122. package/dist/src/sort-filter-bar/img/sort-toggle-disabled.d.ts +1 -1
  123. package/dist/src/sort-filter-bar/img/sort-toggle-disabled.js +2 -2
  124. package/dist/src/sort-filter-bar/img/sort-toggle-down.d.ts +1 -1
  125. package/dist/src/sort-filter-bar/img/sort-toggle-down.js +2 -2
  126. package/dist/src/sort-filter-bar/img/sort-toggle-up.d.ts +1 -1
  127. package/dist/src/sort-filter-bar/img/sort-toggle-up.js +2 -2
  128. package/dist/src/sort-filter-bar/img/sort-triangle.d.ts +1 -1
  129. package/dist/src/sort-filter-bar/img/sort-triangle.js +2 -2
  130. package/dist/src/sort-filter-bar/img/tile.d.ts +1 -1
  131. package/dist/src/sort-filter-bar/img/tile.js +2 -2
  132. package/dist/src/sort-filter-bar/sort-filter-bar.d.ts +219 -221
  133. package/dist/src/sort-filter-bar/sort-filter-bar.js +685 -695
  134. package/dist/src/sort-filter-bar/sort-filter-bar.js.map +1 -1
  135. package/dist/src/styles/ia-button.d.ts +2 -0
  136. package/dist/src/styles/ia-button.js +102 -0
  137. package/dist/src/styles/ia-button.js.map +1 -0
  138. package/dist/src/styles/item-image-styles.d.ts +8 -8
  139. package/dist/src/styles/item-image-styles.js +9 -9
  140. package/dist/src/styles/sr-only.d.ts +1 -1
  141. package/dist/src/styles/sr-only.js +2 -2
  142. package/dist/src/tiles/base-tile-component.d.ts +19 -19
  143. package/dist/src/tiles/base-tile-component.js +63 -63
  144. package/dist/src/tiles/collection-browser-loading-tile.d.ts +5 -5
  145. package/dist/src/tiles/collection-browser-loading-tile.js +15 -15
  146. package/dist/src/tiles/grid/account-tile.d.ts +18 -18
  147. package/dist/src/tiles/grid/account-tile.js +72 -72
  148. package/dist/src/tiles/grid/collection-tile.d.ts +15 -15
  149. package/dist/src/tiles/grid/collection-tile.js +80 -80
  150. package/dist/src/tiles/grid/item-tile.d.ts +30 -30
  151. package/dist/src/tiles/grid/item-tile.js +149 -149
  152. package/dist/src/tiles/grid/search-tile.d.ts +10 -10
  153. package/dist/src/tiles/grid/search-tile.js +51 -51
  154. package/dist/src/tiles/grid/styles/tile-grid-shared-styles.d.ts +1 -1
  155. package/dist/src/tiles/grid/styles/tile-grid-shared-styles.js +8 -8
  156. package/dist/src/tiles/grid/tile-stats.d.ts +11 -11
  157. package/dist/src/tiles/grid/tile-stats.js +53 -53
  158. package/dist/src/tiles/hover/hover-pane-controller.d.ts +219 -219
  159. package/dist/src/tiles/hover/hover-pane-controller.js +352 -352
  160. package/dist/src/tiles/hover/tile-hover-pane.d.ts +15 -15
  161. package/dist/src/tiles/hover/tile-hover-pane.js +38 -38
  162. package/dist/src/tiles/image-block.d.ts +17 -17
  163. package/dist/src/tiles/image-block.js +73 -73
  164. package/dist/src/tiles/item-image.d.ts +39 -39
  165. package/dist/src/tiles/item-image.js +154 -154
  166. package/dist/src/tiles/list/tile-list-compact-header.d.ts +6 -6
  167. package/dist/src/tiles/list/tile-list-compact-header.js +38 -38
  168. package/dist/src/tiles/list/tile-list-compact.d.ts +15 -15
  169. package/dist/src/tiles/list/tile-list-compact.js +114 -114
  170. package/dist/src/tiles/list/tile-list.d.ts +50 -50
  171. package/dist/src/tiles/list/tile-list.js +315 -315
  172. package/dist/src/tiles/mediatype-icon.d.ts +9 -9
  173. package/dist/src/tiles/mediatype-icon.js +47 -47
  174. package/dist/src/tiles/overlay/icon-overlay.d.ts +10 -10
  175. package/dist/src/tiles/overlay/icon-overlay.js +40 -40
  176. package/dist/src/tiles/overlay/icon-text-overlay.d.ts +9 -9
  177. package/dist/src/tiles/overlay/icon-text-overlay.js +38 -38
  178. package/dist/src/tiles/overlay/text-overlay.d.ts +10 -10
  179. package/dist/src/tiles/overlay/text-overlay.js +42 -42
  180. package/dist/src/tiles/review-block.d.ts +12 -12
  181. package/dist/src/tiles/review-block.js +56 -56
  182. package/dist/src/tiles/text-snippet-block.d.ts +27 -27
  183. package/dist/src/tiles/text-snippet-block.js +73 -73
  184. package/dist/src/tiles/tile-dispatcher.d.ts +64 -64
  185. package/dist/src/tiles/tile-dispatcher.js +230 -233
  186. package/dist/src/tiles/tile-dispatcher.js.map +1 -1
  187. package/dist/src/tiles/tile-display-value-provider.d.ts +47 -47
  188. package/dist/src/tiles/tile-display-value-provider.js +94 -94
  189. package/dist/src/utils/analytics-events.d.ts +25 -25
  190. package/dist/src/utils/analytics-events.js +27 -27
  191. package/dist/src/utils/array-equals.d.ts +4 -4
  192. package/dist/src/utils/array-equals.js +10 -10
  193. package/dist/src/utils/collapse-repeated-quotes.d.ts +11 -11
  194. package/dist/src/utils/collapse-repeated-quotes.js +13 -13
  195. package/dist/src/utils/format-count.d.ts +7 -7
  196. package/dist/src/utils/format-count.js +76 -76
  197. package/dist/src/utils/format-date.d.ts +2 -2
  198. package/dist/src/utils/format-date.js +25 -25
  199. package/dist/src/utils/format-unit-size.d.ts +2 -2
  200. package/dist/src/utils/format-unit-size.js +33 -33
  201. package/dist/src/utils/local-date-from-utc.d.ts +9 -9
  202. package/dist/src/utils/local-date-from-utc.js +15 -15
  203. package/dist/src/utils/log.d.ts +7 -7
  204. package/dist/src/utils/log.js +15 -15
  205. package/dist/src/utils/resolve-mediatype.d.ts +8 -8
  206. package/dist/src/utils/resolve-mediatype.js +23 -23
  207. package/dist/src/utils/sha1.d.ts +2 -2
  208. package/dist/src/utils/sha1.js +8 -8
  209. package/dist/test/collection-browser.test.d.ts +1 -1
  210. package/dist/test/collection-browser.test.js +1243 -1215
  211. package/dist/test/collection-browser.test.js.map +1 -1
  212. package/dist/test/collection-facets/facet-row.test.d.ts +1 -1
  213. package/dist/test/collection-facets/facet-row.test.js +203 -203
  214. package/dist/test/collection-facets/facets-template.test.d.ts +1 -1
  215. package/dist/test/collection-facets/facets-template.test.js +105 -105
  216. package/dist/test/collection-facets/more-facets-content.test.d.ts +1 -1
  217. package/dist/test/collection-facets/more-facets-content.test.js +133 -133
  218. package/dist/test/collection-facets/more-facets-pagination.test.d.ts +1 -1
  219. package/dist/test/collection-facets/more-facets-pagination.test.js +117 -117
  220. package/dist/test/collection-facets/toggle-switch.test.d.ts +1 -1
  221. package/dist/test/collection-facets/toggle-switch.test.js +73 -73
  222. package/dist/test/collection-facets.test.d.ts +2 -2
  223. package/dist/test/collection-facets.test.js +652 -652
  224. package/dist/test/data-source/collection-browser-data-source.test.d.ts +1 -1
  225. package/dist/test/data-source/collection-browser-data-source.test.js +89 -89
  226. package/dist/test/empty-placeholder.test.d.ts +1 -1
  227. package/dist/test/empty-placeholder.test.js +63 -63
  228. package/dist/test/expanded-date-picker.test.d.ts +1 -1
  229. package/dist/test/expanded-date-picker.test.js +95 -95
  230. package/dist/test/icon-overlay.test.d.ts +1 -1
  231. package/dist/test/icon-overlay.test.js +24 -24
  232. package/dist/test/image-block.test.d.ts +1 -1
  233. package/dist/test/image-block.test.js +48 -48
  234. package/dist/test/item-image.test.d.ts +1 -1
  235. package/dist/test/item-image.test.js +85 -85
  236. package/dist/test/manage/manage-bar.test.d.ts +1 -1
  237. package/dist/test/manage/manage-bar.test.js +77 -72
  238. package/dist/test/manage/manage-bar.test.js.map +1 -1
  239. package/dist/test/mediatype-config.test.d.ts +1 -1
  240. package/dist/test/mediatype-config.test.js +16 -16
  241. package/dist/test/mocks/mock-analytics-handler.d.ts +10 -10
  242. package/dist/test/mocks/mock-analytics-handler.js +15 -15
  243. package/dist/test/mocks/mock-search-responses.d.ts +24 -24
  244. package/dist/test/mocks/mock-search-responses.js +840 -840
  245. package/dist/test/mocks/mock-search-service.d.ts +15 -15
  246. package/dist/test/mocks/mock-search-service.js +53 -53
  247. package/dist/test/restoration-state-handler.test.d.ts +1 -1
  248. package/dist/test/restoration-state-handler.test.js +270 -270
  249. package/dist/test/review-block.test.d.ts +1 -1
  250. package/dist/test/review-block.test.js +44 -44
  251. package/dist/test/sort-filter-bar/alpha-bar-tooltip.test.d.ts +1 -1
  252. package/dist/test/sort-filter-bar/alpha-bar-tooltip.test.js +12 -12
  253. package/dist/test/sort-filter-bar/alpha-bar.test.d.ts +1 -1
  254. package/dist/test/sort-filter-bar/alpha-bar.test.js +73 -73
  255. package/dist/test/sort-filter-bar/sort-filter-bar.test.d.ts +1 -1
  256. package/dist/test/sort-filter-bar/sort-filter-bar.test.js +420 -420
  257. package/dist/test/text-overlay.test.d.ts +1 -1
  258. package/dist/test/text-overlay.test.js +48 -48
  259. package/dist/test/text-snippet-block.test.d.ts +1 -1
  260. package/dist/test/text-snippet-block.test.js +57 -57
  261. package/dist/test/tile-stats.test.d.ts +1 -1
  262. package/dist/test/tile-stats.test.js +81 -81
  263. package/dist/test/tiles/grid/account-tile.test.d.ts +1 -1
  264. package/dist/test/tiles/grid/account-tile.test.js +76 -76
  265. package/dist/test/tiles/grid/collection-tile.test.d.ts +1 -1
  266. package/dist/test/tiles/grid/collection-tile.test.js +73 -73
  267. package/dist/test/tiles/grid/item-tile.test.d.ts +1 -1
  268. package/dist/test/tiles/grid/item-tile.test.js +312 -312
  269. package/dist/test/tiles/grid/search-tile.test.d.ts +1 -1
  270. package/dist/test/tiles/grid/search-tile.test.js +51 -51
  271. package/dist/test/tiles/hover/hover-pane-controller.test.d.ts +1 -1
  272. package/dist/test/tiles/hover/hover-pane-controller.test.js +259 -259
  273. package/dist/test/tiles/hover/tile-hover-pane.test.d.ts +1 -1
  274. package/dist/test/tiles/hover/tile-hover-pane.test.js +13 -13
  275. package/dist/test/tiles/list/tile-list-compact.test.d.ts +1 -1
  276. package/dist/test/tiles/list/tile-list-compact.test.js +143 -143
  277. package/dist/test/tiles/list/tile-list.test.d.ts +1 -1
  278. package/dist/test/tiles/list/tile-list.test.js +297 -297
  279. package/dist/test/tiles/tile-dispatcher.test.d.ts +1 -1
  280. package/dist/test/tiles/tile-dispatcher.test.js +100 -100
  281. package/dist/test/tiles/tile-display-value-provider.test.d.ts +1 -1
  282. package/dist/test/tiles/tile-display-value-provider.test.js +141 -141
  283. package/dist/test/utils/array-equals.test.d.ts +1 -1
  284. package/dist/test/utils/array-equals.test.js +26 -26
  285. package/dist/test/utils/format-count.test.d.ts +1 -1
  286. package/dist/test/utils/format-count.test.js +23 -23
  287. package/dist/test/utils/format-date.test.d.ts +1 -1
  288. package/dist/test/utils/format-date.test.js +17 -17
  289. package/dist/test/utils/format-unit-size.test.d.ts +1 -1
  290. package/dist/test/utils/format-unit-size.test.js +17 -17
  291. package/dist/test/utils/local-date-from-utc.test.d.ts +1 -1
  292. package/dist/test/utils/local-date-from-utc.test.js +26 -26
  293. package/local.archive.org.cert +86 -86
  294. package/local.archive.org.key +27 -27
  295. package/package.json +1 -1
  296. package/renovate.json +6 -6
  297. package/src/app-root.ts +25 -59
  298. package/src/collection-browser.ts +7 -9
  299. package/src/manage/manage-bar.ts +22 -33
  300. package/src/sort-filter-bar/sort-filter-bar.ts +1 -10
  301. package/src/styles/ia-button.ts +107 -0
  302. package/src/tiles/tile-dispatcher.ts +1 -3
  303. package/test/collection-browser.test.ts +38 -0
  304. package/test/manage/manage-bar.test.ts +5 -0
  305. package/tsconfig.json +21 -21
  306. package/web-dev-server.config.mjs +30 -30
  307. package/web-test-runner.config.mjs +41 -41
@@ -1,931 +1,931 @@
1
- import { FilterConstraint, FilterMapBuilder, SearchType, } from '@internetarchive/search-service';
2
- import { prefixFilterAggregationKeys, TileModel, SortField, SORT_OPTIONS, } from '../models';
3
- import { FACETLESS_PAGE_ELEMENTS } from './models';
4
- import { sha1 } from '../utils/sha1';
5
- import { log } from '../utils/log';
6
- export class CollectionBrowserDataSource {
7
- // eslint-disable-next-line no-useless-constructor
8
- constructor(
9
- /** The host element to which this controller should attach listeners */
10
- host,
11
- /** Default size of result pages */
12
- pageSize = 50) {
13
- this.host = host;
14
- this.pageSize = pageSize;
15
- /**
16
- * All pages of tile models that have been fetched so far, indexed by their page
17
- * number (with the first being page 1).
18
- */
19
- this.pages = {};
20
- /**
21
- * Tile offset to apply when looking up tiles in the pages, in order to maintain
22
- * page alignment after tiles are removed.
23
- */
24
- this.offset = 0;
25
- /**
26
- * Total number of tile models stored in this data source's pages
27
- */
28
- this.numTileModels = 0;
29
- /**
30
- * A set of fetch IDs that are valid for the current query state
31
- */
32
- this.fetchesInProgress = new Set();
33
- /**
34
- * A record of the query key used for the last search.
35
- * If this changes, we need to load new results.
36
- */
37
- this.previousQueryKey = '';
38
- /**
39
- * Whether the initial page of search results for the current query state
40
- * is presently being fetched.
41
- */
42
- this.searchResultsLoading = false;
43
- /**
44
- * Whether the facets (aggregations) for the current query state are
45
- * presently being fetched.
46
- */
47
- this.facetsLoading = false;
48
- /**
49
- * Whether further query changes should be ignored and not trigger fetches
50
- */
51
- this.suppressFetches = false;
52
- /**
53
- * @inheritdoc
54
- */
55
- this.totalResults = 0;
56
- /**
57
- * @inheritdoc
58
- */
59
- this.endOfDataReached = false;
60
- /**
61
- * @inheritdoc
62
- */
63
- this.queryInitialized = false;
64
- /**
65
- * @inheritdoc
66
- */
67
- this.collectionTitles = new Map();
68
- /**
69
- * @inheritdoc
70
- */
71
- this.parentCollections = [];
72
- /**
73
- * @inheritdoc
74
- */
75
- this.prefixFilterCountMap = {};
76
- /**
77
- * Internal property to store the private value backing the `initialSearchComplete` getter.
78
- */
79
- this._initialSearchCompletePromise = new Promise(res => {
80
- this._initialSearchCompleteResolver = res;
81
- });
82
- /**
83
- * @inheritdoc
84
- */
85
- this.checkAllTiles = () => {
86
- this.map(model => {
87
- const cloned = model.clone();
88
- cloned.checked = true;
89
- return cloned;
90
- });
91
- };
92
- /**
93
- * @inheritdoc
94
- */
95
- this.uncheckAllTiles = () => {
96
- this.map(model => {
97
- const cloned = model.clone();
98
- cloned.checked = false;
99
- return cloned;
100
- });
101
- };
102
- /**
103
- * @inheritdoc
104
- */
105
- this.removeCheckedTiles = () => {
106
- // To make sure our data source remains page-aligned, we will offset our data source by
107
- // the number of removed tiles, so that we can just add the offset when the infinite
108
- // scroller queries for cell contents.
109
- // This only matters while we're still viewing the same set of results. If the user changes
110
- // their query/filters/sort, then the data source is overwritten and the offset cleared.
111
- const { checkedTileModels, uncheckedTileModels } = this;
112
- const numChecked = checkedTileModels.length;
113
- if (numChecked === 0)
114
- return;
115
- this.offset += numChecked;
116
- const newPages = {};
117
- // Which page the remaining tile models start on, post-offset
118
- let offsetPageNumber = Math.floor(this.offset / this.pageSize) + 1;
119
- let indexOnPage = this.offset % this.pageSize;
120
- // Fill the pages up to that point with empty models
121
- for (let page = 1; page <= offsetPageNumber; page += 1) {
122
- const remainingHidden = this.offset - this.pageSize * (page - 1);
123
- const offsetCellsOnPage = Math.min(this.pageSize, remainingHidden);
124
- newPages[page] = Array(offsetCellsOnPage).fill(undefined);
125
- }
126
- // Shift all the remaining tiles into their new positions in the data source
127
- for (const model of uncheckedTileModels) {
128
- if (!newPages[offsetPageNumber])
129
- newPages[offsetPageNumber] = [];
130
- newPages[offsetPageNumber].push(model);
131
- indexOnPage += 1;
132
- if (indexOnPage >= this.pageSize) {
133
- offsetPageNumber += 1;
134
- indexOnPage = 0;
135
- }
136
- }
137
- // Swap in the new pages
138
- this.pages = newPages;
139
- this.numTileModels -= numChecked;
140
- this.requestHostUpdate();
141
- this.refreshVisibleResults();
142
- };
143
- // Just setting some property values
144
- }
145
- /**
146
- * @inheritdoc
147
- */
148
- get initialSearchComplete() {
149
- return this._initialSearchCompletePromise;
150
- }
151
- hostConnected() {
152
- this.setSearchResultsLoading(this.searchResultsLoading);
153
- this.setFacetsLoading(this.facetsLoading);
154
- }
155
- hostUpdate() {
156
- // This reactive controller hook is run whenever the host component (collection-browser) performs an update.
157
- // We check whether the host's state has changed in a way which should trigger a reset & new results fetch.
158
- // Only the currently-installed data source should react to the update
159
- if (!this.activeOnHost)
160
- return;
161
- // Copy loading states onto the host
162
- this.setSearchResultsLoading(this.searchResultsLoading);
163
- this.setFacetsLoading(this.facetsLoading);
164
- // Can't perform searches without a search service
165
- if (!this.host.searchService)
166
- return;
167
- // We should only reset if part of the full query state has changed
168
- const queryKeyChanged = this.pageFetchQueryKey !== this.previousQueryKey;
169
- if (!queryKeyChanged)
170
- return;
171
- // We should only reset if either:
172
- // (a) our state permits a valid search, or
173
- // (b) we have a blank query that we want to show empty results for
174
- const shouldShowEmptyQueryResults = this.host.clearResultsOnEmptyQuery && this.host.baseQuery === '';
175
- if (!(this.canPerformSearch || shouldShowEmptyQueryResults))
176
- return;
177
- if (this.activeOnHost)
178
- this.host.emitQueryStateChanged();
179
- this.handleQueryChange();
180
- }
181
- /**
182
- * Returns whether this data source is the one currently installed on the host component.
183
- */
184
- get activeOnHost() {
185
- return this.host.dataSource === this;
186
- }
187
- /**
188
- * @inheritdoc
189
- */
190
- get size() {
191
- return this.numTileModels;
192
- }
193
- /**
194
- * @inheritdoc
195
- */
196
- reset() {
197
- log('Resetting CB data source');
198
- this.pages = {};
199
- this.aggregations = {};
200
- this.yearHistogramAggregation = undefined;
201
- this.pageElements = undefined;
202
- this.parentCollections = [];
203
- this.queryErrorMessage = undefined;
204
- this.offset = 0;
205
- this.numTileModels = 0;
206
- this.endOfDataReached = false;
207
- this.queryInitialized = false;
208
- // Invalidate any fetches in progress
209
- this.fetchesInProgress.clear();
210
- this.setTotalResultCount(0);
211
- this.requestHostUpdate();
212
- }
213
- /**
214
- * @inheritdoc
215
- */
216
- addPage(pageNum, pageTiles) {
217
- this.pages[pageNum] = pageTiles;
218
- this.numTileModels += pageTiles.length;
219
- this.requestHostUpdate();
220
- }
221
- /**
222
- * @inheritdoc
223
- */
224
- addMultiplePages(firstPageNum, tiles) {
225
- const numPages = Math.ceil(tiles.length / this.pageSize);
226
- for (let i = 0; i < numPages; i += 1) {
227
- const pageStartIndex = this.pageSize * i;
228
- this.addPage(firstPageNum + i, tiles.slice(pageStartIndex, pageStartIndex + this.pageSize));
229
- }
230
- const visiblePages = this.host.currentVisiblePageNumbers;
231
- const needsReload = visiblePages.some(page => page >= firstPageNum && page <= firstPageNum + numPages);
232
- if (needsReload) {
233
- this.refreshVisibleResults();
234
- }
235
- }
236
- /**
237
- * @inheritdoc
238
- */
239
- getPage(pageNum) {
240
- return this.pages[pageNum];
241
- }
242
- /**
243
- * @inheritdoc
244
- */
245
- getAllPages() {
246
- return this.pages;
247
- }
248
- /**
249
- * @inheritdoc
250
- */
251
- hasPage(pageNum) {
252
- return !!this.pages[pageNum];
253
- }
254
- /**
255
- * @inheritdoc
256
- */
257
- getTileModelAt(index) {
258
- var _a;
259
- const pageNum = Math.floor(index / this.pageSize) + 1;
260
- const indexOnPage = index % this.pageSize;
261
- return (_a = this.pages[pageNum]) === null || _a === void 0 ? void 0 : _a[indexOnPage];
262
- }
263
- /**
264
- * @inheritdoc
265
- */
266
- indexOf(tile) {
267
- return Object.values(this.pages).flat().indexOf(tile);
268
- }
269
- /**
270
- * @inheritdoc
271
- */
272
- getPageSize() {
273
- return this.pageSize;
274
- }
275
- /**
276
- * @inheritdoc
277
- */
278
- setPageSize(pageSize) {
279
- this.reset();
280
- this.pageSize = pageSize;
281
- }
282
- /**
283
- * @inheritdoc
284
- */
285
- setTotalResultCount(count) {
286
- this.totalResults = count;
287
- if (this.activeOnHost) {
288
- this.host.setTotalResultCount(count);
289
- }
290
- }
291
- /**
292
- * @inheritdoc
293
- */
294
- setFetchesSuppressed(suppressed) {
295
- this.suppressFetches = suppressed;
296
- }
297
- /**
298
- * @inheritdoc
299
- */
300
- async handleQueryChange() {
301
- // Don't react to the change if fetches are suppressed for this data source
302
- if (this.suppressFetches)
303
- return;
304
- this.reset();
305
- // Reset the `initialSearchComplete` promise with a new value for the imminent search
306
- this._initialSearchCompletePromise = new Promise(res => {
307
- this._initialSearchCompleteResolver = res;
308
- });
309
- const shouldFetchFacets = !this.host.suppressFacets &&
310
- !FACETLESS_PAGE_ELEMENTS.includes(this.host.profileElement);
311
- // Fire the initial page & facet requests
312
- this.queryInitialized = true;
313
- await Promise.all([
314
- this.doInitialPageFetch(),
315
- shouldFetchFacets ? this.fetchFacets() : null,
316
- ]);
317
- // Resolve the `initialSearchComplete` promise for this search
318
- this._initialSearchCompleteResolver(true);
319
- }
320
- /**
321
- * @inheritdoc
322
- */
323
- map(callback) {
324
- this.pages = Object.fromEntries(Object.entries(this.pages).map(([page, tileModels]) => [
325
- page,
326
- tileModels.map((model, index, array) => model ? callback(model, index, array) : model),
327
- ]));
328
- this.requestHostUpdate();
329
- this.refreshVisibleResults();
330
- }
331
- /**
332
- * @inheritdoc
333
- */
334
- get checkedTileModels() {
335
- return this.getFilteredTileModels(model => model.checked);
336
- }
337
- /**
338
- * @inheritdoc
339
- */
340
- get uncheckedTileModels() {
341
- return this.getFilteredTileModels(model => !model.checked);
342
- }
343
- /**
344
- * Returns a flattened, filtered array of all the tile models in the data source
345
- * for which the given predicate returns a truthy value.
346
- *
347
- * @param predicate A callback function to apply on each tile model, as with Array.filter
348
- * @returns A filtered array of tile models satisfying the predicate
349
- */
350
- getFilteredTileModels(predicate) {
351
- return Object.values(this.pages)
352
- .flat()
353
- .filter((model, index, array) => model ? predicate(model, index, array) : false);
354
- }
355
- /**
356
- * @inheritdoc
357
- */
358
- get canPerformSearch() {
359
- var _a;
360
- if (!this.host.searchService)
361
- return false;
362
- const trimmedQuery = (_a = this.host.baseQuery) === null || _a === void 0 ? void 0 : _a.trim();
363
- const hasNonEmptyQuery = !!trimmedQuery;
364
- const isCollectionSearch = !!this.host.withinCollection;
365
- const isProfileSearch = !!this.host.withinProfile;
366
- const hasProfileElement = !!this.host.profileElement;
367
- const isMetadataSearch = this.host.searchType === SearchType.METADATA;
368
- // Metadata searches within a collection/profile are allowed to have no query.
369
- // Otherwise, a non-empty query must be set.
370
- return (hasNonEmptyQuery ||
371
- (isCollectionSearch && isMetadataSearch) ||
372
- (isProfileSearch && hasProfileElement && isMetadataSearch));
373
- }
374
- /**
375
- * Sets the state for whether the initial set of search results for the
376
- * current query is loading
377
- */
378
- setSearchResultsLoading(loading) {
379
- this.searchResultsLoading = loading;
380
- if (this.activeOnHost) {
381
- this.host.setSearchResultsLoading(loading);
382
- }
383
- }
384
- /**
385
- * Sets the state for whether the facets for a query is loading
386
- */
387
- setFacetsLoading(loading) {
388
- this.facetsLoading = loading;
389
- if (this.activeOnHost) {
390
- this.host.setFacetsLoading(loading);
391
- }
392
- }
393
- /**
394
- * Requests that the host perform an update, provided this data
395
- * source is actively installed on it.
396
- */
397
- requestHostUpdate() {
398
- if (this.activeOnHost) {
399
- this.host.requestUpdate();
400
- }
401
- }
402
- /**
403
- * Requests that the host refresh its visible tiles, provided this
404
- * data source is actively installed on it.
405
- */
406
- refreshVisibleResults() {
407
- if (this.activeOnHost) {
408
- this.host.refreshVisibleResults();
409
- }
410
- }
411
- /**
412
- * The query key is a string that uniquely identifies the current search.
413
- * It consists of:
414
- * - The current base query
415
- * - The current collection/profile target & page element
416
- * - The current search type
417
- * - Any currently-applied facets
418
- * - Any currently-applied date range
419
- * - Any currently-applied prefix filters
420
- * - The current sort options
421
- *
422
- * This lets us internally keep track of queries so we don't persist data that's
423
- * no longer relevant. Not meant to be human-readable.
424
- */
425
- get pageFetchQueryKey() {
426
- var _a, _b, _c;
427
- const profileKey = `pf;${this.host.withinProfile}--pe;${this.host.profileElement}`;
428
- const pageTarget = (_a = this.host.withinCollection) !== null && _a !== void 0 ? _a : profileKey;
429
- const sortField = (_b = this.host.selectedSort) !== null && _b !== void 0 ? _b : 'none';
430
- const sortDirection = (_c = this.host.sortDirection) !== null && _c !== void 0 ? _c : 'none';
431
- return `fq:${this.fullQuery}-pt:${pageTarget}-st:${this.host.searchType}-sf:${sortField}-sd:${sortDirection}`;
432
- }
433
- /**
434
- * Similar to `pageFetchQueryKey` above, but excludes sort fields since they
435
- * are not relevant in determining aggregation queries.
436
- */
437
- get facetFetchQueryKey() {
438
- var _a;
439
- const profileKey = `pf;${this.host.withinProfile}--pe;${this.host.profileElement}`;
440
- const pageTarget = (_a = this.host.withinCollection) !== null && _a !== void 0 ? _a : profileKey;
441
- return `fq:${this.fullQuery}-pt:${pageTarget}-st:${this.host.searchType}`;
442
- }
443
- /**
444
- * Constructs a search service FilterMap object from the combination of
445
- * all the currently-applied filters. This includes any facets, letter
446
- * filters, and date range.
447
- */
448
- get filterMap() {
449
- const builder = new FilterMapBuilder();
450
- // Add the date range, if applicable
451
- if (this.host.minSelectedDate) {
452
- builder.addFilter('year', this.host.minSelectedDate, FilterConstraint.GREATER_OR_EQUAL);
453
- }
454
- if (this.host.maxSelectedDate) {
455
- builder.addFilter('year', this.host.maxSelectedDate, FilterConstraint.LESS_OR_EQUAL);
456
- }
457
- // Add any selected facets
458
- if (this.host.selectedFacets) {
459
- for (const [facetName, facetValues] of Object.entries(this.host.selectedFacets)) {
460
- const { name, values } = this.prepareFacetForFetch(facetName, facetValues);
461
- for (const [value, bucket] of Object.entries(values)) {
462
- let constraint;
463
- if (bucket.state === 'selected') {
464
- constraint = FilterConstraint.INCLUDE;
465
- }
466
- else if (bucket.state === 'hidden') {
467
- constraint = FilterConstraint.EXCLUDE;
468
- }
469
- if (constraint) {
470
- builder.addFilter(name, value, constraint);
471
- }
472
- }
473
- }
474
- }
475
- // Add any letter filters
476
- if (this.host.selectedTitleFilter) {
477
- builder.addFilter('firstTitle', this.host.selectedTitleFilter, FilterConstraint.INCLUDE);
478
- }
479
- if (this.host.selectedCreatorFilter) {
480
- builder.addFilter('firstCreator', this.host.selectedCreatorFilter, FilterConstraint.INCLUDE);
481
- }
482
- const filterMap = builder.build();
483
- return filterMap;
484
- }
485
- /**
486
- * Produces a compact unique ID for a search request that can help with debugging
487
- * on the backend by making related requests easier to trace through different services.
488
- * (e.g., tying the hits/aggregations requests for the same page back to a single hash).
489
- *
490
- * @param params The search service parameters for the request
491
- * @param kind The kind of request (hits-only, aggregations-only, or both)
492
- * @returns A Promise resolving to the uid to apply to the request
493
- */
494
- async requestUID(params, kind) {
495
- var _a;
496
- const paramsToHash = JSON.stringify({
497
- pageType: params.pageType,
498
- pageTarget: params.pageTarget,
499
- query: params.query,
500
- fields: params.fields,
501
- filters: params.filters,
502
- sort: params.sort,
503
- searchType: this.host.searchType,
504
- });
505
- const fullQueryHash = (await sha1(paramsToHash)).slice(0, 20); // First 80 bits of SHA-1 are plenty for this
506
- const sessionId = (await this.host.getSessionId()).slice(0, 20); // Likewise
507
- const page = (_a = params.page) !== null && _a !== void 0 ? _a : 0;
508
- const kindPrefix = kind.charAt(0); // f = full, h = hits, a = aggregations
509
- const currentTime = Date.now();
510
- return `R:${fullQueryHash}-S:${sessionId}-P:${page}-K:${kindPrefix}-T:${currentTime}`;
511
- }
512
- /**
513
- * @inheritdoc
514
- */
515
- get pageSpecifierParams() {
516
- if (this.host.withinCollection) {
517
- return {
518
- pageType: 'collection_details',
519
- pageTarget: this.host.withinCollection,
520
- };
521
- }
522
- if (this.host.withinProfile) {
523
- return {
524
- pageType: 'account_details',
525
- pageTarget: this.host.withinProfile,
526
- pageElements: this.host.profileElement
527
- ? [this.host.profileElement]
528
- : [],
529
- };
530
- }
531
- return null;
532
- }
533
- /**
534
- * The full query, including year facets and date range clauses
535
- */
536
- get fullQuery() {
537
- var _a, _b;
538
- let fullQuery = (_b = (_a = this.host.baseQuery) === null || _a === void 0 ? void 0 : _a.trim()) !== null && _b !== void 0 ? _b : '';
539
- const { facetQuery, dateRangeQueryClause, sortFilterQueries } = this;
540
- if (facetQuery) {
541
- fullQuery += ` AND ${facetQuery}`;
542
- }
543
- if (dateRangeQueryClause) {
544
- fullQuery += ` AND ${dateRangeQueryClause}`;
545
- }
546
- if (sortFilterQueries) {
547
- fullQuery += ` AND ${sortFilterQueries}`;
548
- }
549
- return fullQuery.trim();
550
- }
551
- /**
552
- * Generates a query string representing the current set of applied facets
553
- *
554
- * Example: `mediatype:("collection" OR "audio" OR -"etree") AND year:("2000" OR "2001")`
555
- */
556
- get facetQuery() {
557
- var _a;
558
- if (!this.host.selectedFacets)
559
- return undefined;
560
- const facetClauses = [];
561
- for (const [facetName, facetValues] of Object.entries(this.host.selectedFacets)) {
562
- facetClauses.push(this.buildFacetClause(facetName, facetValues));
563
- }
564
- return (_a = this.joinFacetClauses(facetClauses)) === null || _a === void 0 ? void 0 : _a.trim();
565
- }
566
- get dateRangeQueryClause() {
567
- if (!this.host.minSelectedDate || !this.host.maxSelectedDate) {
568
- return undefined;
569
- }
570
- return `year:[${this.host.minSelectedDate} TO ${this.host.maxSelectedDate}]`;
571
- }
572
- get sortFilterQueries() {
573
- const queries = [this.titleQuery, this.creatorQuery];
574
- return queries.filter(q => q).join(' AND ');
575
- }
576
- /**
577
- * Returns a query clause identifying the currently selected title filter,
578
- * e.g., `firstTitle:X`.
579
- */
580
- get titleQuery() {
581
- return this.host.selectedTitleFilter
582
- ? `firstTitle:${this.host.selectedTitleFilter}`
583
- : undefined;
584
- }
585
- /**
586
- * Returns a query clause identifying the currently selected creator filter,
587
- * e.g., `firstCreator:X`.
588
- */
589
- get creatorQuery() {
590
- return this.host.selectedCreatorFilter
591
- ? `firstCreator:${this.host.selectedCreatorFilter}`
592
- : undefined;
593
- }
594
- /**
595
- * Builds an OR-joined facet clause for the given facet name and values.
596
- *
597
- * E.g., for name `subject` and values
598
- * `{ foo: { state: 'selected' }, bar: { state: 'hidden' } }`
599
- * this will produce the clause
600
- * `subject:("foo" OR -"bar")`.
601
- *
602
- * @param facetName The facet type (e.g., 'collection')
603
- * @param facetValues The facet buckets, mapped by their keys
604
- */
605
- buildFacetClause(facetName, facetValues) {
606
- const { name: facetQueryName, values } = this.prepareFacetForFetch(facetName, facetValues);
607
- const facetEntries = Object.entries(values);
608
- if (facetEntries.length === 0)
609
- return '';
610
- const facetValuesArray = [];
611
- for (const [key, facetData] of facetEntries) {
612
- const plusMinusPrefix = facetData.state === 'hidden' ? '-' : '';
613
- facetValuesArray.push(`${plusMinusPrefix}"${key}"`);
614
- }
615
- const valueQuery = facetValuesArray.join(` OR `);
616
- return `${facetQueryName}:(${valueQuery})`;
617
- }
618
- /**
619
- * Handles some special pre-request normalization steps for certain facet types
620
- * that require them.
621
- *
622
- * @param facetName The name of the facet type (e.g., 'language')
623
- * @param facetValues An array of values for that facet type
624
- */
625
- prepareFacetForFetch(facetName, facetValues) {
626
- // eslint-disable-next-line prefer-const
627
- let [normalizedName, normalizedValues] = [facetName, facetValues];
628
- // The full "search engine" name of the lending field is "lending___status"
629
- if (facetName === 'lending') {
630
- normalizedName = 'lending___status';
631
- }
632
- return {
633
- name: normalizedName,
634
- values: normalizedValues,
635
- };
636
- }
637
- /**
638
- * Takes an array of facet clauses, and combines them into a
639
- * full AND-joined facet query string. Empty clauses are ignored.
640
- */
641
- joinFacetClauses(facetClauses) {
642
- const nonEmptyFacetClauses = facetClauses.filter(clause => clause.length > 0);
643
- return nonEmptyFacetClauses.length > 0
644
- ? `(${nonEmptyFacetClauses.join(' AND ')})`
645
- : undefined;
646
- }
647
- /**
648
- * Fires a backend request to fetch a set of aggregations (representing UI facets) for
649
- * the current search state.
650
- */
651
- async fetchFacets() {
652
- var _a, _b, _c, _d, _e, _f, _g, _h, _j;
653
- const trimmedQuery = (_a = this.host.baseQuery) === null || _a === void 0 ? void 0 : _a.trim();
654
- if (!this.canPerformSearch)
655
- return;
656
- const { facetFetchQueryKey } = this;
657
- if (this.fetchesInProgress.has(facetFetchQueryKey))
658
- return;
659
- this.fetchesInProgress.add(facetFetchQueryKey);
660
- const sortParams = this.host.sortParam ? [this.host.sortParam] : [];
661
- const params = {
662
- ...this.pageSpecifierParams,
663
- query: trimmedQuery || '',
664
- rows: 0,
665
- filters: this.filterMap,
666
- // Fetch a few extra buckets beyond the 6 we show, in case some get suppressed
667
- aggregationsSize: 10,
668
- // Note: we don't need an aggregations param to fetch the default aggregations from the PPS.
669
- // The default aggregations for the search_results page type should be what we need here.
670
- };
671
- params.uid = await this.requestUID({ ...params, sort: sortParams }, 'aggregations');
672
- this.setFacetsLoading(true);
673
- const searchResponse = await ((_b = this.host.searchService) === null || _b === void 0 ? void 0 : _b.search(params, this.host.searchType));
674
- const success = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.success;
675
- // This is checking to see if the query has changed since the data was fetched.
676
- // If so, we just want to discard this set of aggregations because they are
677
- // likely no longer valid for the newer query.
678
- const queryChangedSinceFetch = !this.fetchesInProgress.has(facetFetchQueryKey);
679
- this.fetchesInProgress.delete(facetFetchQueryKey);
680
- if (queryChangedSinceFetch)
681
- return;
682
- if (!success) {
683
- const errorMsg = (_c = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.error) === null || _c === void 0 ? void 0 : _c.message;
684
- const detailMsg = (_e = (_d = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.error) === null || _d === void 0 ? void 0 : _d.details) === null || _e === void 0 ? void 0 : _e.message;
685
- if (!errorMsg && !detailMsg) {
686
- // @ts-ignore: Property 'Sentry' does not exist on type 'Window & typeof globalThis'
687
- (_g = (_f = window === null || window === void 0 ? void 0 : window.Sentry) === null || _f === void 0 ? void 0 : _f.captureMessage) === null || _g === void 0 ? void 0 : _g.call(_f, 'Missing or malformed facet response from backend', 'error');
688
- }
689
- this.setFacetsLoading(false);
690
- return;
691
- }
692
- const { aggregations, collectionTitles } = success.response;
693
- this.aggregations = aggregations;
694
- if (collectionTitles) {
695
- for (const [id, title] of Object.entries(collectionTitles)) {
696
- this.collectionTitles.set(id, title);
697
- }
698
- }
699
- this.yearHistogramAggregation =
700
- (_j = (_h = success === null || success === void 0 ? void 0 : success.response) === null || _h === void 0 ? void 0 : _h.aggregations) === null || _j === void 0 ? void 0 : _j.year_histogram;
701
- this.setFacetsLoading(false);
702
- this.requestHostUpdate();
703
- }
704
- /**
705
- * Performs the initial page fetch(es) for the current search state.
706
- */
707
- async doInitialPageFetch() {
708
- this.setSearchResultsLoading(true);
709
- // Try to batch 2 initial page requests when possible
710
- await this.fetchPage(this.host.initialPageNumber, 2);
711
- }
712
- /**
713
- * Fetches one or more pages of results and updates the data source.
714
- *
715
- * @param pageNumber The page number to fetch
716
- * @param numInitialPages If this is an initial page fetch (`pageNumber = 1`),
717
- * specifies how many pages to batch together in one request. Ignored
718
- * if `pageNumber != 1`, defaulting to a single page.
719
- */
720
- async fetchPage(pageNumber, numInitialPages = 1) {
721
- var _a, _b, _c, _d, _e, _f, _g, _h, _j;
722
- const trimmedQuery = (_a = this.host.baseQuery) === null || _a === void 0 ? void 0 : _a.trim();
723
- if (!this.canPerformSearch)
724
- return;
725
- // if we already have data, don't fetch again
726
- if (this.hasPage(pageNumber))
727
- return;
728
- if (this.endOfDataReached)
729
- return;
730
- // Batch multiple initial page requests together if needed (e.g., can request
731
- // pages 1 and 2 together in a single request).
732
- let numPages = pageNumber === 1 ? numInitialPages : 1;
733
- const numRows = this.pageSize * numPages;
734
- // if a fetch is already in progress for this query and page, don't fetch again
735
- const { pageFetchQueryKey } = this;
736
- const currentPageKey = `${pageFetchQueryKey}-p:${pageNumber}`;
737
- if (this.fetchesInProgress.has(currentPageKey))
738
- return;
739
- for (let i = 0; i < numPages; i += 1) {
740
- this.fetchesInProgress.add(`${pageFetchQueryKey}-p:${pageNumber + i}`);
741
- }
742
- this.previousQueryKey = pageFetchQueryKey;
743
- let sortParams = this.host.sortParam ? [this.host.sortParam] : [];
744
- // TODO eventually the PPS should handle these defaults natively
745
- const isDefaultProfileSort = this.host.withinProfile && this.host.selectedSort === SortField.default;
746
- if (isDefaultProfileSort && this.host.defaultSortParam) {
747
- const sortOption = SORT_OPTIONS[this.host.defaultSortParam.field];
748
- if (sortOption.searchServiceKey) {
749
- sortParams = [
750
- {
751
- field: sortOption.searchServiceKey,
752
- direction: 'desc',
753
- },
754
- ];
755
- }
756
- }
757
- const params = {
758
- ...this.pageSpecifierParams,
759
- query: trimmedQuery || '',
760
- page: pageNumber,
761
- rows: numRows,
762
- sort: sortParams,
763
- filters: this.filterMap,
764
- aggregations: { omit: true },
765
- };
766
- params.uid = await this.requestUID(params, 'hits');
767
- log('=== FIRING PAGE REQUEST ===', params);
768
- const searchResponse = await ((_b = this.host.searchService) === null || _b === void 0 ? void 0 : _b.search(params, this.host.searchType));
769
- log('=== RECEIVED PAGE RESPONSE IN CB ===', searchResponse);
770
- const success = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.success;
771
- // This is checking to see if the fetch has been invalidated since it was fired off.
772
- // If so, we just want to discard the response since it is for an obsolete query state.
773
- if (!this.fetchesInProgress.has(currentPageKey))
774
- return;
775
- for (let i = 0; i < numPages; i += 1) {
776
- this.fetchesInProgress.delete(`${pageFetchQueryKey}-p:${pageNumber + i}`);
777
- }
778
- if (!success) {
779
- const errorMsg = (_c = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.error) === null || _c === void 0 ? void 0 : _c.message;
780
- const detailMsg = (_e = (_d = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.error) === null || _d === void 0 ? void 0 : _d.details) === null || _e === void 0 ? void 0 : _e.message;
781
- this.queryErrorMessage = `${errorMsg !== null && errorMsg !== void 0 ? errorMsg : ''}${detailMsg ? `; ${detailMsg}` : ''}`;
782
- if (!this.queryErrorMessage) {
783
- this.queryErrorMessage = 'Missing or malformed response from backend';
784
- // @ts-ignore: Property 'Sentry' does not exist on type 'Window & typeof globalThis'
785
- (_g = (_f = window === null || window === void 0 ? void 0 : window.Sentry) === null || _f === void 0 ? void 0 : _f.captureMessage) === null || _g === void 0 ? void 0 : _g.call(_f, this.queryErrorMessage, 'error');
786
- }
787
- this.setSearchResultsLoading(false);
788
- this.requestHostUpdate();
789
- return;
790
- }
791
- this.setTotalResultCount(success.response.totalResults - this.offset);
792
- if (this.activeOnHost && this.totalResults === 0) {
793
- // display event to offshoot when result count is zero.
794
- this.host.emitEmptyResults();
795
- }
796
- if (this.host.withinCollection) {
797
- this.collectionExtraInfo = success.response.collectionExtraInfo;
798
- // For collections, we want the UI to respect the default sort option
799
- // which can be specified in metadata, or otherwise assumed to be week:desc
800
- if (this.activeOnHost) {
801
- this.host.applyDefaultCollectionSort(this.collectionExtraInfo);
802
- }
803
- if (this.collectionExtraInfo) {
804
- this.parentCollections = [].concat((_j = (_h = this.collectionExtraInfo.public_metadata) === null || _h === void 0 ? void 0 : _h.collection) !== null && _j !== void 0 ? _j : []);
805
- }
806
- }
807
- else if (this.host.withinProfile) {
808
- this.accountExtraInfo = success.response.accountExtraInfo;
809
- this.pageElements = success.response.pageElements;
810
- }
811
- const { results, collectionTitles } = success.response;
812
- if (results && results.length > 0) {
813
- // Load any collection titles present on the response into the cache,
814
- // or queue up preload fetches for them if none were present.
815
- if (collectionTitles) {
816
- for (const [id, title] of Object.entries(collectionTitles)) {
817
- this.collectionTitles.set(id, title);
818
- }
819
- }
820
- // Update the data source for each returned page.
821
- // For loans and web archives, we must account for receiving more pages than we asked for.
822
- if (this.host.profileElement === 'lending' ||
823
- this.host.profileElement === 'web_archives') {
824
- numPages = Math.ceil(results.length / this.pageSize);
825
- this.endOfDataReached = true;
826
- if (this.activeOnHost)
827
- this.host.setTileCount(this.totalResults);
828
- }
829
- for (let i = 0; i < numPages; i += 1) {
830
- const pageStartIndex = this.pageSize * i;
831
- this.addFetchedResultsToDataSource(pageNumber + i, results.slice(pageStartIndex, pageStartIndex + this.pageSize));
832
- }
833
- }
834
- // When we reach the end of the data, we can set the infinite scroller's
835
- // item count to the real total number of results (rather than the
836
- // temporary estimates based on pages rendered so far).
837
- const resultCountDiscrepancy = numRows - results.length;
838
- if (resultCountDiscrepancy > 0) {
839
- this.endOfDataReached = true;
840
- if (this.activeOnHost)
841
- this.host.setTileCount(this.totalResults);
842
- }
843
- this.setSearchResultsLoading(false);
844
- this.requestHostUpdate();
845
- }
846
- /**
847
- * Update the datasource from the fetch response
848
- *
849
- * @param pageNumber
850
- * @param results
851
- */
852
- addFetchedResultsToDataSource(pageNumber, results) {
853
- const tiles = [];
854
- results === null || results === void 0 ? void 0 : results.forEach(result => {
855
- if (!result.identifier)
856
- return;
857
- tiles.push(new TileModel(result));
858
- });
859
- this.addPage(pageNumber, tiles);
860
- const visiblePages = this.host.currentVisiblePageNumbers;
861
- const needsReload = visiblePages.includes(pageNumber);
862
- if (needsReload) {
863
- this.refreshVisibleResults();
864
- }
865
- }
866
- /**
867
- * Fetches the aggregation buckets for the given prefix filter type.
868
- */
869
- async fetchPrefixFilterBuckets(filterType) {
870
- var _a, _b, _c, _d, _e, _f, _g;
871
- const trimmedQuery = (_a = this.host.baseQuery) === null || _a === void 0 ? void 0 : _a.trim();
872
- if (!this.canPerformSearch)
873
- return [];
874
- const filterAggregationKey = prefixFilterAggregationKeys[filterType];
875
- const sortParams = this.host.sortParam ? [this.host.sortParam] : [];
876
- const params = {
877
- ...this.pageSpecifierParams,
878
- query: trimmedQuery || '',
879
- rows: 0,
880
- filters: this.filterMap,
881
- // Only fetch the firstTitle or firstCreator aggregation
882
- aggregations: { simpleParams: [filterAggregationKey] },
883
- // Fetch all 26 letter buckets
884
- aggregationsSize: 26,
885
- };
886
- params.uid = await this.requestUID({ ...params, sort: sortParams }, 'aggregations');
887
- const searchResponse = await ((_b = this.host.searchService) === null || _b === void 0 ? void 0 : _b.search(params, this.host.searchType));
888
- return ((_g = (_f = (_e = (_d = (_c = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.success) === null || _c === void 0 ? void 0 : _c.response) === null || _d === void 0 ? void 0 : _d.aggregations) === null || _e === void 0 ? void 0 : _e[filterAggregationKey]) === null || _f === void 0 ? void 0 : _f.buckets) !== null && _g !== void 0 ? _g : []);
889
- }
890
- /**
891
- * Fetches and caches the prefix filter counts for the given filter type.
892
- */
893
- async updatePrefixFilterCounts(filterType) {
894
- const { facetFetchQueryKey } = this;
895
- const buckets = await this.fetchPrefixFilterBuckets(filterType);
896
- // Don't update the filter counts for an outdated query (if it has been changed
897
- // since we sent the request)
898
- const queryChangedSinceFetch = facetFetchQueryKey !== this.facetFetchQueryKey;
899
- if (queryChangedSinceFetch)
900
- return;
901
- // Unpack the aggregation buckets into a simple map like { 'A': 50, 'B': 25, ... }
902
- this.prefixFilterCountMap = { ...this.prefixFilterCountMap }; // Clone the object to trigger an update
903
- this.prefixFilterCountMap[filterType] = buckets.reduce((acc, bucket) => {
904
- acc[bucket.key.toUpperCase()] = bucket.doc_count;
905
- return acc;
906
- }, {});
907
- this.requestHostUpdate();
908
- }
909
- /**
910
- * @inheritdoc
911
- */
912
- async updatePrefixFiltersForCurrentSort() {
913
- if (['title', 'creator'].includes(this.host.selectedSort)) {
914
- const filterType = this.host.selectedSort;
915
- if (!this.prefixFilterCountMap[filterType]) {
916
- this.updatePrefixFilterCounts(filterType);
917
- }
918
- }
919
- }
920
- /**
921
- * @inheritdoc
922
- */
923
- refreshLetterCounts() {
924
- if (Object.keys(this.prefixFilterCountMap).length > 0) {
925
- this.prefixFilterCountMap = {};
926
- }
927
- this.updatePrefixFiltersForCurrentSort();
928
- this.requestHostUpdate();
929
- }
930
- }
1
+ import { FilterConstraint, FilterMapBuilder, SearchType, } from '@internetarchive/search-service';
2
+ import { prefixFilterAggregationKeys, TileModel, SortField, SORT_OPTIONS, } from '../models';
3
+ import { FACETLESS_PAGE_ELEMENTS } from './models';
4
+ import { sha1 } from '../utils/sha1';
5
+ import { log } from '../utils/log';
6
+ export class CollectionBrowserDataSource {
7
+ // eslint-disable-next-line no-useless-constructor
8
+ constructor(
9
+ /** The host element to which this controller should attach listeners */
10
+ host,
11
+ /** Default size of result pages */
12
+ pageSize = 50) {
13
+ this.host = host;
14
+ this.pageSize = pageSize;
15
+ /**
16
+ * All pages of tile models that have been fetched so far, indexed by their page
17
+ * number (with the first being page 1).
18
+ */
19
+ this.pages = {};
20
+ /**
21
+ * Tile offset to apply when looking up tiles in the pages, in order to maintain
22
+ * page alignment after tiles are removed.
23
+ */
24
+ this.offset = 0;
25
+ /**
26
+ * Total number of tile models stored in this data source's pages
27
+ */
28
+ this.numTileModels = 0;
29
+ /**
30
+ * A set of fetch IDs that are valid for the current query state
31
+ */
32
+ this.fetchesInProgress = new Set();
33
+ /**
34
+ * A record of the query key used for the last search.
35
+ * If this changes, we need to load new results.
36
+ */
37
+ this.previousQueryKey = '';
38
+ /**
39
+ * Whether the initial page of search results for the current query state
40
+ * is presently being fetched.
41
+ */
42
+ this.searchResultsLoading = false;
43
+ /**
44
+ * Whether the facets (aggregations) for the current query state are
45
+ * presently being fetched.
46
+ */
47
+ this.facetsLoading = false;
48
+ /**
49
+ * Whether further query changes should be ignored and not trigger fetches
50
+ */
51
+ this.suppressFetches = false;
52
+ /**
53
+ * @inheritdoc
54
+ */
55
+ this.totalResults = 0;
56
+ /**
57
+ * @inheritdoc
58
+ */
59
+ this.endOfDataReached = false;
60
+ /**
61
+ * @inheritdoc
62
+ */
63
+ this.queryInitialized = false;
64
+ /**
65
+ * @inheritdoc
66
+ */
67
+ this.collectionTitles = new Map();
68
+ /**
69
+ * @inheritdoc
70
+ */
71
+ this.parentCollections = [];
72
+ /**
73
+ * @inheritdoc
74
+ */
75
+ this.prefixFilterCountMap = {};
76
+ /**
77
+ * Internal property to store the private value backing the `initialSearchComplete` getter.
78
+ */
79
+ this._initialSearchCompletePromise = new Promise(res => {
80
+ this._initialSearchCompleteResolver = res;
81
+ });
82
+ /**
83
+ * @inheritdoc
84
+ */
85
+ this.checkAllTiles = () => {
86
+ this.map(model => {
87
+ const cloned = model.clone();
88
+ cloned.checked = true;
89
+ return cloned;
90
+ });
91
+ };
92
+ /**
93
+ * @inheritdoc
94
+ */
95
+ this.uncheckAllTiles = () => {
96
+ this.map(model => {
97
+ const cloned = model.clone();
98
+ cloned.checked = false;
99
+ return cloned;
100
+ });
101
+ };
102
+ /**
103
+ * @inheritdoc
104
+ */
105
+ this.removeCheckedTiles = () => {
106
+ // To make sure our data source remains page-aligned, we will offset our data source by
107
+ // the number of removed tiles, so that we can just add the offset when the infinite
108
+ // scroller queries for cell contents.
109
+ // This only matters while we're still viewing the same set of results. If the user changes
110
+ // their query/filters/sort, then the data source is overwritten and the offset cleared.
111
+ const { checkedTileModels, uncheckedTileModels } = this;
112
+ const numChecked = checkedTileModels.length;
113
+ if (numChecked === 0)
114
+ return;
115
+ this.offset += numChecked;
116
+ const newPages = {};
117
+ // Which page the remaining tile models start on, post-offset
118
+ let offsetPageNumber = Math.floor(this.offset / this.pageSize) + 1;
119
+ let indexOnPage = this.offset % this.pageSize;
120
+ // Fill the pages up to that point with empty models
121
+ for (let page = 1; page <= offsetPageNumber; page += 1) {
122
+ const remainingHidden = this.offset - this.pageSize * (page - 1);
123
+ const offsetCellsOnPage = Math.min(this.pageSize, remainingHidden);
124
+ newPages[page] = Array(offsetCellsOnPage).fill(undefined);
125
+ }
126
+ // Shift all the remaining tiles into their new positions in the data source
127
+ for (const model of uncheckedTileModels) {
128
+ if (!newPages[offsetPageNumber])
129
+ newPages[offsetPageNumber] = [];
130
+ newPages[offsetPageNumber].push(model);
131
+ indexOnPage += 1;
132
+ if (indexOnPage >= this.pageSize) {
133
+ offsetPageNumber += 1;
134
+ indexOnPage = 0;
135
+ }
136
+ }
137
+ // Swap in the new pages
138
+ this.pages = newPages;
139
+ this.numTileModels -= numChecked;
140
+ this.requestHostUpdate();
141
+ this.refreshVisibleResults();
142
+ };
143
+ // Just setting some property values
144
+ }
145
+ /**
146
+ * @inheritdoc
147
+ */
148
+ get initialSearchComplete() {
149
+ return this._initialSearchCompletePromise;
150
+ }
151
+ hostConnected() {
152
+ this.setSearchResultsLoading(this.searchResultsLoading);
153
+ this.setFacetsLoading(this.facetsLoading);
154
+ }
155
+ hostUpdate() {
156
+ // This reactive controller hook is run whenever the host component (collection-browser) performs an update.
157
+ // We check whether the host's state has changed in a way which should trigger a reset & new results fetch.
158
+ // Only the currently-installed data source should react to the update
159
+ if (!this.activeOnHost)
160
+ return;
161
+ // Copy loading states onto the host
162
+ this.setSearchResultsLoading(this.searchResultsLoading);
163
+ this.setFacetsLoading(this.facetsLoading);
164
+ // Can't perform searches without a search service
165
+ if (!this.host.searchService)
166
+ return;
167
+ // We should only reset if part of the full query state has changed
168
+ const queryKeyChanged = this.pageFetchQueryKey !== this.previousQueryKey;
169
+ if (!queryKeyChanged)
170
+ return;
171
+ // We should only reset if either:
172
+ // (a) our state permits a valid search, or
173
+ // (b) we have a blank query that we want to show empty results for
174
+ const shouldShowEmptyQueryResults = this.host.clearResultsOnEmptyQuery && this.host.baseQuery === '';
175
+ if (!(this.canPerformSearch || shouldShowEmptyQueryResults))
176
+ return;
177
+ if (this.activeOnHost)
178
+ this.host.emitQueryStateChanged();
179
+ this.handleQueryChange();
180
+ }
181
+ /**
182
+ * Returns whether this data source is the one currently installed on the host component.
183
+ */
184
+ get activeOnHost() {
185
+ return this.host.dataSource === this;
186
+ }
187
+ /**
188
+ * @inheritdoc
189
+ */
190
+ get size() {
191
+ return this.numTileModels;
192
+ }
193
+ /**
194
+ * @inheritdoc
195
+ */
196
+ reset() {
197
+ log('Resetting CB data source');
198
+ this.pages = {};
199
+ this.aggregations = {};
200
+ this.yearHistogramAggregation = undefined;
201
+ this.pageElements = undefined;
202
+ this.parentCollections = [];
203
+ this.queryErrorMessage = undefined;
204
+ this.offset = 0;
205
+ this.numTileModels = 0;
206
+ this.endOfDataReached = false;
207
+ this.queryInitialized = false;
208
+ // Invalidate any fetches in progress
209
+ this.fetchesInProgress.clear();
210
+ this.setTotalResultCount(0);
211
+ this.requestHostUpdate();
212
+ }
213
+ /**
214
+ * @inheritdoc
215
+ */
216
+ addPage(pageNum, pageTiles) {
217
+ this.pages[pageNum] = pageTiles;
218
+ this.numTileModels += pageTiles.length;
219
+ this.requestHostUpdate();
220
+ }
221
+ /**
222
+ * @inheritdoc
223
+ */
224
+ addMultiplePages(firstPageNum, tiles) {
225
+ const numPages = Math.ceil(tiles.length / this.pageSize);
226
+ for (let i = 0; i < numPages; i += 1) {
227
+ const pageStartIndex = this.pageSize * i;
228
+ this.addPage(firstPageNum + i, tiles.slice(pageStartIndex, pageStartIndex + this.pageSize));
229
+ }
230
+ const visiblePages = this.host.currentVisiblePageNumbers;
231
+ const needsReload = visiblePages.some(page => page >= firstPageNum && page <= firstPageNum + numPages);
232
+ if (needsReload) {
233
+ this.refreshVisibleResults();
234
+ }
235
+ }
236
+ /**
237
+ * @inheritdoc
238
+ */
239
+ getPage(pageNum) {
240
+ return this.pages[pageNum];
241
+ }
242
+ /**
243
+ * @inheritdoc
244
+ */
245
+ getAllPages() {
246
+ return this.pages;
247
+ }
248
+ /**
249
+ * @inheritdoc
250
+ */
251
+ hasPage(pageNum) {
252
+ return !!this.pages[pageNum];
253
+ }
254
+ /**
255
+ * @inheritdoc
256
+ */
257
+ getTileModelAt(index) {
258
+ var _a;
259
+ const pageNum = Math.floor(index / this.pageSize) + 1;
260
+ const indexOnPage = index % this.pageSize;
261
+ return (_a = this.pages[pageNum]) === null || _a === void 0 ? void 0 : _a[indexOnPage];
262
+ }
263
+ /**
264
+ * @inheritdoc
265
+ */
266
+ indexOf(tile) {
267
+ return Object.values(this.pages).flat().indexOf(tile);
268
+ }
269
+ /**
270
+ * @inheritdoc
271
+ */
272
+ getPageSize() {
273
+ return this.pageSize;
274
+ }
275
+ /**
276
+ * @inheritdoc
277
+ */
278
+ setPageSize(pageSize) {
279
+ this.reset();
280
+ this.pageSize = pageSize;
281
+ }
282
+ /**
283
+ * @inheritdoc
284
+ */
285
+ setTotalResultCount(count) {
286
+ this.totalResults = count;
287
+ if (this.activeOnHost) {
288
+ this.host.setTotalResultCount(count);
289
+ }
290
+ }
291
+ /**
292
+ * @inheritdoc
293
+ */
294
+ setFetchesSuppressed(suppressed) {
295
+ this.suppressFetches = suppressed;
296
+ }
297
+ /**
298
+ * @inheritdoc
299
+ */
300
+ async handleQueryChange() {
301
+ // Don't react to the change if fetches are suppressed for this data source
302
+ if (this.suppressFetches)
303
+ return;
304
+ this.reset();
305
+ // Reset the `initialSearchComplete` promise with a new value for the imminent search
306
+ this._initialSearchCompletePromise = new Promise(res => {
307
+ this._initialSearchCompleteResolver = res;
308
+ });
309
+ const shouldFetchFacets = !this.host.suppressFacets &&
310
+ !FACETLESS_PAGE_ELEMENTS.includes(this.host.profileElement);
311
+ // Fire the initial page & facet requests
312
+ this.queryInitialized = true;
313
+ await Promise.all([
314
+ this.doInitialPageFetch(),
315
+ shouldFetchFacets ? this.fetchFacets() : null,
316
+ ]);
317
+ // Resolve the `initialSearchComplete` promise for this search
318
+ this._initialSearchCompleteResolver(true);
319
+ }
320
+ /**
321
+ * @inheritdoc
322
+ */
323
+ map(callback) {
324
+ this.pages = Object.fromEntries(Object.entries(this.pages).map(([page, tileModels]) => [
325
+ page,
326
+ tileModels.map((model, index, array) => model ? callback(model, index, array) : model),
327
+ ]));
328
+ this.requestHostUpdate();
329
+ this.refreshVisibleResults();
330
+ }
331
+ /**
332
+ * @inheritdoc
333
+ */
334
+ get checkedTileModels() {
335
+ return this.getFilteredTileModels(model => model.checked);
336
+ }
337
+ /**
338
+ * @inheritdoc
339
+ */
340
+ get uncheckedTileModels() {
341
+ return this.getFilteredTileModels(model => !model.checked);
342
+ }
343
+ /**
344
+ * Returns a flattened, filtered array of all the tile models in the data source
345
+ * for which the given predicate returns a truthy value.
346
+ *
347
+ * @param predicate A callback function to apply on each tile model, as with Array.filter
348
+ * @returns A filtered array of tile models satisfying the predicate
349
+ */
350
+ getFilteredTileModels(predicate) {
351
+ return Object.values(this.pages)
352
+ .flat()
353
+ .filter((model, index, array) => model ? predicate(model, index, array) : false);
354
+ }
355
+ /**
356
+ * @inheritdoc
357
+ */
358
+ get canPerformSearch() {
359
+ var _a;
360
+ if (!this.host.searchService)
361
+ return false;
362
+ const trimmedQuery = (_a = this.host.baseQuery) === null || _a === void 0 ? void 0 : _a.trim();
363
+ const hasNonEmptyQuery = !!trimmedQuery;
364
+ const isCollectionSearch = !!this.host.withinCollection;
365
+ const isProfileSearch = !!this.host.withinProfile;
366
+ const hasProfileElement = !!this.host.profileElement;
367
+ const isMetadataSearch = this.host.searchType === SearchType.METADATA;
368
+ // Metadata searches within a collection/profile are allowed to have no query.
369
+ // Otherwise, a non-empty query must be set.
370
+ return (hasNonEmptyQuery ||
371
+ (isCollectionSearch && isMetadataSearch) ||
372
+ (isProfileSearch && hasProfileElement && isMetadataSearch));
373
+ }
374
+ /**
375
+ * Sets the state for whether the initial set of search results for the
376
+ * current query is loading
377
+ */
378
+ setSearchResultsLoading(loading) {
379
+ this.searchResultsLoading = loading;
380
+ if (this.activeOnHost) {
381
+ this.host.setSearchResultsLoading(loading);
382
+ }
383
+ }
384
+ /**
385
+ * Sets the state for whether the facets for a query is loading
386
+ */
387
+ setFacetsLoading(loading) {
388
+ this.facetsLoading = loading;
389
+ if (this.activeOnHost) {
390
+ this.host.setFacetsLoading(loading);
391
+ }
392
+ }
393
+ /**
394
+ * Requests that the host perform an update, provided this data
395
+ * source is actively installed on it.
396
+ */
397
+ requestHostUpdate() {
398
+ if (this.activeOnHost) {
399
+ this.host.requestUpdate();
400
+ }
401
+ }
402
+ /**
403
+ * Requests that the host refresh its visible tiles, provided this
404
+ * data source is actively installed on it.
405
+ */
406
+ refreshVisibleResults() {
407
+ if (this.activeOnHost) {
408
+ this.host.refreshVisibleResults();
409
+ }
410
+ }
411
+ /**
412
+ * The query key is a string that uniquely identifies the current search.
413
+ * It consists of:
414
+ * - The current base query
415
+ * - The current collection/profile target & page element
416
+ * - The current search type
417
+ * - Any currently-applied facets
418
+ * - Any currently-applied date range
419
+ * - Any currently-applied prefix filters
420
+ * - The current sort options
421
+ *
422
+ * This lets us internally keep track of queries so we don't persist data that's
423
+ * no longer relevant. Not meant to be human-readable.
424
+ */
425
+ get pageFetchQueryKey() {
426
+ var _a, _b, _c;
427
+ const profileKey = `pf;${this.host.withinProfile}--pe;${this.host.profileElement}`;
428
+ const pageTarget = (_a = this.host.withinCollection) !== null && _a !== void 0 ? _a : profileKey;
429
+ const sortField = (_b = this.host.selectedSort) !== null && _b !== void 0 ? _b : 'none';
430
+ const sortDirection = (_c = this.host.sortDirection) !== null && _c !== void 0 ? _c : 'none';
431
+ return `fq:${this.fullQuery}-pt:${pageTarget}-st:${this.host.searchType}-sf:${sortField}-sd:${sortDirection}`;
432
+ }
433
+ /**
434
+ * Similar to `pageFetchQueryKey` above, but excludes sort fields since they
435
+ * are not relevant in determining aggregation queries.
436
+ */
437
+ get facetFetchQueryKey() {
438
+ var _a;
439
+ const profileKey = `pf;${this.host.withinProfile}--pe;${this.host.profileElement}`;
440
+ const pageTarget = (_a = this.host.withinCollection) !== null && _a !== void 0 ? _a : profileKey;
441
+ return `fq:${this.fullQuery}-pt:${pageTarget}-st:${this.host.searchType}`;
442
+ }
443
+ /**
444
+ * Constructs a search service FilterMap object from the combination of
445
+ * all the currently-applied filters. This includes any facets, letter
446
+ * filters, and date range.
447
+ */
448
+ get filterMap() {
449
+ const builder = new FilterMapBuilder();
450
+ // Add the date range, if applicable
451
+ if (this.host.minSelectedDate) {
452
+ builder.addFilter('year', this.host.minSelectedDate, FilterConstraint.GREATER_OR_EQUAL);
453
+ }
454
+ if (this.host.maxSelectedDate) {
455
+ builder.addFilter('year', this.host.maxSelectedDate, FilterConstraint.LESS_OR_EQUAL);
456
+ }
457
+ // Add any selected facets
458
+ if (this.host.selectedFacets) {
459
+ for (const [facetName, facetValues] of Object.entries(this.host.selectedFacets)) {
460
+ const { name, values } = this.prepareFacetForFetch(facetName, facetValues);
461
+ for (const [value, bucket] of Object.entries(values)) {
462
+ let constraint;
463
+ if (bucket.state === 'selected') {
464
+ constraint = FilterConstraint.INCLUDE;
465
+ }
466
+ else if (bucket.state === 'hidden') {
467
+ constraint = FilterConstraint.EXCLUDE;
468
+ }
469
+ if (constraint) {
470
+ builder.addFilter(name, value, constraint);
471
+ }
472
+ }
473
+ }
474
+ }
475
+ // Add any letter filters
476
+ if (this.host.selectedTitleFilter) {
477
+ builder.addFilter('firstTitle', this.host.selectedTitleFilter, FilterConstraint.INCLUDE);
478
+ }
479
+ if (this.host.selectedCreatorFilter) {
480
+ builder.addFilter('firstCreator', this.host.selectedCreatorFilter, FilterConstraint.INCLUDE);
481
+ }
482
+ const filterMap = builder.build();
483
+ return filterMap;
484
+ }
485
+ /**
486
+ * Produces a compact unique ID for a search request that can help with debugging
487
+ * on the backend by making related requests easier to trace through different services.
488
+ * (e.g., tying the hits/aggregations requests for the same page back to a single hash).
489
+ *
490
+ * @param params The search service parameters for the request
491
+ * @param kind The kind of request (hits-only, aggregations-only, or both)
492
+ * @returns A Promise resolving to the uid to apply to the request
493
+ */
494
+ async requestUID(params, kind) {
495
+ var _a;
496
+ const paramsToHash = JSON.stringify({
497
+ pageType: params.pageType,
498
+ pageTarget: params.pageTarget,
499
+ query: params.query,
500
+ fields: params.fields,
501
+ filters: params.filters,
502
+ sort: params.sort,
503
+ searchType: this.host.searchType,
504
+ });
505
+ const fullQueryHash = (await sha1(paramsToHash)).slice(0, 20); // First 80 bits of SHA-1 are plenty for this
506
+ const sessionId = (await this.host.getSessionId()).slice(0, 20); // Likewise
507
+ const page = (_a = params.page) !== null && _a !== void 0 ? _a : 0;
508
+ const kindPrefix = kind.charAt(0); // f = full, h = hits, a = aggregations
509
+ const currentTime = Date.now();
510
+ return `R:${fullQueryHash}-S:${sessionId}-P:${page}-K:${kindPrefix}-T:${currentTime}`;
511
+ }
512
+ /**
513
+ * @inheritdoc
514
+ */
515
+ get pageSpecifierParams() {
516
+ if (this.host.withinCollection) {
517
+ return {
518
+ pageType: 'collection_details',
519
+ pageTarget: this.host.withinCollection,
520
+ };
521
+ }
522
+ if (this.host.withinProfile) {
523
+ return {
524
+ pageType: 'account_details',
525
+ pageTarget: this.host.withinProfile,
526
+ pageElements: this.host.profileElement
527
+ ? [this.host.profileElement]
528
+ : [],
529
+ };
530
+ }
531
+ return null;
532
+ }
533
+ /**
534
+ * The full query, including year facets and date range clauses
535
+ */
536
+ get fullQuery() {
537
+ var _a, _b;
538
+ let fullQuery = (_b = (_a = this.host.baseQuery) === null || _a === void 0 ? void 0 : _a.trim()) !== null && _b !== void 0 ? _b : '';
539
+ const { facetQuery, dateRangeQueryClause, sortFilterQueries } = this;
540
+ if (facetQuery) {
541
+ fullQuery += ` AND ${facetQuery}`;
542
+ }
543
+ if (dateRangeQueryClause) {
544
+ fullQuery += ` AND ${dateRangeQueryClause}`;
545
+ }
546
+ if (sortFilterQueries) {
547
+ fullQuery += ` AND ${sortFilterQueries}`;
548
+ }
549
+ return fullQuery.trim();
550
+ }
551
+ /**
552
+ * Generates a query string representing the current set of applied facets
553
+ *
554
+ * Example: `mediatype:("collection" OR "audio" OR -"etree") AND year:("2000" OR "2001")`
555
+ */
556
+ get facetQuery() {
557
+ var _a;
558
+ if (!this.host.selectedFacets)
559
+ return undefined;
560
+ const facetClauses = [];
561
+ for (const [facetName, facetValues] of Object.entries(this.host.selectedFacets)) {
562
+ facetClauses.push(this.buildFacetClause(facetName, facetValues));
563
+ }
564
+ return (_a = this.joinFacetClauses(facetClauses)) === null || _a === void 0 ? void 0 : _a.trim();
565
+ }
566
+ get dateRangeQueryClause() {
567
+ if (!this.host.minSelectedDate || !this.host.maxSelectedDate) {
568
+ return undefined;
569
+ }
570
+ return `year:[${this.host.minSelectedDate} TO ${this.host.maxSelectedDate}]`;
571
+ }
572
+ get sortFilterQueries() {
573
+ const queries = [this.titleQuery, this.creatorQuery];
574
+ return queries.filter(q => q).join(' AND ');
575
+ }
576
+ /**
577
+ * Returns a query clause identifying the currently selected title filter,
578
+ * e.g., `firstTitle:X`.
579
+ */
580
+ get titleQuery() {
581
+ return this.host.selectedTitleFilter
582
+ ? `firstTitle:${this.host.selectedTitleFilter}`
583
+ : undefined;
584
+ }
585
+ /**
586
+ * Returns a query clause identifying the currently selected creator filter,
587
+ * e.g., `firstCreator:X`.
588
+ */
589
+ get creatorQuery() {
590
+ return this.host.selectedCreatorFilter
591
+ ? `firstCreator:${this.host.selectedCreatorFilter}`
592
+ : undefined;
593
+ }
594
+ /**
595
+ * Builds an OR-joined facet clause for the given facet name and values.
596
+ *
597
+ * E.g., for name `subject` and values
598
+ * `{ foo: { state: 'selected' }, bar: { state: 'hidden' } }`
599
+ * this will produce the clause
600
+ * `subject:("foo" OR -"bar")`.
601
+ *
602
+ * @param facetName The facet type (e.g., 'collection')
603
+ * @param facetValues The facet buckets, mapped by their keys
604
+ */
605
+ buildFacetClause(facetName, facetValues) {
606
+ const { name: facetQueryName, values } = this.prepareFacetForFetch(facetName, facetValues);
607
+ const facetEntries = Object.entries(values);
608
+ if (facetEntries.length === 0)
609
+ return '';
610
+ const facetValuesArray = [];
611
+ for (const [key, facetData] of facetEntries) {
612
+ const plusMinusPrefix = facetData.state === 'hidden' ? '-' : '';
613
+ facetValuesArray.push(`${plusMinusPrefix}"${key}"`);
614
+ }
615
+ const valueQuery = facetValuesArray.join(` OR `);
616
+ return `${facetQueryName}:(${valueQuery})`;
617
+ }
618
+ /**
619
+ * Handles some special pre-request normalization steps for certain facet types
620
+ * that require them.
621
+ *
622
+ * @param facetName The name of the facet type (e.g., 'language')
623
+ * @param facetValues An array of values for that facet type
624
+ */
625
+ prepareFacetForFetch(facetName, facetValues) {
626
+ // eslint-disable-next-line prefer-const
627
+ let [normalizedName, normalizedValues] = [facetName, facetValues];
628
+ // The full "search engine" name of the lending field is "lending___status"
629
+ if (facetName === 'lending') {
630
+ normalizedName = 'lending___status';
631
+ }
632
+ return {
633
+ name: normalizedName,
634
+ values: normalizedValues,
635
+ };
636
+ }
637
+ /**
638
+ * Takes an array of facet clauses, and combines them into a
639
+ * full AND-joined facet query string. Empty clauses are ignored.
640
+ */
641
+ joinFacetClauses(facetClauses) {
642
+ const nonEmptyFacetClauses = facetClauses.filter(clause => clause.length > 0);
643
+ return nonEmptyFacetClauses.length > 0
644
+ ? `(${nonEmptyFacetClauses.join(' AND ')})`
645
+ : undefined;
646
+ }
647
+ /**
648
+ * Fires a backend request to fetch a set of aggregations (representing UI facets) for
649
+ * the current search state.
650
+ */
651
+ async fetchFacets() {
652
+ var _a, _b, _c, _d, _e, _f, _g, _h, _j;
653
+ const trimmedQuery = (_a = this.host.baseQuery) === null || _a === void 0 ? void 0 : _a.trim();
654
+ if (!this.canPerformSearch)
655
+ return;
656
+ const { facetFetchQueryKey } = this;
657
+ if (this.fetchesInProgress.has(facetFetchQueryKey))
658
+ return;
659
+ this.fetchesInProgress.add(facetFetchQueryKey);
660
+ const sortParams = this.host.sortParam ? [this.host.sortParam] : [];
661
+ const params = {
662
+ ...this.pageSpecifierParams,
663
+ query: trimmedQuery || '',
664
+ rows: 0,
665
+ filters: this.filterMap,
666
+ // Fetch a few extra buckets beyond the 6 we show, in case some get suppressed
667
+ aggregationsSize: 10,
668
+ // Note: we don't need an aggregations param to fetch the default aggregations from the PPS.
669
+ // The default aggregations for the search_results page type should be what we need here.
670
+ };
671
+ params.uid = await this.requestUID({ ...params, sort: sortParams }, 'aggregations');
672
+ this.setFacetsLoading(true);
673
+ const searchResponse = await ((_b = this.host.searchService) === null || _b === void 0 ? void 0 : _b.search(params, this.host.searchType));
674
+ const success = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.success;
675
+ // This is checking to see if the query has changed since the data was fetched.
676
+ // If so, we just want to discard this set of aggregations because they are
677
+ // likely no longer valid for the newer query.
678
+ const queryChangedSinceFetch = !this.fetchesInProgress.has(facetFetchQueryKey);
679
+ this.fetchesInProgress.delete(facetFetchQueryKey);
680
+ if (queryChangedSinceFetch)
681
+ return;
682
+ if (!success) {
683
+ const errorMsg = (_c = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.error) === null || _c === void 0 ? void 0 : _c.message;
684
+ const detailMsg = (_e = (_d = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.error) === null || _d === void 0 ? void 0 : _d.details) === null || _e === void 0 ? void 0 : _e.message;
685
+ if (!errorMsg && !detailMsg) {
686
+ // @ts-ignore: Property 'Sentry' does not exist on type 'Window & typeof globalThis'
687
+ (_g = (_f = window === null || window === void 0 ? void 0 : window.Sentry) === null || _f === void 0 ? void 0 : _f.captureMessage) === null || _g === void 0 ? void 0 : _g.call(_f, 'Missing or malformed facet response from backend', 'error');
688
+ }
689
+ this.setFacetsLoading(false);
690
+ return;
691
+ }
692
+ const { aggregations, collectionTitles } = success.response;
693
+ this.aggregations = aggregations;
694
+ if (collectionTitles) {
695
+ for (const [id, title] of Object.entries(collectionTitles)) {
696
+ this.collectionTitles.set(id, title);
697
+ }
698
+ }
699
+ this.yearHistogramAggregation =
700
+ (_j = (_h = success === null || success === void 0 ? void 0 : success.response) === null || _h === void 0 ? void 0 : _h.aggregations) === null || _j === void 0 ? void 0 : _j.year_histogram;
701
+ this.setFacetsLoading(false);
702
+ this.requestHostUpdate();
703
+ }
704
+ /**
705
+ * Performs the initial page fetch(es) for the current search state.
706
+ */
707
+ async doInitialPageFetch() {
708
+ this.setSearchResultsLoading(true);
709
+ // Try to batch 2 initial page requests when possible
710
+ await this.fetchPage(this.host.initialPageNumber, 2);
711
+ }
712
+ /**
713
+ * Fetches one or more pages of results and updates the data source.
714
+ *
715
+ * @param pageNumber The page number to fetch
716
+ * @param numInitialPages If this is an initial page fetch (`pageNumber = 1`),
717
+ * specifies how many pages to batch together in one request. Ignored
718
+ * if `pageNumber != 1`, defaulting to a single page.
719
+ */
720
+ async fetchPage(pageNumber, numInitialPages = 1) {
721
+ var _a, _b, _c, _d, _e, _f, _g, _h, _j;
722
+ const trimmedQuery = (_a = this.host.baseQuery) === null || _a === void 0 ? void 0 : _a.trim();
723
+ if (!this.canPerformSearch)
724
+ return;
725
+ // if we already have data, don't fetch again
726
+ if (this.hasPage(pageNumber))
727
+ return;
728
+ if (this.endOfDataReached)
729
+ return;
730
+ // Batch multiple initial page requests together if needed (e.g., can request
731
+ // pages 1 and 2 together in a single request).
732
+ let numPages = pageNumber === 1 ? numInitialPages : 1;
733
+ const numRows = this.pageSize * numPages;
734
+ // if a fetch is already in progress for this query and page, don't fetch again
735
+ const { pageFetchQueryKey } = this;
736
+ const currentPageKey = `${pageFetchQueryKey}-p:${pageNumber}`;
737
+ if (this.fetchesInProgress.has(currentPageKey))
738
+ return;
739
+ for (let i = 0; i < numPages; i += 1) {
740
+ this.fetchesInProgress.add(`${pageFetchQueryKey}-p:${pageNumber + i}`);
741
+ }
742
+ this.previousQueryKey = pageFetchQueryKey;
743
+ let sortParams = this.host.sortParam ? [this.host.sortParam] : [];
744
+ // TODO eventually the PPS should handle these defaults natively
745
+ const isDefaultProfileSort = this.host.withinProfile && this.host.selectedSort === SortField.default;
746
+ if (isDefaultProfileSort && this.host.defaultSortParam) {
747
+ const sortOption = SORT_OPTIONS[this.host.defaultSortParam.field];
748
+ if (sortOption.searchServiceKey) {
749
+ sortParams = [
750
+ {
751
+ field: sortOption.searchServiceKey,
752
+ direction: 'desc',
753
+ },
754
+ ];
755
+ }
756
+ }
757
+ const params = {
758
+ ...this.pageSpecifierParams,
759
+ query: trimmedQuery || '',
760
+ page: pageNumber,
761
+ rows: numRows,
762
+ sort: sortParams,
763
+ filters: this.filterMap,
764
+ aggregations: { omit: true },
765
+ };
766
+ params.uid = await this.requestUID(params, 'hits');
767
+ log('=== FIRING PAGE REQUEST ===', params);
768
+ const searchResponse = await ((_b = this.host.searchService) === null || _b === void 0 ? void 0 : _b.search(params, this.host.searchType));
769
+ log('=== RECEIVED PAGE RESPONSE IN CB ===', searchResponse);
770
+ const success = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.success;
771
+ // This is checking to see if the fetch has been invalidated since it was fired off.
772
+ // If so, we just want to discard the response since it is for an obsolete query state.
773
+ if (!this.fetchesInProgress.has(currentPageKey))
774
+ return;
775
+ for (let i = 0; i < numPages; i += 1) {
776
+ this.fetchesInProgress.delete(`${pageFetchQueryKey}-p:${pageNumber + i}`);
777
+ }
778
+ if (!success) {
779
+ const errorMsg = (_c = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.error) === null || _c === void 0 ? void 0 : _c.message;
780
+ const detailMsg = (_e = (_d = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.error) === null || _d === void 0 ? void 0 : _d.details) === null || _e === void 0 ? void 0 : _e.message;
781
+ this.queryErrorMessage = `${errorMsg !== null && errorMsg !== void 0 ? errorMsg : ''}${detailMsg ? `; ${detailMsg}` : ''}`;
782
+ if (!this.queryErrorMessage) {
783
+ this.queryErrorMessage = 'Missing or malformed response from backend';
784
+ // @ts-ignore: Property 'Sentry' does not exist on type 'Window & typeof globalThis'
785
+ (_g = (_f = window === null || window === void 0 ? void 0 : window.Sentry) === null || _f === void 0 ? void 0 : _f.captureMessage) === null || _g === void 0 ? void 0 : _g.call(_f, this.queryErrorMessage, 'error');
786
+ }
787
+ this.setSearchResultsLoading(false);
788
+ this.requestHostUpdate();
789
+ return;
790
+ }
791
+ this.setTotalResultCount(success.response.totalResults - this.offset);
792
+ if (this.activeOnHost && this.totalResults === 0) {
793
+ // display event to offshoot when result count is zero.
794
+ this.host.emitEmptyResults();
795
+ }
796
+ if (this.host.withinCollection) {
797
+ this.collectionExtraInfo = success.response.collectionExtraInfo;
798
+ // For collections, we want the UI to respect the default sort option
799
+ // which can be specified in metadata, or otherwise assumed to be week:desc
800
+ if (this.activeOnHost) {
801
+ this.host.applyDefaultCollectionSort(this.collectionExtraInfo);
802
+ }
803
+ if (this.collectionExtraInfo) {
804
+ this.parentCollections = [].concat((_j = (_h = this.collectionExtraInfo.public_metadata) === null || _h === void 0 ? void 0 : _h.collection) !== null && _j !== void 0 ? _j : []);
805
+ }
806
+ }
807
+ else if (this.host.withinProfile) {
808
+ this.accountExtraInfo = success.response.accountExtraInfo;
809
+ this.pageElements = success.response.pageElements;
810
+ }
811
+ const { results, collectionTitles } = success.response;
812
+ if (results && results.length > 0) {
813
+ // Load any collection titles present on the response into the cache,
814
+ // or queue up preload fetches for them if none were present.
815
+ if (collectionTitles) {
816
+ for (const [id, title] of Object.entries(collectionTitles)) {
817
+ this.collectionTitles.set(id, title);
818
+ }
819
+ }
820
+ // Update the data source for each returned page.
821
+ // For loans and web archives, we must account for receiving more pages than we asked for.
822
+ if (this.host.profileElement === 'lending' ||
823
+ this.host.profileElement === 'web_archives') {
824
+ numPages = Math.ceil(results.length / this.pageSize);
825
+ this.endOfDataReached = true;
826
+ if (this.activeOnHost)
827
+ this.host.setTileCount(this.totalResults);
828
+ }
829
+ for (let i = 0; i < numPages; i += 1) {
830
+ const pageStartIndex = this.pageSize * i;
831
+ this.addFetchedResultsToDataSource(pageNumber + i, results.slice(pageStartIndex, pageStartIndex + this.pageSize));
832
+ }
833
+ }
834
+ // When we reach the end of the data, we can set the infinite scroller's
835
+ // item count to the real total number of results (rather than the
836
+ // temporary estimates based on pages rendered so far).
837
+ const resultCountDiscrepancy = numRows - results.length;
838
+ if (resultCountDiscrepancy > 0) {
839
+ this.endOfDataReached = true;
840
+ if (this.activeOnHost)
841
+ this.host.setTileCount(this.totalResults);
842
+ }
843
+ this.setSearchResultsLoading(false);
844
+ this.requestHostUpdate();
845
+ }
846
+ /**
847
+ * Update the datasource from the fetch response
848
+ *
849
+ * @param pageNumber
850
+ * @param results
851
+ */
852
+ addFetchedResultsToDataSource(pageNumber, results) {
853
+ const tiles = [];
854
+ results === null || results === void 0 ? void 0 : results.forEach(result => {
855
+ if (!result.identifier)
856
+ return;
857
+ tiles.push(new TileModel(result));
858
+ });
859
+ this.addPage(pageNumber, tiles);
860
+ const visiblePages = this.host.currentVisiblePageNumbers;
861
+ const needsReload = visiblePages.includes(pageNumber);
862
+ if (needsReload) {
863
+ this.refreshVisibleResults();
864
+ }
865
+ }
866
+ /**
867
+ * Fetches the aggregation buckets for the given prefix filter type.
868
+ */
869
+ async fetchPrefixFilterBuckets(filterType) {
870
+ var _a, _b, _c, _d, _e, _f, _g;
871
+ const trimmedQuery = (_a = this.host.baseQuery) === null || _a === void 0 ? void 0 : _a.trim();
872
+ if (!this.canPerformSearch)
873
+ return [];
874
+ const filterAggregationKey = prefixFilterAggregationKeys[filterType];
875
+ const sortParams = this.host.sortParam ? [this.host.sortParam] : [];
876
+ const params = {
877
+ ...this.pageSpecifierParams,
878
+ query: trimmedQuery || '',
879
+ rows: 0,
880
+ filters: this.filterMap,
881
+ // Only fetch the firstTitle or firstCreator aggregation
882
+ aggregations: { simpleParams: [filterAggregationKey] },
883
+ // Fetch all 26 letter buckets
884
+ aggregationsSize: 26,
885
+ };
886
+ params.uid = await this.requestUID({ ...params, sort: sortParams }, 'aggregations');
887
+ const searchResponse = await ((_b = this.host.searchService) === null || _b === void 0 ? void 0 : _b.search(params, this.host.searchType));
888
+ return ((_g = (_f = (_e = (_d = (_c = searchResponse === null || searchResponse === void 0 ? void 0 : searchResponse.success) === null || _c === void 0 ? void 0 : _c.response) === null || _d === void 0 ? void 0 : _d.aggregations) === null || _e === void 0 ? void 0 : _e[filterAggregationKey]) === null || _f === void 0 ? void 0 : _f.buckets) !== null && _g !== void 0 ? _g : []);
889
+ }
890
+ /**
891
+ * Fetches and caches the prefix filter counts for the given filter type.
892
+ */
893
+ async updatePrefixFilterCounts(filterType) {
894
+ const { facetFetchQueryKey } = this;
895
+ const buckets = await this.fetchPrefixFilterBuckets(filterType);
896
+ // Don't update the filter counts for an outdated query (if it has been changed
897
+ // since we sent the request)
898
+ const queryChangedSinceFetch = facetFetchQueryKey !== this.facetFetchQueryKey;
899
+ if (queryChangedSinceFetch)
900
+ return;
901
+ // Unpack the aggregation buckets into a simple map like { 'A': 50, 'B': 25, ... }
902
+ this.prefixFilterCountMap = { ...this.prefixFilterCountMap }; // Clone the object to trigger an update
903
+ this.prefixFilterCountMap[filterType] = buckets.reduce((acc, bucket) => {
904
+ acc[bucket.key.toUpperCase()] = bucket.doc_count;
905
+ return acc;
906
+ }, {});
907
+ this.requestHostUpdate();
908
+ }
909
+ /**
910
+ * @inheritdoc
911
+ */
912
+ async updatePrefixFiltersForCurrentSort() {
913
+ if (['title', 'creator'].includes(this.host.selectedSort)) {
914
+ const filterType = this.host.selectedSort;
915
+ if (!this.prefixFilterCountMap[filterType]) {
916
+ this.updatePrefixFilterCounts(filterType);
917
+ }
918
+ }
919
+ }
920
+ /**
921
+ * @inheritdoc
922
+ */
923
+ refreshLetterCounts() {
924
+ if (Object.keys(this.prefixFilterCountMap).length > 0) {
925
+ this.prefixFilterCountMap = {};
926
+ }
927
+ this.updatePrefixFiltersForCurrentSort();
928
+ this.requestHostUpdate();
929
+ }
930
+ }
931
931
  //# sourceMappingURL=collection-browser-data-source.js.map