@flamingo-stack/openframe-frontend-core 0.0.295 → 0.0.296-snapshot.20260621021605

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 (291) hide show
  1. package/README.md +9 -0
  2. package/dist/{chunk-7RIYT7ZH.js → chunk-2QG57XOJ.js} +1067 -205
  3. package/dist/chunk-2QG57XOJ.js.map +1 -0
  4. package/dist/{chunk-7KXD7CWD.js → chunk-3JIQVE7T.js} +9 -15
  5. package/dist/{chunk-7KXD7CWD.js.map → chunk-3JIQVE7T.js.map} +1 -1
  6. package/dist/{chunk-FT4FCV7L.cjs → chunk-4PSQS3SW.cjs} +7 -9
  7. package/dist/chunk-4PSQS3SW.cjs.map +1 -0
  8. package/dist/{chunk-OOKKGOPQ.js → chunk-4TLE6VLU.js} +30 -24
  9. package/dist/chunk-4TLE6VLU.js.map +1 -0
  10. package/dist/{chunk-6IBA2MQV.cjs → chunk-53FUMSZ5.cjs} +40 -46
  11. package/dist/chunk-53FUMSZ5.cjs.map +1 -0
  12. package/dist/{chunk-D3LEFMOA.cjs → chunk-54KNMC2R.cjs} +3 -3
  13. package/dist/{chunk-D3LEFMOA.cjs.map → chunk-54KNMC2R.cjs.map} +1 -1
  14. package/dist/{chunk-EYEW6PTA.cjs → chunk-6C526VNN.cjs} +358 -118
  15. package/dist/chunk-6C526VNN.cjs.map +1 -0
  16. package/dist/{chunk-5O6N3BKR.cjs → chunk-7OVGB2DQ.cjs} +19 -25
  17. package/dist/chunk-7OVGB2DQ.cjs.map +1 -0
  18. package/dist/{chunk-6GCI7JOE.js → chunk-AD6C23QY.js} +8 -7
  19. package/dist/{chunk-6GCI7JOE.js.map → chunk-AD6C23QY.js.map} +1 -1
  20. package/dist/chunk-F5OB2YAL.cjs +144 -0
  21. package/dist/chunk-F5OB2YAL.cjs.map +1 -0
  22. package/dist/chunk-FBWXMMRB.cjs +2 -0
  23. package/dist/chunk-FBWXMMRB.cjs.map +1 -0
  24. package/dist/{chunk-YIGPRLQY.cjs → chunk-FCDQNTDG.cjs} +21 -20
  25. package/dist/chunk-FCDQNTDG.cjs.map +1 -0
  26. package/dist/{chunk-XXI7BNB6.cjs → chunk-FQOTC3UU.cjs} +321 -18
  27. package/dist/chunk-FQOTC3UU.cjs.map +1 -0
  28. package/dist/{chunk-INDQMNP6.cjs → chunk-GUTS7HGA.cjs} +11658 -2146
  29. package/dist/chunk-GUTS7HGA.cjs.map +1 -0
  30. package/dist/chunk-GZ4C3XW6.js +2 -0
  31. package/dist/chunk-GZ4C3XW6.js.map +1 -0
  32. package/dist/{chunk-HOVJGXF7.js → chunk-IL47XWV5.js} +8 -14
  33. package/dist/{chunk-HOVJGXF7.js.map → chunk-IL47XWV5.js.map} +1 -1
  34. package/dist/{chunk-LCNMR277.js → chunk-IZ7JSBFP.js} +1 -1
  35. package/dist/chunk-IZ7JSBFP.js.map +1 -0
  36. package/dist/{chunk-5IJ46KAV.js → chunk-JALO4TAZ.js} +360 -57
  37. package/dist/chunk-JALO4TAZ.js.map +1 -0
  38. package/dist/{chunk-AQOWFSMB.cjs → chunk-L6PSSIUQ.cjs} +1 -1
  39. package/dist/chunk-L6PSSIUQ.cjs.map +1 -0
  40. package/dist/{chunk-J3RDKZ32.js → chunk-L7ULJKG7.js} +6 -10
  41. package/dist/{chunk-J3RDKZ32.js.map → chunk-L7ULJKG7.js.map} +1 -1
  42. package/dist/{chunk-6BZEAPNT.js → chunk-PC746XCO.js} +15120 -5608
  43. package/dist/chunk-PC746XCO.js.map +1 -0
  44. package/dist/{chunk-3ZXUQQL4.js → chunk-PI4WSYQV.js} +2 -2
  45. package/dist/{chunk-E4XABBSU.js → chunk-PWQUAVA3.js} +338 -98
  46. package/dist/chunk-PWQUAVA3.js.map +1 -0
  47. package/dist/chunk-SA2WPJVO.js +144 -0
  48. package/dist/chunk-SA2WPJVO.js.map +1 -0
  49. package/dist/{chunk-ETACGX2A.cjs → chunk-UNVE2SDJ.cjs} +37 -31
  50. package/dist/chunk-UNVE2SDJ.cjs.map +1 -0
  51. package/dist/{chunk-5E2HOSSH.cjs → chunk-WMSTJAZT.cjs} +913 -51
  52. package/dist/chunk-WMSTJAZT.cjs.map +1 -0
  53. package/dist/{chunk-EJXHZX2E.js → chunk-X4DOXQRT.js} +4 -6
  54. package/dist/{chunk-EJXHZX2E.js.map → chunk-X4DOXQRT.js.map} +1 -1
  55. package/dist/{chunk-A2YL7QRX.cjs → chunk-YBYI62OE.cjs} +33 -37
  56. package/dist/chunk-YBYI62OE.cjs.map +1 -0
  57. package/dist/components/case-studies/index.cjs +126 -0
  58. package/dist/components/case-studies/index.cjs.map +1 -0
  59. package/dist/components/case-studies/index.d.ts +2 -0
  60. package/dist/components/case-studies/index.d.ts.map +1 -0
  61. package/dist/components/case-studies/index.js +126 -0
  62. package/dist/components/case-studies/index.js.map +1 -0
  63. package/dist/components/case-studies/share-experience-section.d.ts +48 -0
  64. package/dist/components/case-studies/share-experience-section.d.ts.map +1 -0
  65. package/dist/components/chat/chat-container.d.ts.map +1 -1
  66. package/dist/components/chat/error-message-display.d.ts.map +1 -1
  67. package/dist/components/chat/index.cjs +8 -18
  68. package/dist/components/chat/index.cjs.map +1 -1
  69. package/dist/components/chat/index.js +75 -85
  70. package/dist/components/chat/types/component.types.d.ts +2 -0
  71. package/dist/components/chat/types/component.types.d.ts.map +1 -1
  72. package/dist/components/contact/index.cjs +8 -15
  73. package/dist/components/contact/index.cjs.map +1 -1
  74. package/dist/components/contact/index.js +7 -14
  75. package/dist/components/docs/doc-viewer.d.ts +39 -2
  76. package/dist/components/docs/doc-viewer.d.ts.map +1 -1
  77. package/dist/components/docs/docs-hub-page.d.ts +46 -0
  78. package/dist/components/docs/docs-hub-page.d.ts.map +1 -0
  79. package/dist/components/docs/index.cjs +17 -9
  80. package/dist/components/docs/index.cjs.map +1 -1
  81. package/dist/components/docs/index.d.ts +4 -0
  82. package/dist/components/docs/index.d.ts.map +1 -1
  83. package/dist/components/docs/index.js +16 -8
  84. package/dist/components/docs/skeletons.d.ts +32 -0
  85. package/dist/components/docs/skeletons.d.ts.map +1 -0
  86. package/dist/components/docs/use-docs-resolve-link.d.ts +20 -0
  87. package/dist/components/docs/use-docs-resolve-link.d.ts.map +1 -0
  88. package/dist/components/docs/use-document-tree.d.ts.map +1 -1
  89. package/dist/components/embeds/embed-container.d.ts +37 -0
  90. package/dist/components/embeds/embed-container.d.ts.map +1 -0
  91. package/dist/components/embeds/embed-iframe.d.ts.map +1 -1
  92. package/dist/components/embeds/file-download-card.d.ts +18 -0
  93. package/dist/components/embeds/file-download-card.d.ts.map +1 -0
  94. package/dist/components/embeds/index.cjs +38 -15
  95. package/dist/components/embeds/index.cjs.map +1 -1
  96. package/dist/components/embeds/index.d.ts +8 -0
  97. package/dist/components/embeds/index.d.ts.map +1 -1
  98. package/dist/components/embeds/index.js +40 -17
  99. package/dist/components/embeds/linkedin-embed-client.d.ts +8 -0
  100. package/dist/components/embeds/linkedin-embed-client.d.ts.map +1 -0
  101. package/dist/components/embeds/markdown-image.d.ts +5 -0
  102. package/dist/components/embeds/markdown-image.d.ts.map +1 -0
  103. package/dist/components/embeds/reddit-embed-client.d.ts +7 -0
  104. package/dist/components/embeds/reddit-embed-client.d.ts.map +1 -0
  105. package/dist/components/embeds/rich-markdown-runtime.d.ts +46 -0
  106. package/dist/components/embeds/rich-markdown-runtime.d.ts.map +1 -0
  107. package/dist/components/embeds/twitter-embed-client.d.ts +8 -0
  108. package/dist/components/embeds/twitter-embed-client.d.ts.map +1 -0
  109. package/dist/components/faq/index.cjs +9 -16
  110. package/dist/components/faq/index.cjs.map +1 -1
  111. package/dist/components/faq/index.js +8 -15
  112. package/dist/components/features/index.cjs +8 -16
  113. package/dist/components/features/index.cjs.map +1 -1
  114. package/dist/components/features/index.js +24 -32
  115. package/dist/components/features/notifications/notification-drawer.d.ts.map +1 -1
  116. package/dist/components/features/notifications/notifications-context.d.ts +5 -1
  117. package/dist/components/features/notifications/notifications-context.d.ts.map +1 -1
  118. package/dist/components/index.cjs +257 -452
  119. package/dist/components/index.cjs.map +1 -1
  120. package/dist/components/index.js +781 -976
  121. package/dist/components/index.js.map +1 -1
  122. package/dist/components/layout/page-header.d.ts +78 -0
  123. package/dist/components/layout/page-header.d.ts.map +1 -0
  124. package/dist/components/layout/page-layout.d.ts +10 -1
  125. package/dist/components/layout/page-layout.d.ts.map +1 -1
  126. package/dist/components/layout/page-with-header.d.ts +67 -0
  127. package/dist/components/layout/page-with-header.d.ts.map +1 -0
  128. package/dist/components/layout/title-block.d.ts +17 -1
  129. package/dist/components/layout/title-block.d.ts.map +1 -1
  130. package/dist/components/navigation/index.cjs +7 -15
  131. package/dist/components/navigation/index.cjs.map +1 -1
  132. package/dist/components/navigation/index.js +9 -17
  133. package/dist/components/onboarding-guides/index.cjs +35 -36
  134. package/dist/components/onboarding-guides/index.cjs.map +1 -1
  135. package/dist/components/onboarding-guides/index.js +13 -14
  136. package/dist/components/onboarding-guides/index.js.map +1 -1
  137. package/dist/components/onboarding-guides/onboarding-guide-detail-view.d.ts +1 -1
  138. package/dist/components/onboarding-guides/onboarding-guide-detail-view.d.ts.map +1 -1
  139. package/dist/components/related-content/index.cjs +9 -16
  140. package/dist/components/related-content/index.cjs.map +1 -1
  141. package/dist/components/related-content/index.js +8 -15
  142. package/dist/components/shared/dev-section/dev-section-page.d.ts +9 -0
  143. package/dist/components/shared/dev-section/dev-section-page.d.ts.map +1 -1
  144. package/dist/components/shared/dev-section/dev-section-view.d.ts.map +1 -1
  145. package/dist/components/shared/dev-section/index.d.ts +1 -1
  146. package/dist/components/shared/dev-section/index.d.ts.map +1 -1
  147. package/dist/components/shared/doc-search/use-doc-search.d.ts.map +1 -1
  148. package/dist/components/shared/legal-document/legal-document-page.d.ts.map +1 -1
  149. package/dist/components/shared/product-release/release-detail-page.d.ts.map +1 -1
  150. package/dist/components/tickets/index.cjs +100 -112
  151. package/dist/components/tickets/index.cjs.map +1 -1
  152. package/dist/components/tickets/index.js +20 -32
  153. package/dist/components/tickets/index.js.map +1 -1
  154. package/dist/components/ui/button/split-button.d.ts.map +1 -1
  155. package/dist/components/ui/file-manager/index.cjs +50 -52
  156. package/dist/components/ui/file-manager/index.cjs.map +1 -1
  157. package/dist/components/ui/file-manager/index.js +4 -6
  158. package/dist/components/ui/file-manager/index.js.map +1 -1
  159. package/dist/components/ui/index.cjs +13 -19
  160. package/dist/components/ui/index.cjs.map +1 -1
  161. package/dist/components/ui/index.d.ts +2 -0
  162. package/dist/components/ui/index.d.ts.map +1 -1
  163. package/dist/components/ui/index.js +133 -139
  164. package/dist/components/ui/release-changelog-section.d.ts +6 -2
  165. package/dist/components/ui/release-changelog-section.d.ts.map +1 -1
  166. package/dist/components/ui/rich-markdown-renderer.d.ts +34 -0
  167. package/dist/components/ui/rich-markdown-renderer.d.ts.map +1 -0
  168. package/dist/components/ui/simple-markdown-renderer.d.ts +2 -8
  169. package/dist/components/ui/simple-markdown-renderer.d.ts.map +1 -1
  170. package/dist/contexts/chat-runtime-context.d.ts +14 -0
  171. package/dist/contexts/chat-runtime-context.d.ts.map +1 -1
  172. package/dist/contexts/index.cjs +3 -3
  173. package/dist/contexts/index.js +5 -5
  174. package/dist/embed-shims/index.cjs +3 -3
  175. package/dist/embed-shims/index.cjs.map +1 -1
  176. package/dist/embed-shims/index.js +4 -4
  177. package/dist/hooks/index.cjs +4 -9
  178. package/dist/hooks/index.cjs.map +1 -1
  179. package/dist/hooks/index.js +6 -11
  180. package/dist/index.cjs +14 -20
  181. package/dist/index.cjs.map +1 -1
  182. package/dist/index.js +362 -368
  183. package/dist/types/doc-source.d.ts +31 -1
  184. package/dist/types/doc-source.d.ts.map +1 -1
  185. package/dist/utils/index.cjs +4 -0
  186. package/dist/utils/index.cjs.map +1 -1
  187. package/dist/utils/index.d.ts +1 -0
  188. package/dist/utils/index.d.ts.map +1 -1
  189. package/dist/utils/index.js +4 -1
  190. package/dist/utils/index.js.map +1 -1
  191. package/dist/utils/page-header-constants.d.ts +15 -0
  192. package/dist/utils/page-header-constants.d.ts.map +1 -0
  193. package/dist/utils/social-embed-cache.d.ts +29 -0
  194. package/dist/utils/social-embed-cache.d.ts.map +1 -0
  195. package/package.json +7 -1
  196. package/src/components/case-studies/index.ts +4 -0
  197. package/src/components/case-studies/share-experience-section.tsx +185 -0
  198. package/src/components/chat/chat-container.tsx +5 -7
  199. package/src/components/chat/embeddable-chat.tsx +1 -1
  200. package/src/components/chat/error-message-display.tsx +49 -31
  201. package/src/components/chat/types/component.types.ts +2 -0
  202. package/src/components/docs/doc-viewer.tsx +111 -19
  203. package/src/components/docs/docs-hub-page.tsx +149 -0
  204. package/src/components/docs/index.ts +17 -0
  205. package/src/components/docs/skeletons.tsx +138 -0
  206. package/src/components/docs/use-docs-resolve-link.ts +52 -0
  207. package/src/components/docs/use-document-tree.ts +21 -0
  208. package/src/components/embeds/embed-container.tsx +80 -0
  209. package/src/components/embeds/embed-iframe.tsx +7 -9
  210. package/src/components/embeds/file-download-card.tsx +54 -0
  211. package/src/components/embeds/index.ts +30 -0
  212. package/src/components/embeds/linkedin-embed-client.tsx +100 -0
  213. package/src/components/embeds/markdown-image.tsx +88 -0
  214. package/src/components/embeds/og-link-preview.tsx +13 -13
  215. package/src/components/embeds/reddit-embed-client.tsx +550 -0
  216. package/src/components/embeds/rich-markdown-runtime.tsx +79 -0
  217. package/src/components/embeds/twitter-embed-client.tsx +308 -0
  218. package/src/components/features/notifications/notification-drawer.tsx +18 -7
  219. package/src/components/features/notifications/notifications-context.tsx +7 -0
  220. package/src/components/layout/page-header.tsx +182 -0
  221. package/src/components/layout/page-layout.tsx +14 -1
  222. package/src/components/layout/page-with-header.tsx +110 -0
  223. package/src/components/layout/title-block.tsx +40 -62
  224. package/src/components/onboarding-guides/onboarding-guide-detail-view.tsx +3 -3
  225. package/src/components/shared/dev-section/dev-section-page.tsx +9 -1
  226. package/src/components/shared/dev-section/dev-section-view.tsx +14 -9
  227. package/src/components/shared/dev-section/index.ts +1 -1
  228. package/src/components/shared/doc-search/use-doc-search.ts +7 -3
  229. package/src/components/shared/legal-document/legal-document-page.tsx +2 -2
  230. package/src/components/shared/product-release/release-detail-page.tsx +6 -4
  231. package/src/components/ui/button/split-button.tsx +5 -2
  232. package/src/components/ui/index.ts +2 -0
  233. package/src/components/ui/release-changelog-section.tsx +7 -2
  234. package/src/components/ui/rich-markdown-renderer.tsx +1203 -0
  235. package/src/components/ui/simple-markdown-renderer.tsx +7 -11
  236. package/src/contexts/chat-runtime-context.tsx +14 -0
  237. package/src/stories/NotificationDrawer.stories.tsx +2 -0
  238. package/src/types/doc-source.ts +33 -1
  239. package/src/utils/index.ts +1 -0
  240. package/src/utils/page-header-constants.ts +15 -0
  241. package/src/utils/social-embed-cache.ts +391 -0
  242. package/dist/chunk-26PKDALD.js +0 -2379
  243. package/dist/chunk-26PKDALD.js.map +0 -1
  244. package/dist/chunk-3MCHAFHB.js +0 -89
  245. package/dist/chunk-3MCHAFHB.js.map +0 -1
  246. package/dist/chunk-3XIB4VKS.cjs +0 -619
  247. package/dist/chunk-3XIB4VKS.cjs.map +0 -1
  248. package/dist/chunk-4W7NYJ3B.cjs +0 -3009
  249. package/dist/chunk-4W7NYJ3B.cjs.map +0 -1
  250. package/dist/chunk-5E2HOSSH.cjs.map +0 -1
  251. package/dist/chunk-5IJ46KAV.js.map +0 -1
  252. package/dist/chunk-5O6N3BKR.cjs.map +0 -1
  253. package/dist/chunk-6BZEAPNT.js.map +0 -1
  254. package/dist/chunk-6IBA2MQV.cjs.map +0 -1
  255. package/dist/chunk-6JINAOI7.cjs +0 -311
  256. package/dist/chunk-6JINAOI7.cjs.map +0 -1
  257. package/dist/chunk-7RIYT7ZH.js.map +0 -1
  258. package/dist/chunk-A2YL7QRX.cjs.map +0 -1
  259. package/dist/chunk-AQOWFSMB.cjs.map +0 -1
  260. package/dist/chunk-E4XABBSU.js.map +0 -1
  261. package/dist/chunk-ETACGX2A.cjs.map +0 -1
  262. package/dist/chunk-EYEW6PTA.cjs.map +0 -1
  263. package/dist/chunk-FQJK446R.js +0 -1606
  264. package/dist/chunk-FQJK446R.js.map +0 -1
  265. package/dist/chunk-FT4FCV7L.cjs.map +0 -1
  266. package/dist/chunk-INDQMNP6.cjs.map +0 -1
  267. package/dist/chunk-J54Z3OCR.cjs +0 -1606
  268. package/dist/chunk-J54Z3OCR.cjs.map +0 -1
  269. package/dist/chunk-KXCRGTRN.cjs +0 -2379
  270. package/dist/chunk-KXCRGTRN.cjs.map +0 -1
  271. package/dist/chunk-LCNMR277.js.map +0 -1
  272. package/dist/chunk-LFGGF7OT.cjs +0 -449
  273. package/dist/chunk-LFGGF7OT.cjs.map +0 -1
  274. package/dist/chunk-M2OCXTNT.js +0 -311
  275. package/dist/chunk-M2OCXTNT.js.map +0 -1
  276. package/dist/chunk-NSPOYUBH.js +0 -3009
  277. package/dist/chunk-NSPOYUBH.js.map +0 -1
  278. package/dist/chunk-OOKKGOPQ.js.map +0 -1
  279. package/dist/chunk-OQ6X7ZOC.js +0 -449
  280. package/dist/chunk-OQ6X7ZOC.js.map +0 -1
  281. package/dist/chunk-POKKCWKF.js +0 -354
  282. package/dist/chunk-POKKCWKF.js.map +0 -1
  283. package/dist/chunk-TFSYSWPS.cjs +0 -89
  284. package/dist/chunk-TFSYSWPS.cjs.map +0 -1
  285. package/dist/chunk-XXI7BNB6.cjs.map +0 -1
  286. package/dist/chunk-YD43AKI5.js +0 -619
  287. package/dist/chunk-YD43AKI5.js.map +0 -1
  288. package/dist/chunk-YETA25JW.cjs +0 -354
  289. package/dist/chunk-YETA25JW.cjs.map +0 -1
  290. package/dist/chunk-YIGPRLQY.cjs.map +0 -1
  291. /package/dist/{chunk-3ZXUQQL4.js.map → chunk-PI4WSYQV.js.map} +0 -0
