@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,89 @@
1
+ <script setup lang="ts">
2
+ const props = defineProps<{ content: Record<string, unknown> }>();
3
+
4
+ const url = computed(() => (props.content.url as string) || '');
5
+
6
+ const embedUrl = computed(() => {
7
+ const u = url.value;
8
+ if (!u) return '';
9
+
10
+ // YouTube — extract video ID and construct safe embed URL
11
+ const ytMatch = u.match(/(?:youtube\.com\/watch\?v=|youtu\.be\/)([a-zA-Z0-9_-]+)/);
12
+ if (ytMatch) return `https://www.youtube-nocookie.com/embed/${ytMatch[1]}`;
13
+
14
+ // Vimeo — extract video ID and construct safe embed URL
15
+ const vimeoMatch = u.match(/vimeo\.com\/(\d+)/);
16
+ if (vimeoMatch) return `https://player.vimeo.com/video/${vimeoMatch[1]}`;
17
+
18
+ // Only allow http/https URLs for unknown platforms — block javascript:, data:, etc.
19
+ if (u.startsWith('https://') || u.startsWith('http://')) return u;
20
+ return '';
21
+ });
22
+
23
+ const platform = computed(() => {
24
+ const u = url.value;
25
+ if (u.includes('youtube') || u.includes('youtu.be')) return 'YouTube';
26
+ if (u.includes('vimeo')) return 'Vimeo';
27
+ return 'Video';
28
+ });
29
+ </script>
30
+
31
+ <template>
32
+ <div v-if="embedUrl" class="cpub-block-video">
33
+ <div class="cpub-video-label">
34
+ <i class="fa-solid fa-film"></i> {{ platform }}
35
+ </div>
36
+ <div class="cpub-video-wrap">
37
+ <iframe
38
+ :src="embedUrl"
39
+ class="cpub-video-iframe"
40
+ frameborder="0"
41
+ allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture"
42
+ allowfullscreen
43
+ loading="lazy"
44
+ :title="`${platform} video`"
45
+ />
46
+ </div>
47
+ </div>
48
+ </template>
49
+
50
+ <style scoped>
51
+ .cpub-block-video {
52
+ margin: 24px 0;
53
+ border: var(--border-width-default) solid var(--border);
54
+ overflow: hidden;
55
+ box-shadow: var(--shadow-md);
56
+ }
57
+
58
+ .cpub-video-label {
59
+ padding: 6px 12px;
60
+ font-family: var(--font-mono);
61
+ font-size: 10px;
62
+ font-weight: 600;
63
+ text-transform: uppercase;
64
+ letter-spacing: 0.06em;
65
+ color: var(--text-faint);
66
+ background: var(--surface2);
67
+ border-bottom: var(--border-width-default) solid var(--border);
68
+ display: flex;
69
+ align-items: center;
70
+ gap: 6px;
71
+ }
72
+
73
+ .cpub-video-label i { color: var(--accent); }
74
+
75
+ .cpub-video-wrap {
76
+ position: relative;
77
+ padding-bottom: 56.25%;
78
+ height: 0;
79
+ background: var(--text);
80
+ }
81
+
82
+ .cpub-video-iframe {
83
+ position: absolute;
84
+ top: 0;
85
+ left: 0;
86
+ width: 100%;
87
+ height: 100%;
88
+ }
89
+ </style>
@@ -0,0 +1,545 @@
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 blockTypes: BlockTypeGroup[] = [
19
+ {
20
+ name: 'Text',
21
+ blocks: [
22
+ { type: 'paragraph', label: 'Paragraph', icon: 'fa-align-left', description: 'Body text' },
23
+ { type: 'heading', label: 'Section Header', icon: 'fa-heading', description: 'H1-H3 heading' },
24
+ { type: 'blockquote', label: 'Quote', icon: 'fa-quote-left', description: 'Blockquote with attribution' },
25
+ ],
26
+ },
27
+ {
28
+ name: 'Media',
29
+ blocks: [
30
+ { type: 'image', label: 'Image', icon: 'fa-image', description: 'Upload or embed image' },
31
+ { type: 'video', label: 'Video Embed', icon: 'fa-film', description: 'YouTube, Vimeo, etc.' },
32
+ { type: 'horizontal_rule', label: 'Divider', icon: 'fa-minus', description: 'Visual separator' },
33
+ ],
34
+ },
35
+ {
36
+ name: 'Rich',
37
+ blocks: [
38
+ { type: 'code_block', label: 'Code Block', icon: 'fa-code', description: 'Syntax highlighted code' },
39
+ { type: 'callout', label: 'Callout', icon: 'fa-circle-info', description: 'Tip, warning, or note', attrs: { variant: 'info' } },
40
+ { type: 'embed', label: 'Embed', icon: 'fa-globe', description: 'External embed' },
41
+ { type: 'markdown', label: 'Markdown', icon: 'fa-brands fa-markdown', description: 'Raw markdown block' },
42
+ ],
43
+ },
44
+ ];
45
+
46
+ // --- Left panel tabs ---
47
+ const activeLeftTab = ref<'modules' | 'structure' | 'assets'>('modules');
48
+
49
+ // --- Structure: sections derived from H2 headings ---
50
+ interface Section {
51
+ id: string;
52
+ title: string;
53
+ blockCount: number;
54
+ index: number;
55
+ }
56
+
57
+ const sections = computed<Section[]>(() => {
58
+ const result: Section[] = [];
59
+ const blocks = props.blockEditor.blocks.value;
60
+ let currentSection: Section | null = null;
61
+ let sectionIdx = 0;
62
+
63
+ for (const block of blocks) {
64
+ if (block.type === 'heading' && (block.content.level === 2 || block.content.level === 1)) {
65
+ if (currentSection) result.push(currentSection);
66
+ sectionIdx++;
67
+ currentSection = {
68
+ id: block.id,
69
+ title: (block.content.text as string) || 'Untitled section',
70
+ blockCount: 1,
71
+ index: sectionIdx,
72
+ };
73
+ } else if (currentSection) {
74
+ currentSection.blockCount++;
75
+ }
76
+ }
77
+ if (currentSection) result.push(currentSection);
78
+ return result;
79
+ });
80
+
81
+ function scrollToSection(sectionId: string): void {
82
+ props.blockEditor.selectBlock(sectionId);
83
+ }
84
+
85
+ // --- Assets ---
86
+ const uploadedFiles = ref<Array<{ name: string; size: string; type: string }>>([]);
87
+
88
+ function onAssetUpload(event: Event): void {
89
+ const input = event.target as HTMLInputElement;
90
+ if (!input.files?.length) return;
91
+ const file = input.files[0];
92
+ if (!file) return;
93
+ const formData = new FormData();
94
+ formData.append('file', file);
95
+ formData.append('purpose', 'content');
96
+ $fetch<{ url: string; originalName: string; size: number }>('/api/files/upload', { method: 'POST', body: formData })
97
+ .then((res) => {
98
+ uploadedFiles.value.unshift({
99
+ name: res.originalName || file.name,
100
+ size: `${(res.size / 1024).toFixed(0)} KB`,
101
+ type: file.type.startsWith('image/') ? 'image' : 'file',
102
+ });
103
+ })
104
+ .catch(() => { /* silent */ });
105
+ }
106
+
107
+ // --- Cover image ---
108
+ const coverImageUrl = computed(() => (props.metadata.coverImageUrl as string) || '');
109
+
110
+ function onCoverUpload(event: Event): void {
111
+ const input = event.target as HTMLInputElement;
112
+ if (!input.files?.length) return;
113
+ const file = input.files[0];
114
+ if (!file) return;
115
+ const formData = new FormData();
116
+ formData.append('file', file);
117
+ formData.append('purpose', 'cover');
118
+ $fetch<{ url: string }>('/api/files/upload', { method: 'POST', body: formData })
119
+ .then((res) => { updateMeta('coverImageUrl', res.url); })
120
+ .catch(() => { /* silent fallback */ });
121
+ }
122
+
123
+ function onCoverUrl(): void {
124
+ const url = window.prompt('Enter image URL:');
125
+ if (url) updateMeta('coverImageUrl', url);
126
+ }
127
+
128
+ function removeCover(): void {
129
+ updateMeta('coverImageUrl', '');
130
+ }
131
+
132
+ // --- Right panel ---
133
+ const openSections = ref<Record<string, boolean>>({
134
+ content: true, seo: false, publishing: true, cover: false,
135
+ });
136
+ function toggleSection(key: string): void {
137
+ openSections.value[key] = !openSections.value[key];
138
+ }
139
+
140
+ const tags = computed(() => (props.metadata.tags as string[]) || []);
141
+ function onTagsUpdate(newTags: string[]): void { updateMeta('tags', newTags); }
142
+ const visibility = computed(() => (props.metadata.visibility as string) || 'public');
143
+ function onVisibilityUpdate(val: string): void { updateMeta('visibility', val); }
144
+
145
+ // --- Word count / status bar ---
146
+ const wordCount = computed(() => {
147
+ let count = 0;
148
+ for (const block of props.blockEditor.blocks.value) {
149
+ const html = (block.content.html as string) || '';
150
+ const text = (block.content.text as string) || '';
151
+ const code = (block.content.code as string) || '';
152
+ const combined = html.replace(/<[^>]*>/g, ' ') + ' ' + text + ' ' + code;
153
+ count += combined.split(/\s+/).filter((w) => w.length > 0).length;
154
+ }
155
+ return count;
156
+ });
157
+ const readTime = computed(() => Math.max(1, Math.round(wordCount.value / 200)));
158
+ const blockCount = computed(() => props.blockEditor.blocks.value.length);
159
+
160
+ // --- Mobile sidebar toggles ---
161
+ const mobileLeftOpen = ref(false);
162
+ const mobileRightOpen = ref(false);
163
+ function toggleMobileLeft(): void {
164
+ mobileLeftOpen.value = !mobileLeftOpen.value;
165
+ if (mobileLeftOpen.value) mobileRightOpen.value = false;
166
+ }
167
+ function toggleMobileRight(): void {
168
+ mobileRightOpen.value = !mobileRightOpen.value;
169
+ if (mobileRightOpen.value) mobileLeftOpen.value = false;
170
+ }
171
+ function closeMobileSidebars(): void {
172
+ mobileLeftOpen.value = false;
173
+ mobileRightOpen.value = false;
174
+ }
175
+
176
+ // --- Canvas toolbar ---
177
+ const viewportMode = ref<'desktop' | 'tablet' | 'mobile'>('desktop');
178
+ const canvasMaxWidth = computed(() => {
179
+ if (viewportMode.value === 'mobile') return '375px';
180
+ if (viewportMode.value === 'tablet') return '768px';
181
+ return '680px';
182
+ });
183
+ </script>
184
+
185
+ <template>
186
+ <div class="cpub-ae-shell">
187
+ <!-- Mobile sidebar toggles -->
188
+ <div class="cpub-ae-mobile-toggles">
189
+ <button class="cpub-ae-mobile-btn" aria-label="Toggle modules panel" @click="toggleMobileLeft"><i class="fa-solid fa-layer-group"></i></button>
190
+ <button class="cpub-ae-mobile-btn" aria-label="Toggle properties panel" @click="toggleMobileRight"><i class="fa-solid fa-sliders"></i></button>
191
+ </div>
192
+ <div v-if="mobileLeftOpen || mobileRightOpen" class="cpub-ae-mobile-overlay" @click="closeMobileSidebars" />
193
+
194
+ <!-- LEFT: Tabbed panel (Modules / Structure / Assets) -->
195
+ <aside class="cpub-ae-left" :class="{ 'cpub-ae-sidebar-open': mobileLeftOpen }" aria-label="Editor sidebar">
196
+ <div class="cpub-ae-left-tabs">
197
+ <button class="cpub-ae-left-tab" :class="{ active: activeLeftTab === 'modules' }" @click="activeLeftTab = 'modules'">Modules</button>
198
+ <button class="cpub-ae-left-tab" :class="{ active: activeLeftTab === 'structure' }" @click="activeLeftTab = 'structure'">Structure</button>
199
+ <button class="cpub-ae-left-tab" :class="{ active: activeLeftTab === 'assets' }" @click="activeLeftTab = 'assets'">Assets</button>
200
+ </div>
201
+
202
+ <!-- Modules tab -->
203
+ <div v-if="activeLeftTab === 'modules'" class="cpub-ae-left-body">
204
+ <EditorsEditorBlocks :groups="blockTypes" :block-editor="blockEditor" />
205
+ </div>
206
+
207
+ <!-- Structure tab -->
208
+ <div v-else-if="activeLeftTab === 'structure'" class="cpub-ae-left-body">
209
+ <div class="cpub-ae-structure-list">
210
+ <div
211
+ v-for="section in sections"
212
+ :key="section.id"
213
+ class="cpub-ae-structure-item"
214
+ @click="scrollToSection(section.id)"
215
+ >
216
+ <span class="cpub-ae-structure-num">{{ String(section.index).padStart(2, '0') }}</span>
217
+ <span class="cpub-ae-structure-title">{{ section.title }}</span>
218
+ <span class="cpub-ae-structure-badge">{{ section.blockCount }} blk</span>
219
+ </div>
220
+ <p v-if="sections.length === 0" class="cpub-ae-structure-hint">
221
+ Add H2 headings to create sections. Each heading starts a new section.
222
+ </p>
223
+ </div>
224
+ </div>
225
+
226
+ <!-- Assets tab -->
227
+ <div v-else class="cpub-ae-left-body">
228
+ <label class="cpub-ae-assets-drop">
229
+ <i class="fa-solid fa-cloud-arrow-up"></i>
230
+ <div class="cpub-ae-assets-drop-label">Drop files here</div>
231
+ <div class="cpub-ae-assets-drop-sub">JPG, PNG, GIF, SVG, PDF</div>
232
+ <input type="file" class="cpub-sr-only" @change="onAssetUpload">
233
+ </label>
234
+ <div v-if="uploadedFiles.length > 0" class="cpub-ae-assets-list">
235
+ <div class="cpub-ae-assets-heading">Recent Uploads</div>
236
+ <div
237
+ v-for="(file, idx) in uploadedFiles"
238
+ :key="idx"
239
+ class="cpub-ae-asset-item"
240
+ >
241
+ <div class="cpub-ae-asset-icon">
242
+ <i :class="file.type === 'image' ? 'fa-solid fa-image' : 'fa-solid fa-file'" />
243
+ </div>
244
+ <div class="cpub-ae-asset-info">
245
+ <div class="cpub-ae-asset-name">{{ file.name }}</div>
246
+ <div class="cpub-ae-asset-size">{{ file.size }}</div>
247
+ </div>
248
+ </div>
249
+ </div>
250
+ </div>
251
+ </aside>
252
+
253
+ <!-- CENTER: Canvas with toolbar -->
254
+ <div class="cpub-ae-center">
255
+ <!-- Canvas toolbar -->
256
+ <div class="cpub-ae-canvas-toolbar">
257
+ <button class="cpub-ae-tool-btn" title="Previous block"><i class="fa-solid fa-chevron-up"></i></button>
258
+ <button class="cpub-ae-tool-btn" title="Next block"><i class="fa-solid fa-chevron-down"></i></button>
259
+ <div class="cpub-ae-toolbar-divider" />
260
+ <div class="cpub-ae-viewport-tabs">
261
+ <button class="cpub-ae-viewport-tab" :class="{ active: viewportMode === 'desktop' }" title="Desktop" @click="viewportMode = 'desktop'"><i class="fa-solid fa-desktop"></i></button>
262
+ <button class="cpub-ae-viewport-tab" :class="{ active: viewportMode === 'tablet' }" title="Tablet" @click="viewportMode = 'tablet'"><i class="fa-solid fa-tablet-screen-button"></i></button>
263
+ <button class="cpub-ae-viewport-tab" :class="{ active: viewportMode === 'mobile' }" title="Mobile" @click="viewportMode = 'mobile'"><i class="fa-solid fa-mobile-screen"></i></button>
264
+ </div>
265
+ </div>
266
+
267
+ <!-- Scrollable canvas -->
268
+ <div class="cpub-ae-canvas">
269
+ <div class="cpub-ae-canvas-inner" :style="{ maxWidth: canvasMaxWidth }">
270
+ <EditorsBlockCanvas :block-editor="blockEditor" :block-types="blockTypes" />
271
+ </div>
272
+ </div>
273
+
274
+ <!-- Status bar -->
275
+ <div class="cpub-ae-statusbar">
276
+ <div class="cpub-ae-status-item">
277
+ <i class="fa-solid fa-layer-group"></i>
278
+ <span>{{ blockCount }} blocks &middot; {{ sections.length }} sections</span>
279
+ </div>
280
+ <div class="cpub-ae-status-sep" />
281
+ <div class="cpub-ae-status-item">
282
+ <i class="fa-solid fa-align-justify"></i>
283
+ <span>{{ wordCount.toLocaleString() }} words</span>
284
+ </div>
285
+ <div class="cpub-ae-status-sep" />
286
+ <div class="cpub-ae-status-item">
287
+ <i class="fa-regular fa-clock"></i>
288
+ <span>~{{ readTime }} min read</span>
289
+ </div>
290
+ </div>
291
+ </div>
292
+
293
+ <!-- RIGHT: Properties -->
294
+ <aside class="cpub-ae-right" :class="{ 'cpub-ae-sidebar-open': mobileRightOpen }" aria-label="Document properties">
295
+ <div class="cpub-ae-right-header">
296
+ <span class="cpub-ae-right-title">Properties</span>
297
+ </div>
298
+ <div class="cpub-ae-right-body">
299
+ <!-- Content / Metadata -->
300
+ <EditorsEditorSection title="Content" icon="fa-pen-nib" :open="openSections.content" @toggle="toggleSection('content')">
301
+ <div class="cpub-ep-field">
302
+ <label class="cpub-ep-flabel">Slug</label>
303
+ <input class="cpub-ep-input" type="text" :value="metadata.slug" placeholder="auto-generated" @input="updateMeta('slug', ($event.target as HTMLInputElement).value)">
304
+ </div>
305
+ <div class="cpub-ep-field">
306
+ <label class="cpub-ep-flabel">Description</label>
307
+ <textarea class="cpub-ep-textarea" rows="3" :value="metadata.description as string" placeholder="Brief description..." @input="updateMeta('description', ($event.target as HTMLTextAreaElement).value)" />
308
+ </div>
309
+ <div class="cpub-ae-cover" :class="{ 'has-image': !!coverImageUrl }">
310
+ <template v-if="coverImageUrl">
311
+ <img :src="coverImageUrl" alt="Cover image" class="cpub-ae-cover-img" />
312
+ <div class="cpub-ae-cover-actions">
313
+ <button class="cpub-ae-cover-btn" @click="removeCover"><i class="fa-solid fa-trash"></i> Remove</button>
314
+ <label class="cpub-ae-cover-btn">
315
+ <i class="fa-solid fa-arrow-up-from-bracket"></i> Replace
316
+ <input type="file" accept="image/*" class="cpub-sr-only" @change="onCoverUpload">
317
+ </label>
318
+ </div>
319
+ </template>
320
+ <template v-else>
321
+ <div class="cpub-ae-cover-placeholder">
322
+ <div class="cpub-ae-cover-icon"><i class="fa-regular fa-image"></i></div>
323
+ <span class="cpub-ae-cover-text">Cover image</span>
324
+ </div>
325
+ <div class="cpub-ae-cover-overlay">
326
+ <label class="cpub-ae-cover-btn primary">
327
+ <i class="fa-solid fa-arrow-up-from-bracket"></i> Upload
328
+ <input type="file" accept="image/*" class="cpub-sr-only" @change="onCoverUpload">
329
+ </label>
330
+ <button class="cpub-ae-cover-btn" @click="onCoverUrl"><i class="fa-solid fa-link"></i> From URL</button>
331
+ </div>
332
+ </template>
333
+ </div>
334
+ </EditorsEditorSection>
335
+
336
+ <!-- SEO -->
337
+ <EditorsEditorSection title="SEO" icon="fa-magnifying-glass" :open="openSections.seo" @toggle="toggleSection('seo')">
338
+ <div class="cpub-ep-field">
339
+ <label class="cpub-ep-flabel">Meta Description</label>
340
+ <textarea class="cpub-ep-textarea" rows="3" :value="metadata.seoDescription as string" placeholder="Search engine description..." @input="updateMeta('seoDescription', ($event.target as HTMLTextAreaElement).value)" />
341
+ <span class="cpub-ep-hint">{{ ((metadata.seoDescription as string) || '').length }}/160</span>
342
+ </div>
343
+ </EditorsEditorSection>
344
+
345
+ <!-- Publishing -->
346
+ <EditorsEditorSection title="Publishing" icon="fa-rocket" :open="openSections.publishing" @toggle="toggleSection('publishing')">
347
+ <div class="cpub-ep-field">
348
+ <label class="cpub-ep-flabel">Visibility</label>
349
+ <EditorsEditorVisibility :model-value="visibility" @update:model-value="onVisibilityUpdate" />
350
+ </div>
351
+ <div class="cpub-ep-field">
352
+ <label class="cpub-ep-flabel">Category</label>
353
+ <select class="cpub-ep-select" :value="metadata.category || ''" @change="updateMeta('category', ($event.target as HTMLSelectElement).value)">
354
+ <option value="">Select category</option>
355
+ <option value="technology">Technology</option>
356
+ <option value="hardware">Hardware</option>
357
+ <option value="ai-ml">AI &amp; Machine Learning</option>
358
+ <option value="tutorial">Tutorial</option>
359
+ <option value="deep-dive">Deep Dive</option>
360
+ <option value="opinion">Opinion</option>
361
+ </select>
362
+ </div>
363
+ <div class="cpub-ep-field">
364
+ <label class="cpub-ep-flabel">Tags</label>
365
+ <EditorsEditorTagInput :tags="tags" @update:tags="onTagsUpdate" />
366
+ </div>
367
+ </EditorsEditorSection>
368
+ </div>
369
+ </aside>
370
+ </div>
371
+ </template>
372
+
373
+ <style scoped>
374
+ .cpub-ae-shell { display: flex; flex: 1; overflow: hidden; }
375
+
376
+ /* Left panel */
377
+ .cpub-ae-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; }
378
+ .cpub-ae-left-tabs { display: flex; border-bottom: var(--border-width-default) solid var(--border); flex-shrink: 0; }
379
+ .cpub-ae-left-tab {
380
+ flex: 1; padding: 8px 4px; font-family: var(--font-mono); font-size: 10px; font-weight: 600;
381
+ letter-spacing: 0.06em; text-transform: uppercase; text-align: center;
382
+ background: none; border: none; color: var(--text-dim); cursor: pointer;
383
+ border-bottom: var(--border-width-default) solid transparent; margin-bottom: -2px;
384
+ }
385
+ .cpub-ae-left-tab.active { color: var(--accent); border-bottom-color: var(--accent); background: var(--accent-bg); }
386
+ .cpub-ae-left-body { flex: 1; overflow-y: auto; }
387
+
388
+ /* Structure */
389
+ .cpub-ae-structure-list { padding: 8px; }
390
+ .cpub-ae-structure-item {
391
+ display: flex; align-items: center; gap: 8px; padding: 8px 10px; cursor: pointer;
392
+ border: var(--border-width-default) solid transparent; transition: all 0.1s;
393
+ }
394
+ .cpub-ae-structure-item:hover { background: var(--surface2); border-color: var(--border2); }
395
+ .cpub-ae-structure-num { font-family: var(--font-mono); font-size: 10px; color: var(--text-faint); font-weight: 600; }
396
+ .cpub-ae-structure-title { flex: 1; font-size: 12px; color: var(--text); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
397
+ .cpub-ae-structure-badge { font-family: var(--font-mono); font-size: 9px; color: var(--text-faint); background: var(--surface2); border: var(--border-width-default) solid var(--border2); padding: 1px 6px; }
398
+ .cpub-ae-structure-hint { font-size: 11px; color: var(--text-dim); line-height: 1.5; padding: 12px; }
399
+
400
+ /* Assets */
401
+ .cpub-ae-assets-drop {
402
+ display: flex; flex-direction: column; align-items: center; gap: 6px;
403
+ padding: 20px 12px; margin: 8px; border: 2px dashed var(--border2); cursor: pointer;
404
+ transition: border-color 0.15s, background 0.15s; text-align: center;
405
+ }
406
+ .cpub-ae-assets-drop:hover { border-color: var(--accent); background: var(--accent-bg); }
407
+ .cpub-ae-assets-drop i { font-size: 20px; color: var(--text-faint); }
408
+ .cpub-ae-assets-drop-label { font-size: 11px; font-weight: 600; color: var(--text-dim); }
409
+ .cpub-ae-assets-drop-sub { font-size: 10px; color: var(--text-faint); font-family: var(--font-mono); }
410
+ .cpub-ae-assets-list { padding: 8px 12px; }
411
+ .cpub-ae-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; }
412
+ .cpub-ae-asset-item {
413
+ display: flex; align-items: center; gap: 10px; padding: 8px 10px;
414
+ background: var(--surface); border: var(--border-width-default) solid var(--border); cursor: pointer;
415
+ box-shadow: var(--shadow-sm); margin-bottom: 5px;
416
+ }
417
+ .cpub-ae-asset-icon {
418
+ width: 34px; height: 34px; background: var(--surface2); display: flex;
419
+ align-items: center; justify-content: center; flex-shrink: 0; border: var(--border-width-default) solid var(--border2);
420
+ }
421
+ .cpub-ae-asset-icon i { font-size: 11px; color: var(--text-faint); }
422
+ .cpub-ae-asset-info { flex: 1; min-width: 0; }
423
+ .cpub-ae-asset-name { font-size: 10px; color: var(--text); overflow: hidden; text-overflow: ellipsis; white-space: nowrap; font-weight: 600; }
424
+ .cpub-ae-asset-size { font-family: var(--font-mono); font-size: 8px; color: var(--text-faint); }
425
+
426
+ /* Center */
427
+ .cpub-ae-center { flex: 1; display: flex; flex-direction: column; overflow: hidden; }
428
+
429
+ /* Canvas toolbar */
430
+ .cpub-ae-canvas-toolbar {
431
+ display: flex; align-items: center; gap: 2px; padding: 4px 12px;
432
+ background: var(--surface); border-bottom: var(--border-width-default) solid var(--border); flex-shrink: 0; min-height: 32px;
433
+ }
434
+ .cpub-ae-tool-btn {
435
+ width: 24px; height: 24px; display: flex; align-items: center; justify-content: center;
436
+ background: var(--surface2); border: var(--border-width-default) solid var(--border); color: var(--text-dim);
437
+ font-size: 9px; cursor: pointer;
438
+ }
439
+ .cpub-ae-tool-btn:hover { background: var(--surface3); color: var(--text); }
440
+ .cpub-ae-toolbar-divider { width: 2px; height: 16px; background: var(--border); margin: 0 6px; }
441
+ .cpub-ae-viewport-tabs { display: flex; gap: 0; margin-left: auto; }
442
+ .cpub-ae-viewport-tab {
443
+ width: 28px; height: 24px; display: flex; align-items: center; justify-content: center;
444
+ background: none; border: var(--border-width-default) solid var(--border); border-left-width: 0; color: var(--text-faint);
445
+ font-size: 10px; cursor: pointer;
446
+ }
447
+ .cpub-ae-viewport-tab:first-child { border-left-width: 2px; }
448
+ .cpub-ae-viewport-tab.active { background: var(--border); color: var(--color-text-inverse); }
449
+ .cpub-ae-viewport-tab:hover:not(.active) { background: var(--surface2); color: var(--text-dim); }
450
+
451
+ /* Canvas */
452
+ .cpub-ae-canvas { flex: 1; overflow-y: auto; background: var(--bg); }
453
+ .cpub-ae-canvas-inner { margin: 0 auto; transition: max-width 0.2s; }
454
+
455
+ /* Status bar */
456
+ .cpub-ae-statusbar {
457
+ height: 26px; background: var(--surface); border-top: var(--border-width-default) solid var(--border);
458
+ display: flex; align-items: center; padding: 0 14px; gap: 18px; flex-shrink: 0;
459
+ }
460
+ .cpub-ae-status-item {
461
+ display: flex; align-items: center; gap: 5px; font-family: var(--font-mono);
462
+ font-size: 9px; color: var(--text-faint); white-space: nowrap;
463
+ }
464
+ .cpub-ae-status-item i { font-size: 8px; }
465
+ .cpub-ae-status-sep { width: 2px; height: 12px; background: var(--border); }
466
+
467
+ /* Right panel */
468
+ .cpub-ae-right { width: 290px; flex-shrink: 0; background: var(--surface); border-left: var(--border-width-default) solid var(--border); display: flex; flex-direction: column; overflow: hidden; }
469
+ .cpub-ae-right-header {
470
+ padding: 12px 16px; border-bottom: var(--border-width-default) solid var(--border); display: flex;
471
+ align-items: center; gap: 10px; flex-shrink: 0; min-height: 44px;
472
+ }
473
+ .cpub-ae-right-title { font-size: 11px; font-weight: 700; color: var(--text); font-family: var(--font-mono); letter-spacing: 0.04em; text-transform: uppercase; }
474
+ .cpub-ae-right-body { flex: 1; overflow-y: auto; }
475
+
476
+ .cpub-sr-only { position: absolute; width: 1px; height: 1px; padding: 0; margin: -1px; overflow: hidden; clip: rect(0,0,0,0); border: 0; }
477
+
478
+ /* Cover image */
479
+ .cpub-ae-cover {
480
+ position: relative; width: 100%; aspect-ratio: 16/9; background: var(--surface2);
481
+ border: var(--border-width-default) solid var(--border); display: flex; align-items: center; justify-content: center; overflow: hidden;
482
+ }
483
+ .cpub-ae-cover-img { width: 100%; height: 100%; object-fit: cover; }
484
+ .cpub-ae-cover-placeholder { display: flex; flex-direction: column; align-items: center; gap: 6px; }
485
+ .cpub-ae-cover-icon { font-size: 24px; color: var(--text-faint); }
486
+ .cpub-ae-cover-text { font-size: 10px; color: var(--text-faint); font-family: var(--font-mono); }
487
+ .cpub-ae-cover-overlay, .cpub-ae-cover-actions {
488
+ position: absolute; inset: 0; display: flex; align-items: center; justify-content: center; gap: 6px;
489
+ background: var(--color-surface-scrim); opacity: 0; transition: opacity 0.15s;
490
+ }
491
+ .cpub-ae-cover:hover .cpub-ae-cover-overlay,
492
+ .cpub-ae-cover:hover .cpub-ae-cover-actions,
493
+ .cpub-ae-cover:focus-within .cpub-ae-cover-overlay,
494
+ .cpub-ae-cover:focus-within .cpub-ae-cover-actions { opacity: 1; }
495
+ .cpub-ae-cover-btn {
496
+ font-size: 10px; padding: 5px 10px; background: var(--surface); border: var(--border-width-default) solid var(--border);
497
+ color: var(--text-dim); cursor: pointer; display: inline-flex; align-items: center; gap: 4px;
498
+ font-family: var(--font-mono); box-shadow: var(--shadow-sm);
499
+ }
500
+ .cpub-ae-cover-btn.primary { background: var(--accent); color: var(--color-text-inverse); border-color: var(--accent); }
501
+ .cpub-ae-cover-btn:hover { background: var(--surface2); }
502
+ .cpub-ae-cover-btn.primary:hover { opacity: 0.9; background: var(--accent); }
503
+
504
+ /* Mobile sidebar toggles */
505
+ .cpub-ae-mobile-toggles { display: none; }
506
+ .cpub-ae-mobile-overlay { display: none; }
507
+
508
+ @media (max-width: 1200px) {
509
+ .cpub-ae-left {
510
+ position: fixed; top: 0; bottom: 0; left: 0; z-index: 200;
511
+ transform: translateX(-100%); transition: transform 0.2s ease;
512
+ }
513
+ .cpub-ae-left.cpub-ae-sidebar-open { transform: translateX(0); }
514
+ }
515
+
516
+ @media (max-width: 1024px) {
517
+ .cpub-ae-right {
518
+ position: fixed; top: 0; bottom: 0; right: 0; z-index: 200;
519
+ transform: translateX(100%); transition: transform 0.2s ease;
520
+ }
521
+ .cpub-ae-right.cpub-ae-sidebar-open { transform: translateX(0); }
522
+
523
+ .cpub-ae-mobile-toggles {
524
+ display: flex; position: fixed; bottom: 16px; right: 16px;
525
+ gap: 8px; z-index: 100;
526
+ }
527
+ .cpub-ae-mobile-btn {
528
+ width: 44px; height: 44px; border: var(--border-width-default) solid var(--border); background: var(--surface);
529
+ color: var(--text-dim); font-size: 16px; cursor: pointer;
530
+ display: flex; align-items: center; justify-content: center;
531
+ box-shadow: var(--shadow-md);
532
+ }
533
+ .cpub-ae-mobile-btn:hover { background: var(--surface2); color: var(--text); }
534
+ .cpub-ae-mobile-overlay {
535
+ display: block; position: fixed; inset: 0;
536
+ background: var(--color-surface-overlay-light); z-index: 199;
537
+ }
538
+ }
539
+
540
+ /* Touch devices: always show cover overlays */
541
+ @media (hover: none) {
542
+ .cpub-ae-cover-overlay,
543
+ .cpub-ae-cover-actions { opacity: 1; }
544
+ }
545
+ </style>