@commonpub/layer 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (436) hide show
  1. package/LICENSE +661 -0
  2. package/app.vue +7 -0
  3. package/components/AnnouncementBand.vue +117 -0
  4. package/components/AppToast.vue +108 -0
  5. package/components/AuthorCard.vue +119 -0
  6. package/components/AuthorRow.vue +81 -0
  7. package/components/CommentSection.vue +330 -0
  8. package/components/ContentCard.vue +340 -0
  9. package/components/ContentPicker.vue +240 -0
  10. package/components/ContentStarterForm.vue +214 -0
  11. package/components/ContentTypeBadge.vue +46 -0
  12. package/components/CountdownTimer.vue +68 -0
  13. package/components/CpubEditor.vue +87 -0
  14. package/components/DiscussionItem.vue +191 -0
  15. package/components/EditorPropertiesPanel.vue +393 -0
  16. package/components/EngagementBar.vue +131 -0
  17. package/components/FederatedContentCard.vue +291 -0
  18. package/components/FeedItem.vue +283 -0
  19. package/components/FilterChip.vue +21 -0
  20. package/components/HeatmapGrid.vue +92 -0
  21. package/components/ImageUpload.vue +219 -0
  22. package/components/MemberCard.vue +163 -0
  23. package/components/MessageThread.vue +120 -0
  24. package/components/NotificationItem.vue +103 -0
  25. package/components/ProgressTracker.vue +41 -0
  26. package/components/PublishErrorsModal.vue +116 -0
  27. package/components/RemoteActorCard.vue +206 -0
  28. package/components/RemoteUserSearch.vue +117 -0
  29. package/components/SearchFilters.vue +188 -0
  30. package/components/SearchSidebar.vue +181 -0
  31. package/components/SectionHeader.vue +17 -0
  32. package/components/ShareToHubModal.vue +189 -0
  33. package/components/SiteLogo.vue +21 -0
  34. package/components/SkillBar.vue +57 -0
  35. package/components/SortSelect.vue +30 -0
  36. package/components/StatBar.vue +14 -0
  37. package/components/TOCNav.vue +69 -0
  38. package/components/TimelineItem.vue +82 -0
  39. package/components/VideoCard.vue +106 -0
  40. package/components/blocks/BlockBuildStepView.vue +92 -0
  41. package/components/blocks/BlockCalloutView.vue +82 -0
  42. package/components/blocks/BlockCheckpointView.vue +50 -0
  43. package/components/blocks/BlockCodeView.vue +212 -0
  44. package/components/blocks/BlockContentRenderer.vue +143 -0
  45. package/components/blocks/BlockDividerView.vue +11 -0
  46. package/components/blocks/BlockDownloadsView.vue +126 -0
  47. package/components/blocks/BlockEmbedView.vue +61 -0
  48. package/components/blocks/BlockGalleryView.vue +57 -0
  49. package/components/blocks/BlockHeadingView.vue +29 -0
  50. package/components/blocks/BlockImageView.vue +34 -0
  51. package/components/blocks/BlockMarkdownView.vue +118 -0
  52. package/components/blocks/BlockMathView.vue +45 -0
  53. package/components/blocks/BlockPartsListView.vue +104 -0
  54. package/components/blocks/BlockQuizView.vue +239 -0
  55. package/components/blocks/BlockQuoteView.vue +41 -0
  56. package/components/blocks/BlockSectionHeaderView.vue +58 -0
  57. package/components/blocks/BlockSliderView.vue +236 -0
  58. package/components/blocks/BlockTextView.vue +41 -0
  59. package/components/blocks/BlockToolListView.vue +87 -0
  60. package/components/blocks/BlockVideoView.vue +89 -0
  61. package/components/editors/ArticleEditor.vue +545 -0
  62. package/components/editors/BlockCanvas.vue +487 -0
  63. package/components/editors/BlockInsertZone.vue +84 -0
  64. package/components/editors/BlockPicker.vue +285 -0
  65. package/components/editors/BlockWrapper.vue +192 -0
  66. package/components/editors/BlogEditor.vue +567 -0
  67. package/components/editors/EditorBlocks.vue +248 -0
  68. package/components/editors/EditorSection.vue +81 -0
  69. package/components/editors/EditorShell.vue +168 -0
  70. package/components/editors/EditorTagInput.vue +114 -0
  71. package/components/editors/EditorVisibility.vue +110 -0
  72. package/components/editors/ExplainerEditor.vue +503 -0
  73. package/components/editors/MarkdownImportDialog.vue +249 -0
  74. package/components/editors/ProjectEditor.vue +446 -0
  75. package/components/editors/blocks/BuildStepBlock.vue +102 -0
  76. package/components/editors/blocks/CalloutBlock.vue +122 -0
  77. package/components/editors/blocks/CheckpointBlock.vue +27 -0
  78. package/components/editors/blocks/CodeBlock.vue +177 -0
  79. package/components/editors/blocks/DividerBlock.vue +22 -0
  80. package/components/editors/blocks/DownloadsBlock.vue +41 -0
  81. package/components/editors/blocks/EmbedBlock.vue +20 -0
  82. package/components/editors/blocks/GalleryBlock.vue +236 -0
  83. package/components/editors/blocks/HeadingBlock.vue +96 -0
  84. package/components/editors/blocks/ImageBlock.vue +271 -0
  85. package/components/editors/blocks/MarkdownBlock.vue +258 -0
  86. package/components/editors/blocks/MathBlock.vue +37 -0
  87. package/components/editors/blocks/PartsListBlock.vue +358 -0
  88. package/components/editors/blocks/QuizBlock.vue +47 -0
  89. package/components/editors/blocks/QuoteBlock.vue +101 -0
  90. package/components/editors/blocks/SectionHeaderBlock.vue +130 -0
  91. package/components/editors/blocks/SliderBlock.vue +318 -0
  92. package/components/editors/blocks/TextBlock.vue +201 -0
  93. package/components/editors/blocks/ToolListBlock.vue +70 -0
  94. package/components/editors/blocks/VideoBlock.vue +22 -0
  95. package/components/hub/HubDiscussions.vue +47 -0
  96. package/components/hub/HubFeed.vue +199 -0
  97. package/components/hub/HubHero.vue +185 -0
  98. package/components/hub/HubLayout.vue +103 -0
  99. package/components/hub/HubMembers.vue +40 -0
  100. package/components/hub/HubProducts.vue +93 -0
  101. package/components/hub/HubProjects.vue +207 -0
  102. package/components/hub/HubSidebar.vue +12 -0
  103. package/components/hub/HubSidebarCard.vue +35 -0
  104. package/components/views/ArticleView.vue +771 -0
  105. package/components/views/BlogView.vue +667 -0
  106. package/components/views/ExplainerView.vue +688 -0
  107. package/components/views/ProjectView.vue +1500 -0
  108. package/composables/useApiError.ts +39 -0
  109. package/composables/useAuth.ts +87 -0
  110. package/composables/useBlockEditor.ts +187 -0
  111. package/composables/useContentSave.ts +253 -0
  112. package/composables/useContentTypes.ts +37 -0
  113. package/composables/useEngagement.ts +196 -0
  114. package/composables/useFeatures.ts +33 -0
  115. package/composables/useFederation.ts +72 -0
  116. package/composables/useJsonLd.ts +183 -0
  117. package/composables/useMarkdownImport.ts +77 -0
  118. package/composables/useMirrorContent.ts +105 -0
  119. package/composables/useNotifications.ts +73 -0
  120. package/composables/usePublishValidation.ts +65 -0
  121. package/composables/useSanitize.ts +34 -0
  122. package/composables/useSiteName.ts +4 -0
  123. package/composables/useTheme.ts +34 -0
  124. package/composables/useToast.ts +35 -0
  125. package/error.vue +129 -0
  126. package/layouts/admin.vue +213 -0
  127. package/layouts/auth.vue +63 -0
  128. package/layouts/default.vue +269 -0
  129. package/layouts/editor.vue +129 -0
  130. package/middleware/auth.ts +6 -0
  131. package/nuxt.config.ts +83 -0
  132. package/package.json +59 -0
  133. package/pages/[type]/[slug]/edit.vue +676 -0
  134. package/pages/[type]/[slug]/index.vue +313 -0
  135. package/pages/[type]/index.vue +118 -0
  136. package/pages/about.vue +100 -0
  137. package/pages/admin/audit.vue +66 -0
  138. package/pages/admin/content.vue +116 -0
  139. package/pages/admin/federation.vue +446 -0
  140. package/pages/admin/index.vue +62 -0
  141. package/pages/admin/reports.vue +88 -0
  142. package/pages/admin/settings.vue +167 -0
  143. package/pages/admin/users.vue +145 -0
  144. package/pages/auth/forgot-password.vue +103 -0
  145. package/pages/auth/login.vue +216 -0
  146. package/pages/auth/oauth/authorize.vue +178 -0
  147. package/pages/auth/register.vue +246 -0
  148. package/pages/auth/reset-password.vue +124 -0
  149. package/pages/auth/verify-email.vue +80 -0
  150. package/pages/cert/[code].vue +154 -0
  151. package/pages/contests/[slug]/edit.vue +153 -0
  152. package/pages/contests/[slug]/index.vue +556 -0
  153. package/pages/contests/[slug]/judge.vue +160 -0
  154. package/pages/contests/create.vue +192 -0
  155. package/pages/contests/index.vue +69 -0
  156. package/pages/create.vue +219 -0
  157. package/pages/dashboard.vue +521 -0
  158. package/pages/docs/[siteSlug]/[...pagePath].vue +704 -0
  159. package/pages/docs/[siteSlug]/edit.vue +480 -0
  160. package/pages/docs/[siteSlug]/index.vue +380 -0
  161. package/pages/docs/create.vue +60 -0
  162. package/pages/docs/index.vue +181 -0
  163. package/pages/explore.vue +422 -0
  164. package/pages/federated-hubs/[id]/index.vue +385 -0
  165. package/pages/federated-hubs/[id]/posts/[postId].vue +309 -0
  166. package/pages/federation/index.vue +157 -0
  167. package/pages/federation/search.vue +43 -0
  168. package/pages/federation/users/[handle].vue +221 -0
  169. package/pages/feed.vue +135 -0
  170. package/pages/hubs/[slug]/index.vue +513 -0
  171. package/pages/hubs/[slug]/members.vue +134 -0
  172. package/pages/hubs/[slug]/posts/[postId].vue +352 -0
  173. package/pages/hubs/[slug]/settings.vue +254 -0
  174. package/pages/hubs/create.vue +201 -0
  175. package/pages/hubs/index.vue +207 -0
  176. package/pages/index.vue +1005 -0
  177. package/pages/learn/[slug]/[lessonSlug]/edit.vue +413 -0
  178. package/pages/learn/[slug]/[lessonSlug]/index.vue +438 -0
  179. package/pages/learn/[slug]/edit.vue +414 -0
  180. package/pages/learn/[slug]/index.vue +341 -0
  181. package/pages/learn/create.vue +71 -0
  182. package/pages/learn/index.vue +360 -0
  183. package/pages/messages/[conversationId].vue +113 -0
  184. package/pages/messages/index.vue +303 -0
  185. package/pages/mirror/[id].vue +115 -0
  186. package/pages/notifications.vue +91 -0
  187. package/pages/products/[slug].vue +128 -0
  188. package/pages/products/index.vue +122 -0
  189. package/pages/search.vue +692 -0
  190. package/pages/settings/account.vue +170 -0
  191. package/pages/settings/appearance.vue +80 -0
  192. package/pages/settings/index.vue +81 -0
  193. package/pages/settings/notifications.vue +68 -0
  194. package/pages/settings/profile.vue +838 -0
  195. package/pages/tags/[slug].vue +111 -0
  196. package/pages/tags/index.vue +73 -0
  197. package/pages/u/[username]/followers.vue +86 -0
  198. package/pages/u/[username]/following.vue +94 -0
  199. package/pages/u/[username]/index.vue +837 -0
  200. package/pages/videos/[id].vue +212 -0
  201. package/pages/videos/index.vue +327 -0
  202. package/pages/videos/submit.vue +112 -0
  203. package/plugins/auth.ts +23 -0
  204. package/server/api/admin/audit.get.ts +17 -0
  205. package/server/api/admin/content/[id].delete.ts +15 -0
  206. package/server/api/admin/content/[id].patch.ts +37 -0
  207. package/server/api/admin/federation/activity.get.ts +31 -0
  208. package/server/api/admin/federation/clients.get.ts +9 -0
  209. package/server/api/admin/federation/clients.post.ts +16 -0
  210. package/server/api/admin/federation/hub-mirrors/index.get.ts +10 -0
  211. package/server/api/admin/federation/hub-mirrors/index.post.ts +42 -0
  212. package/server/api/admin/federation/mirrors/[id]/backfill.post.ts +39 -0
  213. package/server/api/admin/federation/mirrors/[id].delete.ts +11 -0
  214. package/server/api/admin/federation/mirrors/[id].get.ts +15 -0
  215. package/server/api/admin/federation/mirrors/[id].put.ts +22 -0
  216. package/server/api/admin/federation/mirrors/index.get.ts +9 -0
  217. package/server/api/admin/federation/mirrors/index.post.ts +30 -0
  218. package/server/api/admin/federation/pending.get.ts +22 -0
  219. package/server/api/admin/federation/refederate.post.ts +92 -0
  220. package/server/api/admin/federation/repair-types.post.ts +57 -0
  221. package/server/api/admin/federation/retry.post.ts +41 -0
  222. package/server/api/admin/federation/stats.get.ts +26 -0
  223. package/server/api/admin/reports/[id]/resolve.post.ts +12 -0
  224. package/server/api/admin/reports.get.ts +17 -0
  225. package/server/api/admin/settings.get.ts +22 -0
  226. package/server/api/admin/settings.put.ts +11 -0
  227. package/server/api/admin/stats.get.ts +9 -0
  228. package/server/api/admin/users/[id]/role.put.ts +12 -0
  229. package/server/api/admin/users/[id]/status.put.ts +12 -0
  230. package/server/api/admin/users/[id].delete.ts +10 -0
  231. package/server/api/admin/users.get.ts +18 -0
  232. package/server/api/auth/federated/callback.get.ts +67 -0
  233. package/server/api/auth/federated/login.post.ts +60 -0
  234. package/server/api/auth/oauth2/authorize.get.ts +30 -0
  235. package/server/api/auth/oauth2/authorize.post.ts +51 -0
  236. package/server/api/auth/oauth2/register.post.ts +41 -0
  237. package/server/api/auth/oauth2/token.post.ts +48 -0
  238. package/server/api/cert/[code].get.ts +13 -0
  239. package/server/api/content/[id]/build.post.ts +9 -0
  240. package/server/api/content/[id]/fork.post.ts +10 -0
  241. package/server/api/content/[id]/index.delete.ts +18 -0
  242. package/server/api/content/[id]/index.get.ts +15 -0
  243. package/server/api/content/[id]/index.put.ts +23 -0
  244. package/server/api/content/[id]/products/[productId].delete.ts +28 -0
  245. package/server/api/content/[id]/products-sync.post.ts +29 -0
  246. package/server/api/content/[id]/products.get.ts +9 -0
  247. package/server/api/content/[id]/products.post.ts +31 -0
  248. package/server/api/content/[id]/publish.post.ts +17 -0
  249. package/server/api/content/[id]/report.post.ts +17 -0
  250. package/server/api/content/[id]/versions.get.ts +9 -0
  251. package/server/api/content/[id]/view.post.ts +34 -0
  252. package/server/api/content/index.get.ts +23 -0
  253. package/server/api/content/index.post.ts +11 -0
  254. package/server/api/contests/[slug]/entries.get.ts +18 -0
  255. package/server/api/contests/[slug]/entries.post.ts +23 -0
  256. package/server/api/contests/[slug]/index.delete.ts +21 -0
  257. package/server/api/contests/[slug]/index.get.ts +11 -0
  258. package/server/api/contests/[slug]/index.put.ts +15 -0
  259. package/server/api/contests/[slug]/judge.post.ts +12 -0
  260. package/server/api/contests/[slug]/transition.post.ts +24 -0
  261. package/server/api/contests/index.get.ts +10 -0
  262. package/server/api/contests/index.post.ts +28 -0
  263. package/server/api/docs/[siteSlug]/index.delete.ts +14 -0
  264. package/server/api/docs/[siteSlug]/index.get.ts +17 -0
  265. package/server/api/docs/[siteSlug]/index.put.ts +20 -0
  266. package/server/api/docs/[siteSlug]/nav.get.ts +26 -0
  267. package/server/api/docs/[siteSlug]/pages/[pageId].delete.ts +14 -0
  268. package/server/api/docs/[siteSlug]/pages/[pageId].get.ts +31 -0
  269. package/server/api/docs/[siteSlug]/pages/[pageId].put.ts +15 -0
  270. package/server/api/docs/[siteSlug]/pages/index.get.ts +34 -0
  271. package/server/api/docs/[siteSlug]/pages/index.post.ts +28 -0
  272. package/server/api/docs/[siteSlug]/pages/reorder.post.ts +26 -0
  273. package/server/api/docs/[siteSlug]/search.get.ts +20 -0
  274. package/server/api/docs/[siteSlug]/versions.post.ts +11 -0
  275. package/server/api/docs/index.get.ts +6 -0
  276. package/server/api/docs/index.post.ts +10 -0
  277. package/server/api/federated-hubs/[id]/posts/[postId].get.ts +16 -0
  278. package/server/api/federated-hubs/[id]/posts.get.ts +15 -0
  279. package/server/api/federated-hubs/[id].get.ts +16 -0
  280. package/server/api/federation/boost.post.ts +21 -0
  281. package/server/api/federation/content/[id].get.ts +15 -0
  282. package/server/api/federation/follow.post.ts +16 -0
  283. package/server/api/federation/health.get.ts +56 -0
  284. package/server/api/federation/hub-follow.post.ts +27 -0
  285. package/server/api/federation/hub-post-like.post.ts +115 -0
  286. package/server/api/federation/hub-post-likes.get.ts +24 -0
  287. package/server/api/federation/hub-post-reply.post.ts +42 -0
  288. package/server/api/federation/hub-post.post.ts +33 -0
  289. package/server/api/federation/like.post.ts +21 -0
  290. package/server/api/federation/remote-actor.get.ts +22 -0
  291. package/server/api/federation/reply.post.ts +22 -0
  292. package/server/api/federation/search.post.ts +17 -0
  293. package/server/api/federation/timeline.get.ts +22 -0
  294. package/server/api/federation/unfollow.post.ts +17 -0
  295. package/server/api/files/[id].delete.ts +35 -0
  296. package/server/api/files/mine.get.ts +31 -0
  297. package/server/api/files/upload-from-url.post.ts +68 -0
  298. package/server/api/files/upload.post.ts +105 -0
  299. package/server/api/health.get.ts +4 -0
  300. package/server/api/hubs/[slug]/bans/[userId].delete.ts +13 -0
  301. package/server/api/hubs/[slug]/bans.get.ts +20 -0
  302. package/server/api/hubs/[slug]/bans.post.ts +23 -0
  303. package/server/api/hubs/[slug]/feed.xml.get.ts +60 -0
  304. package/server/api/hubs/[slug]/gallery.get.ts +21 -0
  305. package/server/api/hubs/[slug]/index.delete.ts +18 -0
  306. package/server/api/hubs/[slug]/index.get.ts +14 -0
  307. package/server/api/hubs/[slug]/index.put.ts +22 -0
  308. package/server/api/hubs/[slug]/invites.get.ts +20 -0
  309. package/server/api/hubs/[slug]/invites.post.ts +27 -0
  310. package/server/api/hubs/[slug]/join.post.ts +17 -0
  311. package/server/api/hubs/[slug]/leave.post.ts +13 -0
  312. package/server/api/hubs/[slug]/members/[userId].delete.ts +13 -0
  313. package/server/api/hubs/[slug]/members/[userId].put.ts +16 -0
  314. package/server/api/hubs/[slug]/members.get.ts +20 -0
  315. package/server/api/hubs/[slug]/posts/[postId]/like.post.ts +21 -0
  316. package/server/api/hubs/[slug]/posts/[postId]/lock.post.ts +17 -0
  317. package/server/api/hubs/[slug]/posts/[postId]/pin.post.ts +17 -0
  318. package/server/api/hubs/[slug]/posts/[postId]/replies.get.ts +16 -0
  319. package/server/api/hubs/[slug]/posts/[postId]/replies.post.ts +21 -0
  320. package/server/api/hubs/[slug]/posts/[postId].delete.ts +23 -0
  321. package/server/api/hubs/[slug]/posts/[postId].get.ts +16 -0
  322. package/server/api/hubs/[slug]/posts/index.get.ts +16 -0
  323. package/server/api/hubs/[slug]/posts/index.post.ts +36 -0
  324. package/server/api/hubs/[slug]/products.get.ts +26 -0
  325. package/server/api/hubs/[slug]/products.post.ts +18 -0
  326. package/server/api/hubs/[slug]/share.post.ts +39 -0
  327. package/server/api/hubs/index.get.ts +15 -0
  328. package/server/api/hubs/index.post.ts +11 -0
  329. package/server/api/image-proxy.get.ts +91 -0
  330. package/server/api/learn/[slug]/[lessonSlug]/complete.post.ts +13 -0
  331. package/server/api/learn/[slug]/[lessonSlug]/index.get.ts +68 -0
  332. package/server/api/learn/[slug]/enroll.post.ts +12 -0
  333. package/server/api/learn/[slug]/index.delete.ts +12 -0
  334. package/server/api/learn/[slug]/index.get.ts +14 -0
  335. package/server/api/learn/[slug]/index.put.ts +19 -0
  336. package/server/api/learn/[slug]/lessons/[lessonId].delete.ts +14 -0
  337. package/server/api/learn/[slug]/lessons/[lessonId].put.ts +24 -0
  338. package/server/api/learn/[slug]/lessons.post.ts +10 -0
  339. package/server/api/learn/[slug]/modules/[moduleId].delete.ts +14 -0
  340. package/server/api/learn/[slug]/modules/[moduleId].put.ts +15 -0
  341. package/server/api/learn/[slug]/modules.post.ts +14 -0
  342. package/server/api/learn/[slug]/publish.post.ts +13 -0
  343. package/server/api/learn/[slug]/unenroll.post.ts +12 -0
  344. package/server/api/learn/certificates.get.ts +9 -0
  345. package/server/api/learn/enrollments.get.ts +9 -0
  346. package/server/api/learn/index.get.ts +17 -0
  347. package/server/api/learn/index.post.ts +11 -0
  348. package/server/api/me.get.ts +13 -0
  349. package/server/api/messages/[conversationId]/info.get.ts +43 -0
  350. package/server/api/messages/[conversationId]/stream.get.ts +73 -0
  351. package/server/api/messages/[conversationId].get.ts +13 -0
  352. package/server/api/messages/[conversationId].post.ts +12 -0
  353. package/server/api/messages/index.get.ts +39 -0
  354. package/server/api/messages/index.post.ts +58 -0
  355. package/server/api/notifications/[id].delete.ts +11 -0
  356. package/server/api/notifications/count.get.ts +10 -0
  357. package/server/api/notifications/index.get.ts +24 -0
  358. package/server/api/notifications/read.post.ts +15 -0
  359. package/server/api/notifications/stream.get.ts +59 -0
  360. package/server/api/openapi.get.ts +5 -0
  361. package/server/api/products/[id].delete.ts +15 -0
  362. package/server/api/products/[id].put.ts +18 -0
  363. package/server/api/products/[slug]/content.get.ts +21 -0
  364. package/server/api/products/[slug].get.ts +16 -0
  365. package/server/api/products/index.get.ts +28 -0
  366. package/server/api/profile.get.ts +15 -0
  367. package/server/api/profile.put.ts +24 -0
  368. package/server/api/resolve-identity.post.ts +34 -0
  369. package/server/api/search/index.get.ts +24 -0
  370. package/server/api/search/trending.get.ts +22 -0
  371. package/server/api/social/bookmark.get.ts +16 -0
  372. package/server/api/social/bookmark.post.ts +15 -0
  373. package/server/api/social/bookmarks.get.ts +16 -0
  374. package/server/api/social/comments/[id].delete.ts +13 -0
  375. package/server/api/social/comments.get.ts +18 -0
  376. package/server/api/social/comments.post.ts +11 -0
  377. package/server/api/social/like.get.ts +17 -0
  378. package/server/api/social/like.post.ts +35 -0
  379. package/server/api/stats.get.ts +8 -0
  380. package/server/api/user/hubs.get.ts +22 -0
  381. package/server/api/users/[username]/content.get.ts +21 -0
  382. package/server/api/users/[username]/feed.xml.get.ts +65 -0
  383. package/server/api/users/[username]/follow.delete.ts +15 -0
  384. package/server/api/users/[username]/follow.post.ts +15 -0
  385. package/server/api/users/[username]/followers.get.ts +22 -0
  386. package/server/api/users/[username]/following.get.ts +22 -0
  387. package/server/api/users/[username]/learning.get.ts +18 -0
  388. package/server/api/users/[username].get.ts +25 -0
  389. package/server/api/users/index.get.ts +78 -0
  390. package/server/api/videos/[id].get.ts +18 -0
  391. package/server/api/videos/categories/[id].delete.ts +15 -0
  392. package/server/api/videos/categories/[id].put.ts +17 -0
  393. package/server/api/videos/categories.get.ts +7 -0
  394. package/server/api/videos/categories.post.ts +11 -0
  395. package/server/api/videos/index.get.ts +20 -0
  396. package/server/api/videos/index.post.ts +11 -0
  397. package/server/middleware/auth.ts +180 -0
  398. package/server/middleware/features.ts +31 -0
  399. package/server/middleware/security.ts +54 -0
  400. package/server/plugins/auto-admin.ts +69 -0
  401. package/server/plugins/federation-delivery.ts +74 -0
  402. package/server/routes/.well-known/nodeinfo.ts +15 -0
  403. package/server/routes/.well-known/webfinger.ts +86 -0
  404. package/server/routes/actor/followers.ts +34 -0
  405. package/server/routes/actor/following.ts +34 -0
  406. package/server/routes/actor/outbox.ts +31 -0
  407. package/server/routes/actor.ts +30 -0
  408. package/server/routes/feed.xml.ts +59 -0
  409. package/server/routes/hubs/[slug]/followers.ts +39 -0
  410. package/server/routes/hubs/[slug]/inbox.ts +35 -0
  411. package/server/routes/hubs/[slug]/outbox.ts +43 -0
  412. package/server/routes/hubs/[slug]/posts/[postId].ts +63 -0
  413. package/server/routes/hubs/[slug].ts +29 -0
  414. package/server/routes/inbox.ts +34 -0
  415. package/server/routes/nodeinfo/2.1.ts +27 -0
  416. package/server/routes/robots.txt.ts +19 -0
  417. package/server/routes/sitemap.xml.ts +105 -0
  418. package/server/routes/users/[username]/followers.ts +33 -0
  419. package/server/routes/users/[username]/following.ts +33 -0
  420. package/server/routes/users/[username]/inbox.ts +34 -0
  421. package/server/routes/users/[username]/outbox.ts +35 -0
  422. package/server/routes/users/[username].ts +62 -0
  423. package/server/utils/auth.ts +36 -0
  424. package/server/utils/db.ts +34 -0
  425. package/server/utils/errors.ts +24 -0
  426. package/server/utils/inbox.ts +122 -0
  427. package/server/utils/validate.ts +82 -0
  428. package/theme/base.css +283 -0
  429. package/theme/components.css +322 -0
  430. package/theme/dark.css +87 -0
  431. package/theme/editor-panels.css +63 -0
  432. package/theme/forms.css +216 -0
  433. package/theme/generics.css +68 -0
  434. package/theme/layouts.css +415 -0
  435. package/theme/prose.css +342 -0
  436. package/types/hub.ts +61 -0