@@ -0,0 +1,308 @@
1
+ "use client";
2
+
3
+ import { useState, useEffect, useRef } from 'react';
4
+ import { XLogo } from '../icons/x-logo';
5
+ import { socialCache } from '../../utils/social-embed-cache';
6
+ import { TwitterContainer } from './embed-container';
7
+ import { useRichMarkdownRuntime } from './rich-markdown-runtime';
8
+
9
+ // Using inline SVG icons to avoid dependency issues
10
+ const MessageCircleIcon = () => (
11
+ <svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24">
12
+ <path d="M21 15a2 2 0 01-2 2H7l-4 4V5a2 2 0 012-2h14a2 2 0 012 2z"/>
13
+ </svg>
14
+ );
15
+
16
+ const ExternalLinkIcon = () => (
17
+ <svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24">
18
+ <path d="M18 13v6a2 2 0 01-2 2H5a2 2 0 01-2-2V8a2 2 0 012-2h6"/>
19
+ <polyline points="15,3 21,3 21,9"/>
20
+ <line x1="10" y1="14" x2="21" y2="3"/>
21
+ </svg>
22
+ );
23
+
24
+ const HeartIcon = () => (
25
+ <svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24">
26
+ <path d="M20.84 4.61a5.5 5.5 0 0 0-7.78 0L12 5.67l-1.06-1.06a5.5 5.5 0 0 0-7.78 7.78l1.06 1.06L12 21.23l7.78-7.78 1.06-1.06a5.5 5.5 0 0 0 0-7.78z"/>
27
+ </svg>
28
+ );
29
+
30
+ const RepeatIcon = () => (
31
+ <svg width="16" height="16" fill="none" stroke="currentColor" viewBox="0 0 24 24">
32
+ <polyline points="17,1 21,5 17,9"/>
33
+ <path d="M3 11V9a4 4 0 0 1 4-4h14"/>
34
+ <polyline points="7,23 3,19 7,15"/>
35
+ <path d="M21 13v2a4 4 0 0 1-4 4H3"/>
36
+ </svg>
37
+ );
38
+
39
+ const ClockIcon = () => (
40
+ <svg width="12" height="12" fill="none" stroke="currentColor" viewBox="0 0 24 24">
41
+ <circle cx="12" cy="12" r="10"/>
42
+ <polyline points="12,6 12,12 16,14"/>
43
+ </svg>
44
+ );
45
+
46
+ const UserIcon = () => (
47
+ <svg width="12" height="12" fill="none" stroke="currentColor" viewBox="0 0 24 24">
48
+ <path d="M20 21v-2a4 4 0 00-4-4H8a4 4 0 00-4 4v2"/>
49
+ <circle cx="12" cy="7" r="4"/>
50
+ </svg>
51
+ );
52
+
53
+ // X glyph: the lib's standard XLogo (color follows the text context).
54
+ const XIcon = () => <XLogo className="w-5 h-5" color="currentColor" />;
55
+
56
+
57
+
58
+ interface TwitterOEmbedResponse {
59
+ url: string;
60
+ author_name: string;
61
+ author_url: string;
62
+ html: string;
63
+ width: number;
64
+ height: number;
65
+ type: string;
66
+ cache_age: string;
67
+ provider_name: string;
68
+ provider_url: string;
69
+ version: string;
70
+ }
71
+
72
+ interface TwitterEmbedProps {
73
+ url: string;
74
+ tweetId?: string;
75
+ maxWidth?: number;
76
+ }
77
+
78
+ export function TwitterEmbedClient({ url, tweetId, maxWidth = 700 }: TwitterEmbedProps) {
79
+ const { twitterProxyUrl } = useRichMarkdownRuntime();
80
+ const [tweetData, setTweetData] = useState<TwitterOEmbedResponse | null>(null);
81
+ const [loading, setLoading] = useState(true);
82
+ const [error, setError] = useState<string | null>(null);
83
+ const initializationDone = useRef(false);
84
+
85
+ // Extract tweet ID from URL if not provided
86
+ const extractedTweetId = tweetId || url.match(/status\/(\d+)/)?.[1];
87
+
88
+ // Normalize the Twitter URL
89
+ const tweetUrl = url.includes('twitter.com') || url.includes('x.com')
90
+ ? url
91
+ : `https://twitter.com/twitter/status/${extractedTweetId}`;
92
+
93
+ useEffect(() => {
94
+ // Only run once
95
+ if (initializationDone.current) return;
96
+ initializationDone.current = true;
97
+
98
+ if (!extractedTweetId) {
99
+ setError('Invalid tweet URL or ID');
100
+ setLoading(false);
101
+ return;
102
+ }
103
+
104
+ // Use centralized cache hierarchy
105
+ socialCache.fetchWithHierarchy({
106
+ platform: 'twitter',
107
+ url: tweetUrl,
108
+ apiEndpoint: twitterProxyUrl,
109
+ dataValidator: (data) => data && data.html,
110
+ onDataUpdate: (data) => setTweetData(data),
111
+ onError: (errorMsg) => setError(errorMsg),
112
+ onLoading: (loading) => setLoading(loading)
113
+ });
114
+ }, []); // Empty dependency array - only run once
115
+
116
+ if (loading) {
117
+ return (
118
+ <TwitterContainer>
119
+ <div className="border border-ods-border rounded-lg p-6 bg-ods-card animate-pulse">
120
+ <div className="flex items-center space-x-3 mb-4">
121
+ <div className="w-12 h-12 bg-ods-border rounded-full"></div>
122
+ <div>
123
+ <div className="h-4 bg-ods-border rounded w-32 mb-2"></div>
124
+ <div className="h-3 bg-ods-border rounded w-24"></div>
125
+ </div>
126
+ </div>
127
+ <div className="space-y-2 mb-4">
128
+ <div className="h-4 bg-ods-border rounded w-full"></div>
129
+ <div className="h-4 bg-ods-border rounded w-3/4"></div>
130
+ </div>
131
+ <div className="flex items-center space-x-4">
132
+ <div className="h-4 bg-ods-border rounded w-16"></div>
133
+ <div className="h-4 bg-ods-border rounded w-16"></div>
134
+ <div className="h-4 bg-ods-border rounded w-16"></div>
135
+ </div>
136
+ </div>
137
+ </TwitterContainer>
138
+ );
139
+ }
140
+
141
+ if (error || !tweetData) {
142
+ return (
143
+ <TwitterContainer>
144
+ <div className="border border-ods-border rounded-lg p-6 bg-ods-card">
145
+ <div className="flex items-center space-x-3 text-ods-text-secondary mb-4">
146
+ <XIcon />
147
+ <span>Tweet unavailable</span>
148
+ </div>
149
+
150
+ <div className="text-center">
151
+ <p className="text-ods-text-secondary text-sm mb-4">
152
+ This tweet could not be loaded. It may have been deleted, made private, or the account may be suspended.
153
+ </p>
154
+ <a
155
+ href={url}
156
+ target="_blank"
157
+ rel="noopener noreferrer"
158
+ className="inline-flex items-center space-x-2 px-4 py-2 bg-ods-bg-secondary text-ods-text-primary rounded-md text-sm font-medium hover:bg-ods-bg-tertiary transition-colors"
159
+ >
160
+ <XIcon />
161
+ <span>View on X</span>
162
+ </a>
163
+ </div>
164
+ </div>
165
+ </TwitterContainer>
166
+ );
167
+ }
168
+
169
+ // Parse the HTML to extract detailed tweet information and media
170
+ const parser = new DOMParser();
171
+ const doc = parser.parseFromString(tweetData.html, 'text/html');
172
+ const blockquote = doc.querySelector('blockquote');
173
+
174
+ // Extract tweet text (remove attribution line)
175
+ const fullText = blockquote?.textContent || '';
176
+ const tweetText = fullText.replace(/- .* \(@.*\).*$/, '').trim();
177
+
178
+ // Extract username from author_url (e.g., https://twitter.com/username)
179
+ const username = tweetData.author_url ? tweetData.author_url.split('/').pop() : '';
180
+
181
+
182
+
183
+ // Extract any links from the tweet
184
+ const links = Array.from(blockquote?.querySelectorAll('a') || [])
185
+ .map(link => ({
186
+ url: link.href,
187
+ text: link.textContent || link.href
188
+ }))
189
+ .filter(link => !link.url.includes('twitter.com') && !link.url.includes('x.com'));
190
+
191
+ // Format time (simulated - we don't have real timestamp from oEmbed)
192
+ const formatTime = () => {
193
+ return 'on X'; // Simplified since we don't have actual timestamp
194
+ };
195
+
196
+ const truncateText = (text: string, maxLength: number = 600) => {
197
+ if (text.length <= maxLength) return text;
198
+ return text.slice(0, maxLength) + '...';
199
+ };
200
+
201
+ // Profile picture URL using Unavatar service.
202
+ // The hub used to proxy this via `useProxiedImageUrl` (chat runtime), but
203
+ // docs / blog pages don't mount a chat runtime, so we fetch unavatar
204
+ // directly. Embedders that need a proxy can register a runtime later.
205
+ const getProfilePicUrl = (username: string | undefined) => {
206
+ if (!username) return '';
207
+ return `https://unavatar.io/twitter/${username}`;
208
+ };
209
+
210
+ return (
211
+ <TwitterContainer>
212
+ <div className="border border-ods-border rounded-lg bg-ods-card overflow-hidden">
213
+ {/* Header with Profile Picture */}
214
+ <div className="p-4 border-b border-ods-border">
215
+ <div className="flex items-center justify-between">
216
+ <div className="flex items-center space-x-3">
217
+ {/* User Profile Picture */}
218
+ <div className="w-8 h-8 rounded-full overflow-hidden flex-shrink-0">
219
+ <img
220
+ src={getProfilePicUrl(username)}
221
+ alt={`${tweetData.author_name} profile picture`}
222
+ className="w-full h-full object-cover"
223
+ onError={(e) => {
224
+ // Simple fallback without state updates
225
+ const target = e.target as HTMLImageElement;
226
+ target.src = 'data:image/svg+xml;base64,PHN2ZyB3aWR0aD0iMjQiIGhlaWdodD0iMjQiIGZpbGw9Im5vbmUiIHN0cm9rZT0iY3VycmVudENvbG9yIiB2aWV3Qm94PSIwIDAgMjQgMjQiPjxwYXRoIGQ9Ik0yMCAyMXYtMmE0IDQgMCAwIDAtNC00SDhhNCA0IDAgMCAwLTQgNHYyIi8+PGNpcmNsZSBjeD0iMTIiIGN5PSI3IiByPSI0Ii8+PC9zdmc+';
227
+ }}
228
+ />
229
+ </div>
230
+ <div>
231
+ <p className="text-ods-text-primary font-medium">@{username}</p>
232
+ <div className="flex items-center space-x-2 text-ods-text-secondary text-sm">
233
+ <UserIcon />
234
+ <span>{tweetData.author_name}</span>
235
+ <ClockIcon />
236
+ <span>{formatTime()}</span>
237
+ </div>
238
+ </div>
239
+ </div>
240
+ </div>
241
+ </div>
242
+
243
+ {/* Content */}
244
+ <div className="p-4">
245
+ {tweetText && (
246
+ <div
247
+ className="text-ods-text-secondary text-sm leading-relaxed mb-4 overflow-hidden"
248
+ style={{ maxHeight: `${maxWidth - 200}px` }}
249
+ >
250
+ <p className="whitespace-pre-wrap">
251
+ {truncateText(tweetText)}
252
+ </p>
253
+ </div>
254
+ )}
255
+
256
+
257
+
258
+ {/* Links Section */}
259
+ {links.length > 0 && (
260
+ <div className="mb-4 space-y-2">
261
+ {links.map((link, index) => (
262
+ <a
263
+ key={index}
264
+ href={link.url}
265
+ target="_blank"
266
+ rel="noopener noreferrer"
267
+ className="inline-flex items-center space-x-2 text-[#1DA1F2] hover:text-ods-accent transition-colors text-sm"
268
+ >
269
+ <ExternalLinkIcon />
270
+ <span className="underline">{link.text}</span>
271
+ </a>
272
+ ))}
273
+ </div>
274
+ )}
275
+
276
+ {/* Stats */}
277
+ <div className="flex items-center space-x-6 text-ods-text-secondary text-sm">
278
+ <div className="flex items-center space-x-1">
279
+ <HeartIcon />
280
+ <span>Likes</span>
281
+ </div>
282
+ <div className="flex items-center space-x-1">
283
+ <RepeatIcon />
284
+ <span>Retweets</span>
285
+ </div>
286
+ <div className="flex items-center space-x-1">
287
+ <MessageCircleIcon />
288
+ <span>Replies</span>
289
+ </div>
290
+ </div>
291
+ </div>
292
+
293
+ {/* Footer */}
294
+ <div className="px-4 py-3 bg-ods-bg-secondary border-t border-ods-border">
295
+ <a
296
+ href={url}
297
+ target="_blank"
298
+ rel="noopener noreferrer"
299
+ className="inline-flex items-center space-x-2 text-ods-accent hover:opacity-80 transition-colors text-sm font-medium"
300
+ >
301
+ <ExternalLinkIcon />
302
+ <span>View on X</span>
303
+ </a>
304
+ </div>
305
+ </div>
306
+ </TwitterContainer>
307
+ );
308
+ }
@@ -2,8 +2,8 @@
2
2
 
3
3
  import { useEffect, useRef } from 'react'
4
4
  import { BellOffIcon } from '../../icons-v2-generated/interface/bell-off-icon'
5
- import { ClockHistoryIcon } from '../../icons-v2-generated/date-and-time/clock-history-icon'
6
- import { Button } from '../../ui/button/button'
5
+ import { ClockHistoryIcon, ArrowRightUpIcon } from '../../icons-v2-generated'
6
+ import { SplitButton } from '../../ui/button'
7
7
  import { Drawer, DrawerContent, DrawerTitle } from '../../ui/drawer'
8
8
  import { Switch } from '../../ui/switch'
9
9
  import { cn } from '../../../utils/cn'
@@ -45,6 +45,7 @@ export function NotificationDrawer({
45
45
  setShowDesktopPopups,
46
46
  desktopPopupsConfigured,
47
47
  onHistoryClick,
48
+ historyHref,
48
49
  hasMore,
49
50
  isLoadingMore,
50
51
  loadMore,
@@ -106,6 +107,7 @@ export function NotificationDrawer({
106
107
  </div>
107
108
  <NotificationsHistoryButton
108
109
  onClick={onHistoryClick ? () => { onHistoryClick(); close() } : undefined}
110
+ historyHref={historyHref}
109
111
  />
110
112
  </div>
111
113
  </DrawerContent>
@@ -271,18 +273,27 @@ function DesktopNotificationsToggleRow({ checked, onChange }: ToggleRowProps) {
271
273
 
272
274
  interface HistoryButtonProps {
273
275
  onClick?: () => void
276
+ historyHref?: string
274
277
  }
275
278
 
276
- function NotificationsHistoryButton({ onClick }: HistoryButtonProps) {
279
+ function NotificationsHistoryButton({ onClick, historyHref }: HistoryButtonProps) {
277
280
  return (
278
- <Button
281
+ <SplitButton
279
282
  variant="outline"
280
283
  fullWidth
281
- disabled={!onClick}
282
284
  onClick={onClick}
283
- leftIcon={<ClockHistoryIcon className="!size-6 text-ods-text-secondary" />}
285
+ mainDisabled={!onClick}
286
+ leftIcon={<ClockHistoryIcon className="text-ods-text-secondary" />}
287
+ groupAriaLabel="Notifications history"
288
+ iconAction={{
289
+ icon: <ArrowRightUpIcon className="text-ods-text-secondary" />,
290
+ 'aria-label': 'Open notifications history in a new tab',
291
+ href: historyHref,
292
+ openInNewTab: true,
293
+ disabled: !historyHref,
294
+ }}
284
295
  >
285
296
  Notifications History
286
- </Button>
297
+ </SplitButton>
287
298
  )
288
299
  }
@@ -26,6 +26,8 @@ interface NotificationsContextValue {
26
26
  /** True when the host app wired desktop notifications (passed `onShowDesktopPopupsChange`). */
27
27
  desktopPopupsConfigured: boolean
28
28
  onHistoryClick?: () => void
29
+ /** Destination for the history button's "open in a new tab" split action. */
30
+ historyHref?: string
29
31
  hasMore: boolean
30
32
  isLoadingMore: boolean
31
33
  loadMore?: () => void
@@ -55,6 +57,8 @@ export interface NotificationsProviderProps {
55
57
  defaultShowDesktopPopups?: boolean
56
58
  onShowDesktopPopupsChange?: (value: boolean) => void
57
59
  onHistoryClick?: () => void
60
+ /** Destination for the history button's "open in a new tab" split action. */
61
+ historyHref?: string
58
62
  actions?: NotificationsActions
59
63
  /** Pagination — when omitted, the drawer hides its load-more sentinel. */
60
64
  hasMore?: boolean
@@ -149,6 +153,7 @@ export function NotificationsProvider({
149
153
  defaultShowDesktopPopups = false,
150
154
  onShowDesktopPopupsChange,
151
155
  onHistoryClick,
156
+ historyHref,
152
157
  actions,
153
158
  hasMore = false,
154
159
  isLoadingMore = false,
@@ -268,6 +273,7 @@ export function NotificationsProvider({
268
273
  setShowDesktopPopups,
269
274
  desktopPopupsConfigured,
270
275
  onHistoryClick,
276
+ historyHref,
271
277
  hasMore,
272
278
  isLoadingMore,
273
279
  loadMore: onLoadMore,
@@ -294,6 +300,7 @@ export function NotificationsProvider({
294
300
  toggle,
295
301
  setShowPopups,
296
302
  onHistoryClick,
303
+ historyHref,
297
304
  hasMore,
298
305
  isLoadingMore,
299
306
  onLoadMore,
@@ -0,0 +1,182 @@
1
+ 'use client'
2
+
3
+ import React from 'react'
4
+ import { cn } from '../../utils/cn'
5
+ import { EntityImage } from '../ui/entity-image'
6
+ import { BackButton } from './back-button'
7
+
8
+ /**
9
+ * Page-header primitive — the canonical "back-button + title + subtitle
10
+ * + (optional) image / actions" chrome every lib page uses.
11
+ *
12
+ * Owns the SSOT for the page-header DOM/CSS that the rest of the lib
13
+ * was duplicating: pre-`mb` top padding, h1 typography (`text-h2`), h6
14
+ * subtitle (`text-h6`), the gap between the back button and the title
15
+ * block, and the right-side actions slot. Consumers either render this
16
+ * directly (e.g. `<DocViewer>` / `<DocsHubPage>`) or compose it through
17
+ * the `<TitleBlock>` adapter which adds the `PageActions` /
18
+ * `ActionsMenu` / selector wiring on top.
19
+ *
20
+ * Why this exists: knowledge-hub vs releases sat at different vertical
21
+ * rhythms (px-perfect mismatch on title baseline + subtitle offset)
22
+ * because the docs surface hand-rolled its own chrome instead of going
23
+ * through `TitleBlock`. Centralizing the layout here means every
24
+ * embeddable lib page (DocViewer, DevSectionPage, LegalDocumentPage,
25
+ * OnboardingGuideDetailView) renders pixel-identical title/subtitle/
26
+ * back-button typography + spacing — and a future spacing/typography
27
+ * tweak is one file.
28
+ */
29
+ export interface PageHeaderProps {
30
+ /** Page title (h1). Plain string — ReactNode is intentionally not
31
+ * supported here so every consumer renders the same typography. */
32
+ title?: string
33
+ /** Optional icon rendered inline before the title text (e.g. the
34
+ * rocket emoji on /releases, the docs icon on /knowledge-base).
35
+ * Same `flex items-center gap-3` row as `<DevSectionView>`'s hero. */
36
+ titleIcon?: React.ReactNode
37
+ /** Page subtitle (description paragraph). */
38
+ subtitle?: string
39
+ /**
40
+ * Render a yellow accent dot (`.`) after the title. Mirrors the
41
+ * hub's legacy `<AdminPageHeader accentDot>` flag — now lib-wide so
42
+ * surfaces like `/knowledge-base` keep their existing accent styling
43
+ * after the migration.
44
+ */
45
+ accentDot?: boolean
46
+ /** Optional thumbnail / hero image rendered to the left of the
47
+ * title block. Used by entity-image-style headers (onboarding
48
+ * guides, knowledge-base entries). */
49
+ image?: { src: string; alt?: string }
50
+ /** Back-button shown above the title block. Hidden on mobile (matches
51
+ * the existing TitleBlock + DocViewer behavior). */
52
+ backButton?: { label?: string; onClick: () => void }
53
+ /** Right-side actions slot (action buttons / menu / tab selector).
54
+ * Composed externally (e.g. `<TitleBlock>` builds `PageActions` + menu
55
+ * + selector and passes the result here). */
56
+ actions?: React.ReactNode
57
+ /**
58
+ * Visual variant.
59
+ * - `plain` (default): transparent background, no border.
60
+ * - `card`: card background, border, and padding on mobile only —
61
+ * collapses to plain on md+ (legacy `TitleBlock` variant — kept so
62
+ * surfaces that depend on the card affordance don't regress).
63
+ */
64
+ variant?: 'plain' | 'card'
65
+ /** When the consumer wraps `<PageHeader>` in its OWN spacing container
66
+ * (e.g. `<DevSectionView>`'s `gap-10 flex-col`), the default `mb-l`
67
+ * bottom margin doubles up. Pass `noBottomMargin` to opt out. */
68
+ noBottomMargin?: boolean
69
+ /** Same as `noBottomMargin` for the default `pt-l` top padding. Set
70
+ * this when PageHeader is nested INSIDE another layout that already
71
+ * provides top spacing (e.g. `<DevSectionView>`'s hero, which sits
72
+ * inside `<PageLayout>`'s children flow). */
73
+ noTopPadding?: boolean
74
+ className?: string
75
+ }
76
+
77
+ // Title typography — copied verbatim from <DevSectionView>'s hero h1
78
+ // (`src/components/shared/dev-section/dev-section-view.tsx`). The user-
79
+ // reported "header text not aligned" between /knowledge-base and
80
+ // /releases bottomed out here: DevSectionView rendered text-h1 with
81
+ // tracking-[-1.12px] while this component used text-h2 — visually huge
82
+ // gap. Now both render through the exact same class string. DevSectionView
83
+ // is being refactored in this commit to delegate to <PageHeader> so the
84
+ // shared-component claim is enforced at the code level too.
85
+ const TITLE_CLASS = 'text-h1 tracking-[-1.12px] text-ods-text-primary flex items-center gap-3'
86
+ // Subtitle ALWAYS occupies exactly 2 lines of vertical space.
87
+ // `min-h-[56px]` (= 2 × 28px leading) reserves the row height so a
88
+ // single-line subtitle doesn't shrink the header — page-to-page height
89
+ // stays consistent.
90
+ // `line-clamp-2` caps long copy at 2 lines + ellipsis so wrapping doesn't
91
+ // push the search bar down.
92
+ const SUBTITLE_CLASS = "font-['DM_Sans'] font-medium text-[18px] leading-[28px] text-ods-text-secondary max-w-3xl line-clamp-2 min-h-[56px]"
93
+
94
+ export function PageHeader({
95
+ title,
96
+ titleIcon,
97
+ subtitle,
98
+ accentDot,
99
+ image,
100
+ backButton,
101
+ actions,
102
+ variant = 'plain',
103
+ noBottomMargin = false,
104
+ noTopPadding = false,
105
+ className,
106
+ }: PageHeaderProps) {
107
+ return (
108
+ <div
109
+ className={cn(
110
+ 'flex items-end justify-between gap-[var(--spacing-system-m)]',
111
+ 'md:flex-col md:items-start md:justify-start lg:flex-row lg:items-end lg:justify-between',
112
+ !noTopPadding && 'pt-[var(--spacing-system-l)]',
113
+ variant === 'card'
114
+ ? cn(
115
+ 'bg-ods-card border-b border-ods-border',
116
+ 'px-[var(--spacing-system-l)] pb-[var(--spacing-system-l)]',
117
+ 'md:bg-transparent md:border-b-0',
118
+ 'md:px-0 md:pb-0',
119
+ !noBottomMargin && 'md:mb-[var(--spacing-system-l)]',
120
+ )
121
+ : !noBottomMargin && 'mb-[var(--spacing-system-l)]',
122
+ className,
123
+ )}
124
+ >
125
+ <div className="flex flex-col gap-[var(--spacing-system-xs)] flex-1 min-w-0">
126
+ {backButton && (
127
+ <BackButton
128
+ onClick={backButton.onClick}
129
+ label={backButton.label}
130
+ className="hidden md:inline-flex"
131
+ />
132
+ )}
133
+ {/* Title + subtitle stack. Matches `<DevSectionView>`'s hero
134
+ * exactly: `space-y-4` between h1 and p, `flex items-center
135
+ * gap-3` for icon-inline title row, identical class strings.
136
+ * Image (entity-image-style) prefixes the title row, NOT a
137
+ * separate column with its own vertical rhythm — that's the
138
+ * legacy TitleBlock 2-col layout which broke the title-to-
139
+ * subtitle gap. */}
140
+ {(title || subtitle || image || titleIcon) && (
141
+ <div className="space-y-4">
142
+ {(title || image || titleIcon) && (
143
+ <h1 className={TITLE_CLASS}>
144
+ {image && (
145
+ <EntityImage
146
+ src={image.src}
147
+ alt={image.alt}
148
+ fallbackText={image.alt || title}
149
+ />
150
+ )}
151
+ {titleIcon}
152
+ {title && (
153
+ <span>
154
+ {title}
155
+ {accentDot && <span className="text-ods-accent">.</span>}
156
+ </span>
157
+ )}
158
+ </h1>
159
+ )}
160
+ {(title || titleIcon || image) && (
161
+ // Always render the subtitle <p> when the title block exists,
162
+ // even when `subtitle` is empty — `SUBTITLE_CLASS` reserves
163
+ // `min-h-[56px]` so headers WITH and WITHOUT subtitles share a
164
+ // baseline (the JSDoc on SUBTITLE_CLASS claims "ALWAYS occupies
165
+ // exactly 2 lines"; gating the <p> on truthy `subtitle` broke
166
+ // that contract — pages without subtitle were ~56px shorter).
167
+ // Falsy subtitles render an NBSP placeholder so the empty <p>
168
+ // still takes its reserved height.
169
+ <p className={SUBTITLE_CLASS}>{subtitle || ' '}</p>
170
+ )}
171
+ </div>
172
+ )}
173
+ </div>
174
+
175
+ {actions && (
176
+ <div className="flex gap-2 items-center shrink-0">{actions}</div>
177
+ )}
178
+ </div>
179
+ )
180
+ }
181
+
182
+ export default PageHeader
@@ -10,6 +10,11 @@ export interface PageLayoutProps {
10
10
  children: React.ReactNode
11
11
  title?: string
12
12
  subtitle?: string
13
+ /** Inline icon rendered before the title text — forwarded to
14
+ * TitleBlock/PageHeader. Same shape as DevSectionPage's hero icon. */
15
+ titleIcon?: React.ReactNode
16
+ /** Yellow accent dot after the title — forwarded to TitleBlock/PageHeader. */
17
+ accentDot?: boolean
13
18
  image?: { src: string; alt?: string }
14
19
  backButton?: { label?: string; onClick: () => void }
15
20
  actions?: PageActionButton[]
@@ -33,6 +38,8 @@ export function PageLayout({
33
38
  children,
34
39
  title,
35
40
  subtitle,
41
+ titleIcon,
42
+ accentDot,
36
43
  image,
37
44
  backButton,
38
45
  actions,
@@ -46,14 +53,16 @@ export function PageLayout({
46
53
  }: PageLayoutProps) {
47
54
  const hasActions = actions && actions.length > 0
48
55
  const needsBottomPadding = hasActions && actionsVariant === 'primary-buttons'
49
- const hasHeader = showHeader && (title || subtitle || image || backButton || hasActions || selector)
56
+ const hasHeader = showHeader && (title || subtitle || titleIcon || image || backButton || hasActions || selector)
50
57
 
51
58
  return (
52
59
  <div className={cn('flex flex-col w-full', className)}>
53
60
  {hasHeader && (
54
61
  <TitleBlock
55
62
  title={title}
63
+ titleIcon={titleIcon}
56
64
  subtitle={subtitle}
65
+ accentDot={accentDot}
57
66
  image={image}
58
67
  backButton={backButton}
59
68
  actions={actions}
@@ -74,4 +83,8 @@ export function PageLayout({
74
83
  export type { PageActionButton } from '../ui/page-actions'
75
84
  export { TitleBlock } from './title-block'
76
85
  export type { TitleBlockProps } from './title-block'
86
+ export { PageHeader } from './page-header'
87
+ export type { PageHeaderProps } from './page-header'
88
+ export { PageWithHeader } from './page-with-header'
89
+ export type { PageWithHeaderProps } from './page-with-header'
77
90
  export default PageLayout