@@ -0,0 +1,503 @@
1
+ <script setup lang="ts">
2
+ import type { BlockEditor } from '../../composables/useBlockEditor';
3
+ import type { BlockTypeGroup } from './BlockPicker.vue';
4
+
5
+ const props = defineProps<{
6
+ blockEditor: BlockEditor;
7
+ metadata: Record<string, unknown>;
8
+ }>();
9
+
10
+ const emit = defineEmits<{
11
+ 'update:metadata': [metadata: Record<string, unknown>];
12
+ }>();
13
+
14
+ function updateMeta(key: string, value: unknown): void {
15
+ emit('update:metadata', { ...props.metadata, [key]: value });
16
+ }
17
+
18
+ const activeLeftTab = ref<'modules' | 'structure' | 'assets'>('modules');
19
+
20
+ // Interactive blocks FIRST — they're the core of an explainer
21
+ const blockTypes: BlockTypeGroup[] = [
22
+ {
23
+ name: 'Interactive',
24
+ blocks: [
25
+ { type: 'interactiveSlider', label: 'Range Slider', icon: 'fa-sliders', description: 'Slider with feedback ranges' },
26
+ { type: 'quiz', label: 'Knowledge Check', icon: 'fa-circle-question', description: 'Quiz with answer feedback' },
27
+ { type: 'checkpoint', label: 'Checkpoint', icon: 'fa-flag-checkered', description: 'Section completion marker' },
28
+ { type: 'callout', label: 'Key Insight', icon: 'fa-lightbulb', description: 'Highlight discovery moments', attrs: { variant: 'tip' } },
29
+ ],
30
+ },
31
+ {
32
+ name: 'Content',
33
+ blocks: [
34
+ { type: 'paragraph', label: 'Body Text', icon: 'fa-align-left', description: '2-3 short paragraphs max' },
35
+ { type: 'heading', label: 'Heading', icon: 'fa-heading', description: 'In-section heading (H2/H3)' },
36
+ { type: 'image', label: 'Diagram', icon: 'fa-image', description: 'Visual explanation' },
37
+ { type: 'code_block', label: 'Code Example', icon: 'fa-code', description: 'Runnable code snippet' },
38
+ ],
39
+ },
40
+ {
41
+ name: 'Data & Viz',
42
+ blocks: [
43
+ { type: 'mathNotation', label: 'Math Block', icon: 'fa-square-root-variable', description: 'Formula or equation' },
44
+ { type: 'embed', label: 'Embed', icon: 'fa-globe', description: 'External interactive' },
45
+ { type: 'callout', label: 'Warning', icon: 'fa-triangle-exclamation', description: 'Important caveat', attrs: { variant: 'warning' } },
46
+ { type: 'markdown', label: 'Markdown', icon: 'fa-brands fa-markdown', description: 'Raw markdown block' },
47
+ ],
48
+ },
49
+ {
50
+ name: 'Structure',
51
+ blocks: [
52
+ { type: 'sectionHeader', label: 'Section Header', icon: 'fa-heading', description: 'Tag + title + intro — starts a section' },
53
+ { type: 'horizontal_rule', label: 'Section Divider', icon: 'fa-minus', description: 'Visual break' },
54
+ ],
55
+ },
56
+ ];
57
+
58
+ // Interactive block types for counting
59
+ const INTERACTIVE_TYPES = new Set(['interactiveSlider', 'slider', 'quiz', 'checkpoint']);
60
+
61
+ // --- Structure: sections from sectionHeader blocks (fallback to H2 headings) ---
62
+ interface ExplainerSection {
63
+ id: string;
64
+ title: string;
65
+ blockCount: number;
66
+ interactiveCount: number;
67
+ index: number;
68
+ }
69
+
70
+ const structureSections = computed<ExplainerSection[]>(() => {
71
+ const result: ExplainerSection[] = [];
72
+ const blocks = props.blockEditor.blocks.value;
73
+ let current: ExplainerSection | null = null;
74
+ let idx = 0;
75
+
76
+ // Check if any sectionHeader blocks exist
77
+ const hasSectionHeaders = blocks.some(b => b.type === 'sectionHeader');
78
+ const sectionType = hasSectionHeaders ? 'sectionHeader' : 'heading';
79
+
80
+ for (const block of blocks) {
81
+ const isSectionStart = sectionType === 'sectionHeader'
82
+ ? block.type === 'sectionHeader'
83
+ : block.type === 'heading' && ((block.content.level as number) ?? 2) <= 2;
84
+
85
+ if (isSectionStart) {
86
+ if (current) result.push(current);
87
+ idx++;
88
+ const title = sectionType === 'sectionHeader'
89
+ ? (block.content.title as string) || 'Untitled section'
90
+ : (block.content.text as string) || 'Untitled section';
91
+ current = {
92
+ id: block.id,
93
+ title,
94
+ blockCount: 1,
95
+ interactiveCount: 0,
96
+ index: idx,
97
+ };
98
+ } else if (current) {
99
+ current.blockCount++;
100
+ if (INTERACTIVE_TYPES.has(block.type)) {
101
+ current.interactiveCount++;
102
+ }
103
+ }
104
+ }
105
+ if (current) result.push(current);
106
+ return result;
107
+ });
108
+
109
+ // Total interactive blocks across all content
110
+ const totalInteractives = computed(() => {
111
+ return props.blockEditor.blocks.value.filter(b => INTERACTIVE_TYPES.has(b.type)).length;
112
+ });
113
+
114
+ // --- Assets ---
115
+ const uploadedFiles = ref<Array<{ name: string; size: string; type: string }>>([]);
116
+
117
+ function onAssetUpload(event: Event): void {
118
+ const input = event.target as HTMLInputElement;
119
+ if (!input.files?.length) return;
120
+ const file = input.files[0];
121
+ if (!file) return;
122
+ const formData = new FormData();
123
+ formData.append('file', file);
124
+ formData.append('purpose', 'content');
125
+ $fetch<{ url: string; originalName: string; size: number }>('/api/files/upload', { method: 'POST', body: formData })
126
+ .then((res) => {
127
+ uploadedFiles.value.unshift({
128
+ name: res.originalName || file.name,
129
+ size: `${(res.size / 1024).toFixed(0)} KB`,
130
+ type: file.type.startsWith('image/') ? 'image' : 'file',
131
+ });
132
+ })
133
+ .catch(() => { /* silent */ });
134
+ }
135
+
136
+ const openSections = ref<Record<string, boolean>>({
137
+ section: true, difficulty: true, visibility: false, cover: false,
138
+ });
139
+ function toggleSection(key: string): void {
140
+ openSections.value[key] = !openSections.value[key];
141
+ }
142
+
143
+ const tags = computed(() => (props.metadata.tags as string[]) || []);
144
+ function onTagsUpdate(newTags: string[]): void { updateMeta('tags', newTags); }
145
+ const visibility = computed(() => (props.metadata.visibility as string) || 'public');
146
+ function onVisibilityUpdate(val: string): void { updateMeta('visibility', val); }
147
+
148
+ // --- Mobile sidebar toggles ---
149
+ const mobileLeftOpen = ref(false);
150
+ const mobileRightOpen = ref(false);
151
+ function toggleMobileLeft(): void {
152
+ mobileLeftOpen.value = !mobileLeftOpen.value;
153
+ if (mobileLeftOpen.value) mobileRightOpen.value = false;
154
+ }
155
+ function toggleMobileRight(): void {
156
+ mobileRightOpen.value = !mobileRightOpen.value;
157
+ if (mobileRightOpen.value) mobileLeftOpen.value = false;
158
+ }
159
+ function closeMobileSidebars(): void {
160
+ mobileLeftOpen.value = false;
161
+ mobileRightOpen.value = false;
162
+ }
163
+
164
+ // --- Canvas toolbar ---
165
+ const viewportMode = ref<'desktop' | 'tablet' | 'mobile'>('desktop');
166
+ const canvasMaxWidth = computed(() => {
167
+ if (viewportMode.value === 'mobile') return '375px';
168
+ if (viewportMode.value === 'tablet') return '768px';
169
+ return '720px';
170
+ });
171
+
172
+ // --- Word count / status bar ---
173
+ const wordCount = computed(() => {
174
+ let count = 0;
175
+ for (const block of props.blockEditor.blocks.value) {
176
+ const html = (block.content.html as string) || '';
177
+ const text = (block.content.text as string) || '';
178
+ const code = (block.content.code as string) || '';
179
+ const combined = html.replace(/<[^>]*>/g, ' ') + ' ' + text + ' ' + code;
180
+ count += combined.split(/\s+/).filter((w) => w.length > 0).length;
181
+ }
182
+ return count;
183
+ });
184
+ const readTime = computed(() => Math.max(1, Math.round(wordCount.value / 200)));
185
+ const blockCount = computed(() => props.blockEditor.blocks.value.length);
186
+ </script>
187
+
188
+ <template>
189
+ <div class="cpub-ee-shell">
190
+ <!-- Mobile sidebar toggles -->
191
+ <div class="cpub-ee-mobile-toggles">
192
+ <button class="cpub-ee-mobile-btn" aria-label="Toggle modules panel" @click="toggleMobileLeft"><i class="fa-solid fa-layer-group"></i></button>
193
+ <button class="cpub-ee-mobile-btn" aria-label="Toggle properties panel" @click="toggleMobileRight"><i class="fa-solid fa-sliders"></i></button>
194
+ </div>
195
+ <div v-if="mobileLeftOpen || mobileRightOpen" class="cpub-ee-mobile-overlay" @click="closeMobileSidebars" />
196
+
197
+ <!-- LEFT: Modules/Structure/Assets -->
198
+ <aside class="cpub-ee-left" :class="{ 'cpub-ee-sidebar-open': mobileLeftOpen }" aria-label="Editor sidebar">
199
+ <div class="cpub-ee-left-tabs">
200
+ <button class="cpub-ee-left-tab" :class="{ active: activeLeftTab === 'modules' }" @click="activeLeftTab = 'modules'">Modules</button>
201
+ <button class="cpub-ee-left-tab" :class="{ active: activeLeftTab === 'structure' }" @click="activeLeftTab = 'structure'">Structure</button>
202
+ <button class="cpub-ee-left-tab" :class="{ active: activeLeftTab === 'assets' }" @click="activeLeftTab = 'assets'">Assets</button>
203
+ </div>
204
+
205
+ <div v-if="activeLeftTab === 'modules'" class="cpub-ee-left-body">
206
+ <EditorsEditorBlocks :groups="blockTypes" :block-editor="blockEditor" />
207
+ </div>
208
+
209
+ <div v-else-if="activeLeftTab === 'structure'" class="cpub-ee-left-body" style="padding: 10px;">
210
+ <!-- Flow guidance -->
211
+ <div class="cpub-ee-flow-guide">
212
+ <div class="cpub-ee-flow-title"><i class="fa-solid fa-route"></i> Section Flow</div>
213
+ <div class="cpub-ee-flow-steps">
214
+ <span class="cpub-ee-flow-step">Question</span>
215
+ <i class="fa-solid fa-arrow-right cpub-ee-flow-arrow"></i>
216
+ <span class="cpub-ee-flow-step cpub-ee-flow-step--interactive">Interact</span>
217
+ <i class="fa-solid fa-arrow-right cpub-ee-flow-arrow"></i>
218
+ <span class="cpub-ee-flow-step">Insight</span>
219
+ <i class="fa-solid fa-arrow-right cpub-ee-flow-arrow"></i>
220
+ <span class="cpub-ee-flow-step">Bridge</span>
221
+ </div>
222
+ </div>
223
+
224
+ <!-- Interactive count summary -->
225
+ <div class="cpub-ee-interactive-summary">
226
+ <span class="cpub-ee-interactive-count">{{ totalInteractives }}</span>
227
+ <span>interactive{{ totalInteractives === 1 ? '' : 's' }} across {{ structureSections.length }} section{{ structureSections.length === 1 ? '' : 's' }}</span>
228
+ </div>
229
+
230
+ <!-- Section list -->
231
+ <div v-if="structureSections.length > 0" class="cpub-ee-section-list">
232
+ <div
233
+ v-for="section in structureSections"
234
+ :key="section.id"
235
+ class="cpub-ee-section-item"
236
+ @click="blockEditor.selectBlock(section.id)"
237
+ >
238
+ <span class="cpub-ee-section-num">{{ String(section.index).padStart(2, '0') }}</span>
239
+ <div class="cpub-ee-section-info">
240
+ <span class="cpub-ee-section-title">{{ section.title }}</span>
241
+ <span class="cpub-ee-section-meta">
242
+ {{ section.blockCount }} blocks
243
+ <template v-if="section.interactiveCount > 0">
244
+ <span class="cpub-ee-section-interactive-badge">
245
+ <i class="fa-solid fa-bolt"></i> {{ section.interactiveCount }}
246
+ </span>
247
+ </template>
248
+ <template v-else>
249
+ <span class="cpub-ee-section-no-interactive">no interactive</span>
250
+ </template>
251
+ </span>
252
+ </div>
253
+ </div>
254
+ </div>
255
+
256
+ <div v-else class="cpub-ee-structure-empty">
257
+ <i class="fa-solid fa-layer-group"></i>
258
+ <p>Add Section Header blocks to create sections.</p>
259
+ <p>Each section should follow:<br /><strong>Question &rarr; Interactive &rarr; Insight</strong></p>
260
+ </div>
261
+ </div>
262
+
263
+ <div v-else class="cpub-ee-left-body">
264
+ <label class="cpub-ee-assets-drop">
265
+ <i class="fa-solid fa-cloud-arrow-up"></i>
266
+ <div class="cpub-ee-assets-drop-label">Drop files here</div>
267
+ <div class="cpub-ee-assets-drop-sub">JPG, PNG, GIF, SVG, PDF</div>
268
+ <input type="file" class="cpub-sr-only" @change="onAssetUpload">
269
+ </label>
270
+ <div v-if="uploadedFiles.length > 0" class="cpub-ee-assets-list">
271
+ <div class="cpub-ee-assets-heading">Recent Uploads</div>
272
+ <div
273
+ v-for="(file, idx) in uploadedFiles"
274
+ :key="idx"
275
+ class="cpub-ee-asset-item"
276
+ >
277
+ <div class="cpub-ee-asset-icon">
278
+ <i :class="file.type === 'image' ? 'fa-solid fa-image' : 'fa-solid fa-file'" />
279
+ </div>
280
+ <div class="cpub-ee-asset-info">
281
+ <div class="cpub-ee-asset-name">{{ file.name }}</div>
282
+ <div class="cpub-ee-asset-size">{{ file.size }}</div>
283
+ </div>
284
+ </div>
285
+ </div>
286
+ </div>
287
+ </aside>
288
+
289
+ <!-- CENTER: Canvas with toolbar -->
290
+ <div class="cpub-ee-center">
291
+ <!-- Canvas toolbar -->
292
+ <div class="cpub-ee-canvas-toolbar">
293
+ <div class="cpub-ee-viewport-tabs">
294
+ <button class="cpub-ee-viewport-tab" :class="{ active: viewportMode === 'desktop' }" title="Desktop" @click="viewportMode = 'desktop'"><i class="fa-solid fa-desktop"></i></button>
295
+ <button class="cpub-ee-viewport-tab" :class="{ active: viewportMode === 'tablet' }" title="Tablet" @click="viewportMode = 'tablet'"><i class="fa-solid fa-tablet-screen-button"></i></button>
296
+ <button class="cpub-ee-viewport-tab" :class="{ active: viewportMode === 'mobile' }" title="Mobile" @click="viewportMode = 'mobile'"><i class="fa-solid fa-mobile-screen"></i></button>
297
+ </div>
298
+ </div>
299
+
300
+ <div class="cpub-ee-canvas">
301
+ <div class="cpub-ee-canvas-inner" :style="{ maxWidth: canvasMaxWidth }">
302
+ <EditorsBlockCanvas :block-editor="blockEditor" :block-types="blockTypes" />
303
+ </div>
304
+ </div>
305
+
306
+ <!-- Status bar -->
307
+ <div class="cpub-ee-statusbar">
308
+ <div class="cpub-ee-status-item"><i class="fa-solid fa-layer-group"></i> <span>{{ structureSections.length }} sections</span></div>
309
+ <div class="cpub-ee-status-sep" />
310
+ <div class="cpub-ee-status-item"><i class="fa-solid fa-bolt"></i> <span>{{ totalInteractives }} interactives</span></div>
311
+ <div class="cpub-ee-status-sep" />
312
+ <div class="cpub-ee-status-item"><i class="fa-solid fa-align-justify"></i> <span>{{ wordCount.toLocaleString() }} words</span></div>
313
+ <div class="cpub-ee-status-sep" />
314
+ <div class="cpub-ee-status-item"><i class="fa-regular fa-clock"></i> <span>~{{ readTime }} min</span></div>
315
+ </div>
316
+ </div>
317
+
318
+ <!-- RIGHT: Properties -->
319
+ <aside class="cpub-ee-right" :class="{ 'cpub-ee-sidebar-open': mobileRightOpen }" aria-label="Explainer properties">
320
+ <div class="cpub-ee-right-body">
321
+ <EditorsEditorSection title="Content" icon="fa-sliders" :open="openSections.section" @toggle="toggleSection('section')">
322
+ <div class="cpub-ep-field">
323
+ <label class="cpub-ep-flabel">Slug</label>
324
+ <input class="cpub-ep-input" type="text" :value="metadata.slug" placeholder="auto-generated" @input="updateMeta('slug', ($event.target as HTMLInputElement).value)">
325
+ </div>
326
+ <div class="cpub-ep-field">
327
+ <label class="cpub-ep-flabel">Description</label>
328
+ <textarea class="cpub-ep-textarea" rows="3" :value="metadata.description as string" placeholder="What does this explainer teach?" @input="updateMeta('description', ($event.target as HTMLTextAreaElement).value)" />
329
+ </div>
330
+ <div class="cpub-ep-field">
331
+ <label class="cpub-ep-flabel">Estimated Minutes</label>
332
+ <input class="cpub-ep-input" type="number" :value="metadata.estimatedMinutes" placeholder="10" @input="updateMeta('estimatedMinutes', Number(($event.target as HTMLInputElement).value))">
333
+ </div>
334
+ <div class="cpub-ep-field">
335
+ <label class="cpub-ep-flabel">Tags</label>
336
+ <EditorsEditorTagInput :tags="tags" @update:tags="onTagsUpdate" />
337
+ </div>
338
+ </EditorsEditorSection>
339
+
340
+ <EditorsEditorSection title="Difficulty" icon="fa-gauge-high" :open="openSections.difficulty" @toggle="toggleSection('difficulty')">
341
+ <select class="cpub-ep-select" :value="metadata.difficulty || 'beginner'" @change="updateMeta('difficulty', ($event.target as HTMLSelectElement).value)">
342
+ <option value="beginner">Beginner</option>
343
+ <option value="intermediate">Intermediate</option>
344
+ <option value="advanced">Advanced</option>
345
+ </select>
346
+ </EditorsEditorSection>
347
+
348
+ <EditorsEditorSection title="Visibility" icon="fa-eye" :open="openSections.visibility" @toggle="toggleSection('visibility')">
349
+ <EditorsEditorVisibility :model-value="visibility" @update:model-value="onVisibilityUpdate" />
350
+ </EditorsEditorSection>
351
+
352
+ <EditorsEditorSection title="Cover Image" icon="fa-image" :open="openSections.cover" @toggle="toggleSection('cover')">
353
+ <div class="cpub-ep-field">
354
+ <input class="cpub-ep-input" type="url" :value="metadata.coverImageUrl" placeholder="https://..." @input="updateMeta('coverImageUrl', ($event.target as HTMLInputElement).value)">
355
+ </div>
356
+ </EditorsEditorSection>
357
+ </div>
358
+ </aside>
359
+ </div>
360
+ </template>
361
+
362
+ <style scoped>
363
+ .cpub-ee-shell { display: flex; flex: 1; overflow: hidden; }
364
+ .cpub-ee-left { width: 240px; flex-shrink: 0; background: var(--surface); border-right: var(--border-width-default) solid var(--border); display: flex; flex-direction: column; overflow: hidden; }
365
+ .cpub-ee-left-tabs { display: flex; border-bottom: var(--border-width-default) solid var(--border); flex-shrink: 0; }
366
+ .cpub-ee-left-tab {
367
+ flex: 1; padding: 8px; font-family: var(--font-mono); font-size: 10px; font-weight: 600;
368
+ letter-spacing: 0.06em; text-transform: uppercase; text-align: center;
369
+ background: none; border: none; color: var(--text-dim); cursor: pointer;
370
+ border-bottom: var(--border-width-default) solid transparent; margin-bottom: -2px;
371
+ }
372
+ .cpub-ee-left-tab.active { color: var(--accent); border-bottom-color: var(--accent); background: var(--accent-bg); }
373
+ .cpub-ee-left-body { flex: 1; overflow-y: auto; }
374
+
375
+ /* Flow guide */
376
+ .cpub-ee-flow-guide {
377
+ background: var(--accent-bg); border: var(--border-width-default) solid var(--accent-border); padding: 10px 12px; margin-bottom: 8px;
378
+ }
379
+ .cpub-ee-flow-title { font-family: var(--font-mono); font-size: 9px; font-weight: 700; letter-spacing: 0.08em; text-transform: uppercase; color: var(--accent); margin-bottom: 8px; display: flex; align-items: center; gap: 6px; }
380
+ .cpub-ee-flow-steps { display: flex; align-items: center; gap: 4px; flex-wrap: wrap; }
381
+ .cpub-ee-flow-step { font-size: 10px; font-family: var(--font-mono); color: var(--text-dim); padding: 2px 6px; background: var(--surface); border: var(--border-width-default) solid var(--border2); }
382
+ .cpub-ee-flow-step--interactive { color: var(--accent); border-color: var(--accent-border); font-weight: 600; }
383
+ .cpub-ee-flow-arrow { font-size: 7px; color: var(--text-faint); }
384
+
385
+ /* Interactive summary */
386
+ .cpub-ee-interactive-summary {
387
+ display: flex; align-items: center; gap: 6px; padding: 6px 8px; margin-bottom: 8px;
388
+ font-size: 10px; font-family: var(--font-mono); color: var(--text-dim);
389
+ }
390
+ .cpub-ee-interactive-count { font-size: 16px; font-weight: 700; color: var(--accent); }
391
+
392
+ /* Section list */
393
+ .cpub-ee-section-list { display: flex; flex-direction: column; gap: 2px; }
394
+ .cpub-ee-section-item { display: flex; align-items: flex-start; gap: 8px; padding: 8px; cursor: pointer; border: var(--border-width-default) solid transparent; transition: all 0.1s; }
395
+ .cpub-ee-section-item:hover { background: var(--surface2); border-color: var(--border2); }
396
+ .cpub-ee-section-num { font-family: var(--font-mono); font-size: 10px; font-weight: 700; color: var(--text-faint); min-width: 18px; margin-top: 1px; }
397
+ .cpub-ee-section-info { flex: 1; min-width: 0; display: flex; flex-direction: column; gap: 2px; }
398
+ .cpub-ee-section-title { font-size: 12px; color: var(--text); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
399
+ .cpub-ee-section-meta { font-size: 10px; font-family: var(--font-mono); color: var(--text-faint); display: flex; align-items: center; gap: 6px; }
400
+ .cpub-ee-section-interactive-badge { display: inline-flex; align-items: center; gap: 3px; color: var(--accent); font-weight: 600; }
401
+ .cpub-ee-section-no-interactive { color: var(--yellow); font-style: italic; }
402
+
403
+ /* Empty state */
404
+ .cpub-ee-structure-empty {
405
+ text-align: center; padding: 20px 12px; color: var(--text-dim);
406
+ }
407
+ .cpub-ee-structure-empty i { font-size: 20px; color: var(--text-faint); margin-bottom: 8px; }
408
+ .cpub-ee-structure-empty p { font-size: 11px; line-height: 1.6; margin: 0 0 6px; }
409
+ .cpub-ee-structure-empty strong { color: var(--accent); }
410
+
411
+ /* Assets */
412
+ .cpub-ee-assets-drop {
413
+ display: flex; flex-direction: column; align-items: center; gap: 6px;
414
+ padding: 20px 12px; margin: 8px; border: 2px dashed var(--border2); cursor: pointer;
415
+ transition: border-color 0.15s, background 0.15s; text-align: center;
416
+ }
417
+ .cpub-ee-assets-drop:hover { border-color: var(--accent); background: var(--accent-bg); }
418
+ .cpub-ee-assets-drop i { font-size: 20px; color: var(--text-faint); }
419
+ .cpub-ee-assets-drop-label { font-size: 11px; font-weight: 600; color: var(--text-dim); }
420
+ .cpub-ee-assets-drop-sub { font-size: 10px; color: var(--text-faint); font-family: var(--font-mono); }
421
+ .cpub-ee-assets-list { padding: 8px 12px; }
422
+ .cpub-ee-assets-heading { font-family: var(--font-mono); font-size: 10px; font-weight: 600; letter-spacing: 0.06em; text-transform: uppercase; color: var(--text-faint); padding: 4px 0 10px; }
423
+ .cpub-ee-asset-item {
424
+ display: flex; align-items: center; gap: 10px; padding: 8px 10px;
425
+ background: var(--surface); border: var(--border-width-default) solid var(--border); cursor: pointer;
426
+ box-shadow: var(--shadow-sm); margin-bottom: 5px;
427
+ }
428
+ .cpub-ee-asset-icon { width: 34px; height: 34px; background: var(--surface2); display: flex; align-items: center; justify-content: center; flex-shrink: 0; border: var(--border-width-default) solid var(--border2); }
429
+ .cpub-ee-asset-icon i { font-size: 11px; color: var(--text-faint); }
430
+ .cpub-ee-asset-info { flex: 1; min-width: 0; }
431
+ .cpub-ee-asset-name { font-size: 10px; color: var(--text); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; font-weight: 600; }
432
+ .cpub-ee-asset-size { font-family: var(--font-mono); font-size: 8px; color: var(--text-faint); }
433
+
434
+ .cpub-ee-center { flex: 1; display: flex; flex-direction: column; overflow: hidden; }
435
+ .cpub-ee-canvas { flex: 1; overflow-y: auto; background: var(--bg); }
436
+ .cpub-ee-canvas-inner { margin: 0 auto; transition: max-width 0.2s; }
437
+
438
+ /* Canvas toolbar */
439
+ .cpub-ee-canvas-toolbar {
440
+ display: flex; align-items: center; gap: 2px; padding: 4px 12px;
441
+ background: var(--surface); border-bottom: var(--border-width-default) solid var(--border); flex-shrink: 0; min-height: 32px;
442
+ justify-content: flex-end;
443
+ }
444
+ .cpub-ee-viewport-tabs { display: flex; gap: 0; }
445
+ .cpub-ee-viewport-tab {
446
+ width: 28px; height: 24px; display: flex; align-items: center; justify-content: center;
447
+ background: none; border: var(--border-width-default) solid var(--border); border-left-width: 0; color: var(--text-faint);
448
+ font-size: 10px; cursor: pointer;
449
+ }
450
+ .cpub-ee-viewport-tab:first-child { border-left-width: 2px; }
451
+ .cpub-ee-viewport-tab.active { background: var(--border); color: var(--color-text-inverse); }
452
+ .cpub-ee-viewport-tab:hover:not(.active) { background: var(--surface2); color: var(--text-dim); }
453
+
454
+ /* Status bar */
455
+ .cpub-ee-statusbar {
456
+ height: 26px; background: var(--surface); border-top: var(--border-width-default) solid var(--border);
457
+ display: flex; align-items: center; padding: 0 14px; gap: 18px; flex-shrink: 0;
458
+ }
459
+ .cpub-ee-status-item {
460
+ display: flex; align-items: center; gap: 5px; font-family: var(--font-mono);
461
+ font-size: 9px; color: var(--text-faint); white-space: nowrap;
462
+ }
463
+ .cpub-ee-status-item i { font-size: 8px; }
464
+ .cpub-ee-status-sep { width: 2px; height: 12px; background: var(--border); }
465
+ .cpub-ee-right { width: 280px; flex-shrink: 0; background: var(--surface); border-left: var(--border-width-default) solid var(--border); display: flex; flex-direction: column; overflow: hidden; }
466
+ .cpub-ee-right-body { flex: 1; overflow-y: auto; }
467
+
468
+ .cpub-sr-only { position: absolute; width: 1px; height: 1px; padding: 0; margin: -1px; overflow: hidden; clip: rect(0,0,0,0); border: 0; }
469
+
470
+ /* Mobile sidebar toggles */
471
+ .cpub-ee-mobile-toggles { display: none; }
472
+ .cpub-ee-mobile-overlay { display: none; }
473
+
474
+ @media (max-width: 1024px) {
475
+ .cpub-ee-left {
476
+ position: fixed; top: 0; bottom: 0; left: 0; z-index: 200;
477
+ transform: translateX(-100%); transition: transform 0.2s ease;
478
+ }
479
+ .cpub-ee-left.cpub-ee-sidebar-open { transform: translateX(0); }
480
+
481
+ .cpub-ee-right {
482
+ position: fixed; top: 0; bottom: 0; right: 0; z-index: 200;
483
+ transform: translateX(100%); transition: transform 0.2s ease;
484
+ }
485
+ .cpub-ee-right.cpub-ee-sidebar-open { transform: translateX(0); }
486
+
487
+ .cpub-ee-mobile-toggles {
488
+ display: flex; position: fixed; bottom: 16px; right: 16px;
489
+ gap: 8px; z-index: 100;
490
+ }
491
+ .cpub-ee-mobile-btn {
492
+ width: 44px; height: 44px; border: var(--border-width-default) solid var(--border); background: var(--surface);
493
+ color: var(--text-dim); font-size: 16px; cursor: pointer;
494
+ display: flex; align-items: center; justify-content: center;
495
+ box-shadow: var(--shadow-md);
496
+ }
497
+ .cpub-ee-mobile-btn:hover { background: var(--surface2); color: var(--text); }
498
+ .cpub-ee-mobile-overlay {
499
+ display: block; position: fixed; inset: 0;
500
+ background: var(--color-surface-overlay-light); z-index: 199;
501
+ }
502
+ }
503
+ </style>