emdash 0.0.0-b → 0.0.1

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 (661) hide show
  1. package/README.md +87 -43
  2. package/dist/adapters-BLMa4JGD.d.mts +106 -0
  3. package/dist/adapters-BLMa4JGD.d.mts.map +1 -0
  4. package/dist/apply-Bjfq_b4-.mjs +1293 -0
  5. package/dist/apply-Bjfq_b4-.mjs.map +1 -0
  6. package/dist/astro/index.d.mts +51 -0
  7. package/dist/astro/index.d.mts.map +1 -0
  8. package/dist/astro/index.mjs +1333 -0
  9. package/dist/astro/index.mjs.map +1 -0
  10. package/dist/astro/middleware/auth.d.mts +31 -0
  11. package/dist/astro/middleware/auth.d.mts.map +1 -0
  12. package/dist/astro/middleware/auth.mjs +654 -0
  13. package/dist/astro/middleware/auth.mjs.map +1 -0
  14. package/dist/astro/middleware/redirect.d.mts +22 -0
  15. package/dist/astro/middleware/redirect.d.mts.map +1 -0
  16. package/dist/astro/middleware/redirect.mjs +63 -0
  17. package/dist/astro/middleware/redirect.mjs.map +1 -0
  18. package/dist/astro/middleware/request-context.d.mts +18 -0
  19. package/dist/astro/middleware/request-context.d.mts.map +1 -0
  20. package/dist/astro/middleware/request-context.mjs +1310 -0
  21. package/dist/astro/middleware/request-context.mjs.map +1 -0
  22. package/dist/astro/middleware/setup.d.mts +20 -0
  23. package/dist/astro/middleware/setup.d.mts.map +1 -0
  24. package/dist/astro/middleware/setup.mjs +47 -0
  25. package/dist/astro/middleware/setup.mjs.map +1 -0
  26. package/dist/astro/middleware.d.mts +13 -0
  27. package/dist/astro/middleware.d.mts.map +1 -0
  28. package/dist/astro/middleware.mjs +1613 -0
  29. package/dist/astro/middleware.mjs.map +1 -0
  30. package/dist/astro/types.d.mts +250 -0
  31. package/dist/astro/types.d.mts.map +1 -0
  32. package/dist/astro/types.mjs +1 -0
  33. package/dist/base64-MBPo9ozB.mjs +59 -0
  34. package/dist/base64-MBPo9ozB.mjs.map +1 -0
  35. package/dist/byline-CL847F26.mjs +213 -0
  36. package/dist/byline-CL847F26.mjs.map +1 -0
  37. package/dist/bylines-C2a-2TGt.mjs +136 -0
  38. package/dist/bylines-C2a-2TGt.mjs.map +1 -0
  39. package/dist/chunk-ClPoSABd.mjs +21 -0
  40. package/dist/cli/index.d.mts +1 -0
  41. package/dist/cli/index.mjs +3909 -0
  42. package/dist/cli/index.mjs.map +1 -0
  43. package/dist/client/cf-access.d.mts +60 -0
  44. package/dist/client/cf-access.d.mts.map +1 -0
  45. package/dist/client/cf-access.mjs +179 -0
  46. package/dist/client/cf-access.mjs.map +1 -0
  47. package/dist/client/index.d.mts +398 -0
  48. package/dist/client/index.d.mts.map +1 -0
  49. package/dist/client/index.mjs +346 -0
  50. package/dist/client/index.mjs.map +1 -0
  51. package/dist/config-CKE8p9xM.mjs +55 -0
  52. package/dist/config-CKE8p9xM.mjs.map +1 -0
  53. package/dist/connection-B4zVnQIa.mjs +40 -0
  54. package/dist/connection-B4zVnQIa.mjs.map +1 -0
  55. package/dist/content-D6C2WsZC.mjs +824 -0
  56. package/dist/content-D6C2WsZC.mjs.map +1 -0
  57. package/dist/db/index.d.mts +4 -0
  58. package/dist/db/index.mjs +62 -0
  59. package/dist/db/index.mjs.map +1 -0
  60. package/dist/db/libsql.d.mts +11 -0
  61. package/dist/db/libsql.d.mts.map +1 -0
  62. package/dist/db/libsql.mjs +17 -0
  63. package/dist/db/libsql.mjs.map +1 -0
  64. package/dist/db/postgres.d.mts +11 -0
  65. package/dist/db/postgres.d.mts.map +1 -0
  66. package/dist/db/postgres.mjs +30 -0
  67. package/dist/db/postgres.mjs.map +1 -0
  68. package/dist/db/sqlite.d.mts +11 -0
  69. package/dist/db/sqlite.d.mts.map +1 -0
  70. package/dist/db/sqlite.mjs +16 -0
  71. package/dist/db/sqlite.mjs.map +1 -0
  72. package/dist/default-Cyi4aAxu.mjs +81 -0
  73. package/dist/default-Cyi4aAxu.mjs.map +1 -0
  74. package/dist/dialect-helpers-B9uSp2GJ.mjs +90 -0
  75. package/dist/dialect-helpers-B9uSp2GJ.mjs.map +1 -0
  76. package/dist/error-Cxz0tQeO.mjs +27 -0
  77. package/dist/error-Cxz0tQeO.mjs.map +1 -0
  78. package/dist/index-C1xF3OGh.d.mts +4527 -0
  79. package/dist/index-C1xF3OGh.d.mts.map +1 -0
  80. package/dist/index.d.mts +16 -0
  81. package/dist/index.mjs +30 -0
  82. package/dist/load-yOOlckBj.mjs +28 -0
  83. package/dist/load-yOOlckBj.mjs.map +1 -0
  84. package/dist/loader-fz8Q_3EO.mjs +447 -0
  85. package/dist/loader-fz8Q_3EO.mjs.map +1 -0
  86. package/dist/manifest-schema-Dcl0R6nM.mjs +184 -0
  87. package/dist/manifest-schema-Dcl0R6nM.mjs.map +1 -0
  88. package/dist/media/index.d.mts +26 -0
  89. package/dist/media/index.d.mts.map +1 -0
  90. package/dist/media/index.mjs +55 -0
  91. package/dist/media/index.mjs.map +1 -0
  92. package/dist/media/local-runtime.d.mts +39 -0
  93. package/dist/media/local-runtime.d.mts.map +1 -0
  94. package/dist/media/local-runtime.mjs +133 -0
  95. package/dist/media/local-runtime.mjs.map +1 -0
  96. package/dist/media-DqHVh136.mjs +200 -0
  97. package/dist/media-DqHVh136.mjs.map +1 -0
  98. package/dist/mode-C2EzN1uE.mjs +23 -0
  99. package/dist/mode-C2EzN1uE.mjs.map +1 -0
  100. package/dist/page/index.d.mts +140 -0
  101. package/dist/page/index.d.mts.map +1 -0
  102. package/dist/page/index.mjs +416 -0
  103. package/dist/page/index.mjs.map +1 -0
  104. package/dist/placeholder-CmGAmqeO.d.mts +276 -0
  105. package/dist/placeholder-CmGAmqeO.d.mts.map +1 -0
  106. package/dist/placeholder-SmpOx-_v.mjs +243 -0
  107. package/dist/placeholder-SmpOx-_v.mjs.map +1 -0
  108. package/dist/plugin-utils.d.mts +58 -0
  109. package/dist/plugin-utils.d.mts.map +1 -0
  110. package/dist/plugin-utils.mjs +78 -0
  111. package/dist/plugin-utils.mjs.map +1 -0
  112. package/dist/plugins/adapt-sandbox-entry.d.mts +22 -0
  113. package/dist/plugins/adapt-sandbox-entry.d.mts.map +1 -0
  114. package/dist/plugins/adapt-sandbox-entry.mjs +113 -0
  115. package/dist/plugins/adapt-sandbox-entry.mjs.map +1 -0
  116. package/dist/query-CS_iSj34.mjs +460 -0
  117. package/dist/query-CS_iSj34.mjs.map +1 -0
  118. package/dist/redirect-DIfIni3r.mjs +329 -0
  119. package/dist/redirect-DIfIni3r.mjs.map +1 -0
  120. package/dist/registry-D_w5HW4G.mjs +863 -0
  121. package/dist/registry-D_w5HW4G.mjs.map +1 -0
  122. package/dist/request-context.d.mts +49 -0
  123. package/dist/request-context.d.mts.map +1 -0
  124. package/dist/request-context.mjs +43 -0
  125. package/dist/request-context.mjs.map +1 -0
  126. package/dist/runner-B-u2F2b6.mjs +1412 -0
  127. package/dist/runner-B-u2F2b6.mjs.map +1 -0
  128. package/dist/runner-EAtf0ZIe.d.mts +27 -0
  129. package/dist/runner-EAtf0ZIe.d.mts.map +1 -0
  130. package/dist/runtime.d.mts +26 -0
  131. package/dist/runtime.d.mts.map +1 -0
  132. package/dist/runtime.mjs +42 -0
  133. package/dist/runtime.mjs.map +1 -0
  134. package/dist/search-DG603UrT.mjs +9211 -0
  135. package/dist/search-DG603UrT.mjs.map +1 -0
  136. package/dist/seed/index.d.mts +3 -0
  137. package/dist/seed/index.mjs +15 -0
  138. package/dist/seo/index.d.mts +70 -0
  139. package/dist/seo/index.d.mts.map +1 -0
  140. package/dist/seo/index.mjs +70 -0
  141. package/dist/seo/index.mjs.map +1 -0
  142. package/dist/storage/local.d.mts +39 -0
  143. package/dist/storage/local.d.mts.map +1 -0
  144. package/dist/storage/local.mjs +166 -0
  145. package/dist/storage/local.mjs.map +1 -0
  146. package/dist/storage/s3.d.mts +32 -0
  147. package/dist/storage/s3.d.mts.map +1 -0
  148. package/dist/storage/s3.mjs +175 -0
  149. package/dist/storage/s3.mjs.map +1 -0
  150. package/dist/tokens-DpgrkrXK.mjs +171 -0
  151. package/dist/tokens-DpgrkrXK.mjs.map +1 -0
  152. package/dist/transport-BFGblqwG.d.mts +42 -0
  153. package/dist/transport-BFGblqwG.d.mts.map +1 -0
  154. package/dist/transport-yxiQsi8I.mjs +418 -0
  155. package/dist/transport-yxiQsi8I.mjs.map +1 -0
  156. package/dist/types-BRuPJGdV.d.mts +102 -0
  157. package/dist/types-BRuPJGdV.d.mts.map +1 -0
  158. package/dist/types-C4-fAxN3.d.mts +182 -0
  159. package/dist/types-C4-fAxN3.d.mts.map +1 -0
  160. package/dist/types-CMMN0pNg.mjs +31 -0
  161. package/dist/types-CMMN0pNg.mjs.map +1 -0
  162. package/dist/types-CUBbjgmP.mjs +16 -0
  163. package/dist/types-CUBbjgmP.mjs.map +1 -0
  164. package/dist/types-DRjfYOEv.d.mts +426 -0
  165. package/dist/types-DRjfYOEv.d.mts.map +1 -0
  166. package/dist/types-DY5zk5HN.mjs +73 -0
  167. package/dist/types-DY5zk5HN.mjs.map +1 -0
  168. package/dist/types-DaNLHo_T.d.mts +184 -0
  169. package/dist/types-DaNLHo_T.d.mts.map +1 -0
  170. package/dist/types-DvhsUmSJ.d.mts +1111 -0
  171. package/dist/types-DvhsUmSJ.d.mts.map +1 -0
  172. package/dist/validate-CpBtVMsD.d.mts +378 -0
  173. package/dist/validate-CpBtVMsD.d.mts.map +1 -0
  174. package/dist/validate-CqRJb_xU.mjs +97 -0
  175. package/dist/validate-CqRJb_xU.mjs.map +1 -0
  176. package/dist/validate-O7PWmlnq.mjs +328 -0
  177. package/dist/validate-O7PWmlnq.mjs.map +1 -0
  178. package/locals.d.ts +46 -0
  179. package/package.json +233 -19
  180. package/src/api/authorize.ts +63 -0
  181. package/src/api/csrf.ts +48 -0
  182. package/src/api/error.ts +99 -0
  183. package/src/api/errors.ts +445 -0
  184. package/src/api/escape.ts +9 -0
  185. package/src/api/handlers/api-tokens.ts +240 -0
  186. package/src/api/handlers/comments.ts +314 -0
  187. package/src/api/handlers/content.ts +1315 -0
  188. package/src/api/handlers/dashboard.ts +205 -0
  189. package/src/api/handlers/device-flow.ts +687 -0
  190. package/src/api/handlers/index.ts +163 -0
  191. package/src/api/handlers/manifest.ts +158 -0
  192. package/src/api/handlers/marketplace.ts +930 -0
  193. package/src/api/handlers/media.ts +207 -0
  194. package/src/api/handlers/menus.ts +493 -0
  195. package/src/api/handlers/oauth-authorization.ts +429 -0
  196. package/src/api/handlers/oauth-clients.ts +353 -0
  197. package/src/api/handlers/oauth-user-lookup.ts +39 -0
  198. package/src/api/handlers/plugins.ts +254 -0
  199. package/src/api/handlers/redirects.ts +360 -0
  200. package/src/api/handlers/revision.ts +145 -0
  201. package/src/api/handlers/schema.ts +534 -0
  202. package/src/api/handlers/sections.ts +289 -0
  203. package/src/api/handlers/seo.ts +115 -0
  204. package/src/api/handlers/settings.ts +49 -0
  205. package/src/api/handlers/snapshot.ts +350 -0
  206. package/src/api/handlers/taxonomies.ts +523 -0
  207. package/src/api/index.ts +6 -0
  208. package/src/api/openapi/document.ts +2368 -0
  209. package/src/api/openapi/index.ts +1 -0
  210. package/src/api/parse.ts +139 -0
  211. package/src/api/redirect.ts +14 -0
  212. package/src/api/rev.ts +67 -0
  213. package/src/api/schemas/auth.ts +112 -0
  214. package/src/api/schemas/bylines.ts +85 -0
  215. package/src/api/schemas/comments.ts +117 -0
  216. package/src/api/schemas/common.ts +89 -0
  217. package/src/api/schemas/content.ts +191 -0
  218. package/src/api/schemas/import.ts +52 -0
  219. package/src/api/schemas/index.ts +17 -0
  220. package/src/api/schemas/media.ts +116 -0
  221. package/src/api/schemas/menus.ts +111 -0
  222. package/src/api/schemas/redirects.ts +155 -0
  223. package/src/api/schemas/schema.ts +203 -0
  224. package/src/api/schemas/search.ts +63 -0
  225. package/src/api/schemas/sections.ts +67 -0
  226. package/src/api/schemas/settings.ts +63 -0
  227. package/src/api/schemas/setup.ts +37 -0
  228. package/src/api/schemas/taxonomies.ts +113 -0
  229. package/src/api/schemas/users.ts +96 -0
  230. package/src/api/schemas/widgets.ts +80 -0
  231. package/src/api/site-url.ts +25 -0
  232. package/src/api/types.ts +82 -0
  233. package/src/astro/index.ts +27 -0
  234. package/src/astro/integration/index.ts +303 -0
  235. package/src/astro/integration/routes.ts +834 -0
  236. package/src/astro/integration/runtime.ts +338 -0
  237. package/src/astro/integration/virtual-modules.ts +469 -0
  238. package/src/astro/integration/vite-config.ts +328 -0
  239. package/src/astro/middleware/auth.ts +743 -0
  240. package/src/astro/middleware/redirect.ts +89 -0
  241. package/src/astro/middleware/request-context.ts +129 -0
  242. package/src/astro/middleware/setup.ts +89 -0
  243. package/src/astro/middleware.ts +398 -0
  244. package/src/astro/routes/PluginRegistry.tsx +15 -0
  245. package/src/astro/routes/admin.astro +81 -0
  246. package/src/astro/routes/api/admin/allowed-domains/[domain].ts +112 -0
  247. package/src/astro/routes/api/admin/allowed-domains/index.ts +108 -0
  248. package/src/astro/routes/api/admin/api-tokens/[id].ts +40 -0
  249. package/src/astro/routes/api/admin/api-tokens/index.ts +68 -0
  250. package/src/astro/routes/api/admin/bylines/[id]/index.ts +87 -0
  251. package/src/astro/routes/api/admin/bylines/index.ts +72 -0
  252. package/src/astro/routes/api/admin/comments/[id]/status.ts +120 -0
  253. package/src/astro/routes/api/admin/comments/[id].ts +64 -0
  254. package/src/astro/routes/api/admin/comments/bulk.ts +42 -0
  255. package/src/astro/routes/api/admin/comments/counts.ts +30 -0
  256. package/src/astro/routes/api/admin/comments/index.ts +46 -0
  257. package/src/astro/routes/api/admin/hooks/exclusive/[hookName].ts +91 -0
  258. package/src/astro/routes/api/admin/hooks/exclusive/index.ts +51 -0
  259. package/src/astro/routes/api/admin/oauth-clients/[id].ts +110 -0
  260. package/src/astro/routes/api/admin/oauth-clients/index.ts +71 -0
  261. package/src/astro/routes/api/admin/plugins/[id]/disable.ts +39 -0
  262. package/src/astro/routes/api/admin/plugins/[id]/enable.ts +39 -0
  263. package/src/astro/routes/api/admin/plugins/[id]/index.ts +38 -0
  264. package/src/astro/routes/api/admin/plugins/[id]/uninstall.ts +48 -0
  265. package/src/astro/routes/api/admin/plugins/[id]/update.ts +59 -0
  266. package/src/astro/routes/api/admin/plugins/index.ts +32 -0
  267. package/src/astro/routes/api/admin/plugins/marketplace/[id]/icon.ts +61 -0
  268. package/src/astro/routes/api/admin/plugins/marketplace/[id]/index.ts +33 -0
  269. package/src/astro/routes/api/admin/plugins/marketplace/[id]/install.ts +62 -0
  270. package/src/astro/routes/api/admin/plugins/marketplace/index.ts +38 -0
  271. package/src/astro/routes/api/admin/plugins/updates.ts +28 -0
  272. package/src/astro/routes/api/admin/themes/marketplace/[id]/index.ts +33 -0
  273. package/src/astro/routes/api/admin/themes/marketplace/[id]/thumbnail.ts +61 -0
  274. package/src/astro/routes/api/admin/themes/marketplace/index.ts +45 -0
  275. package/src/astro/routes/api/admin/users/[id]/disable.ts +69 -0
  276. package/src/astro/routes/api/admin/users/[id]/enable.ts +48 -0
  277. package/src/astro/routes/api/admin/users/[id]/index.ts +146 -0
  278. package/src/astro/routes/api/admin/users/[id]/send-recovery.ts +72 -0
  279. package/src/astro/routes/api/admin/users/index.ts +66 -0
  280. package/src/astro/routes/api/auth/dev-bypass.ts +139 -0
  281. package/src/astro/routes/api/auth/invite/accept.ts +52 -0
  282. package/src/astro/routes/api/auth/invite/complete.ts +84 -0
  283. package/src/astro/routes/api/auth/invite/index.ts +99 -0
  284. package/src/astro/routes/api/auth/logout.ts +40 -0
  285. package/src/astro/routes/api/auth/magic-link/send.ts +89 -0
  286. package/src/astro/routes/api/auth/magic-link/verify.ts +71 -0
  287. package/src/astro/routes/api/auth/me.ts +60 -0
  288. package/src/astro/routes/api/auth/oauth/[provider]/callback.ts +219 -0
  289. package/src/astro/routes/api/auth/oauth/[provider].ts +119 -0
  290. package/src/astro/routes/api/auth/passkey/[id].ts +124 -0
  291. package/src/astro/routes/api/auth/passkey/index.ts +54 -0
  292. package/src/astro/routes/api/auth/passkey/options.ts +82 -0
  293. package/src/astro/routes/api/auth/passkey/register/options.ts +86 -0
  294. package/src/astro/routes/api/auth/passkey/register/verify.ts +117 -0
  295. package/src/astro/routes/api/auth/passkey/verify.ts +66 -0
  296. package/src/astro/routes/api/auth/signup/complete.ts +85 -0
  297. package/src/astro/routes/api/auth/signup/request.ts +77 -0
  298. package/src/astro/routes/api/auth/signup/verify.ts +53 -0
  299. package/src/astro/routes/api/comments/[collection]/[contentId]/index.ts +312 -0
  300. package/src/astro/routes/api/content/[collection]/[id]/compare.ts +28 -0
  301. package/src/astro/routes/api/content/[collection]/[id]/discard-draft.ts +54 -0
  302. package/src/astro/routes/api/content/[collection]/[id]/duplicate.ts +61 -0
  303. package/src/astro/routes/api/content/[collection]/[id]/permanent.ts +33 -0
  304. package/src/astro/routes/api/content/[collection]/[id]/preview-url.ts +107 -0
  305. package/src/astro/routes/api/content/[collection]/[id]/publish.ts +56 -0
  306. package/src/astro/routes/api/content/[collection]/[id]/restore.ts +54 -0
  307. package/src/astro/routes/api/content/[collection]/[id]/revisions.ts +31 -0
  308. package/src/astro/routes/api/content/[collection]/[id]/schedule.ts +105 -0
  309. package/src/astro/routes/api/content/[collection]/[id]/terms/[taxonomy].ts +140 -0
  310. package/src/astro/routes/api/content/[collection]/[id]/translations.ts +30 -0
  311. package/src/astro/routes/api/content/[collection]/[id]/unpublish.ts +56 -0
  312. package/src/astro/routes/api/content/[collection]/[id].ts +137 -0
  313. package/src/astro/routes/api/content/[collection]/index.ts +59 -0
  314. package/src/astro/routes/api/content/[collection]/trash.ts +33 -0
  315. package/src/astro/routes/api/dashboard.ts +32 -0
  316. package/src/astro/routes/api/dev/emails.ts +36 -0
  317. package/src/astro/routes/api/import/probe.ts +47 -0
  318. package/src/astro/routes/api/import/wordpress/analyze.ts +510 -0
  319. package/src/astro/routes/api/import/wordpress/execute.ts +283 -0
  320. package/src/astro/routes/api/import/wordpress/media.ts +338 -0
  321. package/src/astro/routes/api/import/wordpress/prepare.ts +181 -0
  322. package/src/astro/routes/api/import/wordpress/rewrite-urls.ts +393 -0
  323. package/src/astro/routes/api/import/wordpress-plugin/analyze.ts +111 -0
  324. package/src/astro/routes/api/import/wordpress-plugin/callback.ts +58 -0
  325. package/src/astro/routes/api/import/wordpress-plugin/execute.ts +347 -0
  326. package/src/astro/routes/api/manifest.ts +62 -0
  327. package/src/astro/routes/api/mcp.ts +124 -0
  328. package/src/astro/routes/api/media/[id]/confirm.ts +93 -0
  329. package/src/astro/routes/api/media/[id].ts +145 -0
  330. package/src/astro/routes/api/media/file/[key].ts +79 -0
  331. package/src/astro/routes/api/media/providers/[providerId]/[itemId].ts +86 -0
  332. package/src/astro/routes/api/media/providers/[providerId]/index.ts +111 -0
  333. package/src/astro/routes/api/media/providers/index.ts +30 -0
  334. package/src/astro/routes/api/media/upload-url.ts +137 -0
  335. package/src/astro/routes/api/media.ts +190 -0
  336. package/src/astro/routes/api/menus/[name]/items.ts +87 -0
  337. package/src/astro/routes/api/menus/[name]/reorder.ts +33 -0
  338. package/src/astro/routes/api/menus/[name].ts +65 -0
  339. package/src/astro/routes/api/menus/index.ts +47 -0
  340. package/src/astro/routes/api/oauth/authorize.ts +412 -0
  341. package/src/astro/routes/api/oauth/device/authorize.ts +45 -0
  342. package/src/astro/routes/api/oauth/device/code.ts +51 -0
  343. package/src/astro/routes/api/oauth/device/token.ts +69 -0
  344. package/src/astro/routes/api/oauth/token/refresh.ts +38 -0
  345. package/src/astro/routes/api/oauth/token/revoke.ts +38 -0
  346. package/src/astro/routes/api/oauth/token.ts +184 -0
  347. package/src/astro/routes/api/openapi.json.ts +32 -0
  348. package/src/astro/routes/api/plugins/[pluginId]/[...path].ts +92 -0
  349. package/src/astro/routes/api/redirects/404s/index.ts +72 -0
  350. package/src/astro/routes/api/redirects/404s/summary.ts +33 -0
  351. package/src/astro/routes/api/redirects/[id].ts +84 -0
  352. package/src/astro/routes/api/redirects/index.ts +52 -0
  353. package/src/astro/routes/api/revisions/[revisionId]/index.ts +29 -0
  354. package/src/astro/routes/api/revisions/[revisionId]/restore.ts +62 -0
  355. package/src/astro/routes/api/schema/collections/[slug]/fields/[fieldSlug].ts +76 -0
  356. package/src/astro/routes/api/schema/collections/[slug]/fields/index.ts +52 -0
  357. package/src/astro/routes/api/schema/collections/[slug]/fields/reorder.ts +32 -0
  358. package/src/astro/routes/api/schema/collections/[slug]/index.ts +80 -0
  359. package/src/astro/routes/api/schema/collections/index.ts +47 -0
  360. package/src/astro/routes/api/schema/index.ts +109 -0
  361. package/src/astro/routes/api/schema/orphans/[slug].ts +36 -0
  362. package/src/astro/routes/api/schema/orphans/index.ts +26 -0
  363. package/src/astro/routes/api/search/enable.ts +64 -0
  364. package/src/astro/routes/api/search/index.ts +55 -0
  365. package/src/astro/routes/api/search/rebuild.ts +72 -0
  366. package/src/astro/routes/api/search/stats.ts +35 -0
  367. package/src/astro/routes/api/search/suggest.ts +53 -0
  368. package/src/astro/routes/api/sections/[slug].ts +84 -0
  369. package/src/astro/routes/api/sections/index.ts +52 -0
  370. package/src/astro/routes/api/settings/email.ts +150 -0
  371. package/src/astro/routes/api/settings.ts +67 -0
  372. package/src/astro/routes/api/setup/admin-verify.ts +100 -0
  373. package/src/astro/routes/api/setup/admin.ts +94 -0
  374. package/src/astro/routes/api/setup/dev-bypass.ts +199 -0
  375. package/src/astro/routes/api/setup/dev-reset.ts +40 -0
  376. package/src/astro/routes/api/setup/index.ts +126 -0
  377. package/src/astro/routes/api/setup/status.ts +122 -0
  378. package/src/astro/routes/api/snapshot.ts +75 -0
  379. package/src/astro/routes/api/taxonomies/[name]/terms/[slug].ts +95 -0
  380. package/src/astro/routes/api/taxonomies/[name]/terms/index.ts +69 -0
  381. package/src/astro/routes/api/taxonomies/index.ts +59 -0
  382. package/src/astro/routes/api/themes/preview.ts +77 -0
  383. package/src/astro/routes/api/typegen.ts +114 -0
  384. package/src/astro/routes/api/well-known/auth.ts +68 -0
  385. package/src/astro/routes/api/well-known/oauth-authorization-server.ts +44 -0
  386. package/src/astro/routes/api/well-known/oauth-protected-resource.ts +37 -0
  387. package/src/astro/routes/api/widget-areas/[name]/reorder.ts +72 -0
  388. package/src/astro/routes/api/widget-areas/[name]/widgets/[id].ts +127 -0
  389. package/src/astro/routes/api/widget-areas/[name]/widgets.ts +80 -0
  390. package/src/astro/routes/api/widget-areas/[name].ts +87 -0
  391. package/src/astro/routes/api/widget-areas/index.ts +99 -0
  392. package/src/astro/routes/api/widget-components.ts +22 -0
  393. package/src/astro/routes/robots.txt.ts +77 -0
  394. package/src/astro/routes/sitemap.xml.ts +97 -0
  395. package/src/astro/storage/adapters.ts +74 -0
  396. package/src/astro/storage/index.ts +19 -0
  397. package/src/astro/storage/types.ts +60 -0
  398. package/src/astro/types.ts +346 -0
  399. package/src/auth/api-tokens.ts +25 -0
  400. package/src/auth/challenge-store.ts +80 -0
  401. package/src/auth/mode.ts +96 -0
  402. package/src/auth/oauth-state-store.ts +96 -0
  403. package/src/auth/passkey-config.ts +27 -0
  404. package/src/auth/rate-limit.ts +158 -0
  405. package/src/auth/scopes.ts +33 -0
  406. package/src/auth/types.ts +104 -0
  407. package/src/aws-sdk.d.ts +100 -0
  408. package/src/bylines/index.ts +237 -0
  409. package/src/cleanup.ts +153 -0
  410. package/src/cli/client-factory.ts +100 -0
  411. package/src/cli/commands/auth.ts +46 -0
  412. package/src/cli/commands/bundle-utils.ts +247 -0
  413. package/src/cli/commands/bundle.ts +609 -0
  414. package/src/cli/commands/content.ts +442 -0
  415. package/src/cli/commands/dev.ts +191 -0
  416. package/src/cli/commands/doctor.ts +211 -0
  417. package/src/cli/commands/export-seed.ts +630 -0
  418. package/src/cli/commands/import/wordpress.ts +1056 -0
  419. package/src/cli/commands/init.ts +192 -0
  420. package/src/cli/commands/login.ts +547 -0
  421. package/src/cli/commands/media.ts +165 -0
  422. package/src/cli/commands/menu.ts +67 -0
  423. package/src/cli/commands/plugin-init.ts +291 -0
  424. package/src/cli/commands/plugin-validate.ts +31 -0
  425. package/src/cli/commands/plugin.ts +33 -0
  426. package/src/cli/commands/publish.ts +699 -0
  427. package/src/cli/commands/schema.ts +233 -0
  428. package/src/cli/commands/search-cmd.ts +54 -0
  429. package/src/cli/commands/seed.ts +288 -0
  430. package/src/cli/commands/taxonomy.ts +128 -0
  431. package/src/cli/commands/types.ts +68 -0
  432. package/src/cli/credentials.ts +236 -0
  433. package/src/cli/index.ts +70 -0
  434. package/src/cli/output.ts +75 -0
  435. package/src/cli/wxr/parser.ts +969 -0
  436. package/src/client/cf-access.ts +193 -0
  437. package/src/client/index.ts +854 -0
  438. package/src/client/portable-text.ts +413 -0
  439. package/src/client/transport.ts +200 -0
  440. package/src/comments/moderator.ts +46 -0
  441. package/src/comments/notifications.ts +144 -0
  442. package/src/comments/query.ts +105 -0
  443. package/src/comments/service.ts +213 -0
  444. package/src/components/Break.astro +45 -0
  445. package/src/components/Button.astro +71 -0
  446. package/src/components/Buttons.astro +49 -0
  447. package/src/components/Code.astro +59 -0
  448. package/src/components/Columns.astro +59 -0
  449. package/src/components/CommentForm.astro +315 -0
  450. package/src/components/Comments.astro +232 -0
  451. package/src/components/Cover.astro +128 -0
  452. package/src/components/EmDashBodyEnd.astro +32 -0
  453. package/src/components/EmDashBodyStart.astro +32 -0
  454. package/src/components/EmDashHead.astro +53 -0
  455. package/src/components/EmDashImage.astro +178 -0
  456. package/src/components/EmDashMedia.astro +167 -0
  457. package/src/components/Embed.astro +128 -0
  458. package/src/components/File.astro +122 -0
  459. package/src/components/Gallery.astro +93 -0
  460. package/src/components/HtmlBlock.astro +33 -0
  461. package/src/components/Image.astro +178 -0
  462. package/src/components/InlineEditor.astro +27 -0
  463. package/src/components/InlinePortableTextEditor.tsx +1905 -0
  464. package/src/components/LiveSearch.astro +614 -0
  465. package/src/components/PortableText.astro +51 -0
  466. package/src/components/Pullquote.astro +51 -0
  467. package/src/components/Table.astro +108 -0
  468. package/src/components/WidgetArea.astro +22 -0
  469. package/src/components/WidgetRenderer.astro +72 -0
  470. package/src/components/index.ts +116 -0
  471. package/src/components/marks/Link.astro +31 -0
  472. package/src/components/marks/StrikeThrough.astro +7 -0
  473. package/src/components/marks/Subscript.astro +7 -0
  474. package/src/components/marks/Superscript.astro +7 -0
  475. package/src/components/marks/Underline.astro +7 -0
  476. package/src/components/widgets/Archives.astro +65 -0
  477. package/src/components/widgets/Categories.astro +35 -0
  478. package/src/components/widgets/RecentPosts.astro +51 -0
  479. package/src/components/widgets/Search.astro +18 -0
  480. package/src/components/widgets/Tags.astro +38 -0
  481. package/src/content/converters/index.ts +9 -0
  482. package/src/content/converters/portable-text-to-prosemirror.ts +385 -0
  483. package/src/content/converters/prosemirror-to-portable-text.ts +413 -0
  484. package/src/content/converters/types.ts +120 -0
  485. package/src/content/index.ts +5 -0
  486. package/src/database/connection.ts +67 -0
  487. package/src/database/dialect-helpers.ts +138 -0
  488. package/src/database/index.ts +5 -0
  489. package/src/database/migrations/001_initial.ts +136 -0
  490. package/src/database/migrations/002_media_status.ts +26 -0
  491. package/src/database/migrations/003_schema_registry.ts +79 -0
  492. package/src/database/migrations/004_plugins.ts +62 -0
  493. package/src/database/migrations/005_menus.ts +67 -0
  494. package/src/database/migrations/006_taxonomy_defs.ts +51 -0
  495. package/src/database/migrations/007_widgets.ts +42 -0
  496. package/src/database/migrations/008_auth.ts +194 -0
  497. package/src/database/migrations/009_user_disabled.ts +27 -0
  498. package/src/database/migrations/011_sections.ts +65 -0
  499. package/src/database/migrations/012_search.ts +25 -0
  500. package/src/database/migrations/013_scheduled_publishing.ts +51 -0
  501. package/src/database/migrations/014_draft_revisions.ts +72 -0
  502. package/src/database/migrations/015_indexes.ts +82 -0
  503. package/src/database/migrations/016_api_tokens.ts +89 -0
  504. package/src/database/migrations/017_authorization_codes.ts +45 -0
  505. package/src/database/migrations/018_seo.ts +56 -0
  506. package/src/database/migrations/019_i18n.ts +618 -0
  507. package/src/database/migrations/020_collection_url_pattern.ts +23 -0
  508. package/src/database/migrations/021_remove_section_categories.ts +43 -0
  509. package/src/database/migrations/022_marketplace_plugin_state.ts +46 -0
  510. package/src/database/migrations/023_plugin_metadata.ts +33 -0
  511. package/src/database/migrations/024_media_placeholders.ts +32 -0
  512. package/src/database/migrations/025_oauth_clients.ts +28 -0
  513. package/src/database/migrations/026_cron_tasks.ts +49 -0
  514. package/src/database/migrations/027_comments.ts +87 -0
  515. package/src/database/migrations/028_drop_author_url.ts +9 -0
  516. package/src/database/migrations/029_redirects.ts +67 -0
  517. package/src/database/migrations/030_widen_scheduled_index.ts +48 -0
  518. package/src/database/migrations/031_bylines.ts +90 -0
  519. package/src/database/migrations/032_rate_limits.ts +42 -0
  520. package/src/database/migrations/runner.ts +170 -0
  521. package/src/database/repositories/audit.ts +294 -0
  522. package/src/database/repositories/byline.ts +387 -0
  523. package/src/database/repositories/comment.ts +458 -0
  524. package/src/database/repositories/content.ts +1144 -0
  525. package/src/database/repositories/index.ts +30 -0
  526. package/src/database/repositories/media.ts +347 -0
  527. package/src/database/repositories/options.ts +150 -0
  528. package/src/database/repositories/plugin-storage.ts +373 -0
  529. package/src/database/repositories/redirect.ts +480 -0
  530. package/src/database/repositories/revision.ts +200 -0
  531. package/src/database/repositories/seo.ts +176 -0
  532. package/src/database/repositories/taxonomy.ts +294 -0
  533. package/src/database/repositories/types.ts +132 -0
  534. package/src/database/repositories/user.ts +258 -0
  535. package/src/database/transaction.ts +54 -0
  536. package/src/database/types.ts +501 -0
  537. package/src/database/validate.ts +138 -0
  538. package/src/db/adapters.ts +125 -0
  539. package/src/db/index.ts +37 -0
  540. package/src/db/libsql.ts +23 -0
  541. package/src/db/postgres.ts +30 -0
  542. package/src/db/sqlite.ts +27 -0
  543. package/src/emdash-runtime.ts +2096 -0
  544. package/src/fields/boolean.ts +34 -0
  545. package/src/fields/datetime.ts +44 -0
  546. package/src/fields/file.ts +41 -0
  547. package/src/fields/image.ts +34 -0
  548. package/src/fields/index.ts +42 -0
  549. package/src/fields/integer.ts +50 -0
  550. package/src/fields/json.ts +37 -0
  551. package/src/fields/multiselect.ts +48 -0
  552. package/src/fields/number.ts +52 -0
  553. package/src/fields/portable-text.ts +33 -0
  554. package/src/fields/reference.ts +29 -0
  555. package/src/fields/richtext.ts +31 -0
  556. package/src/fields/select.ts +46 -0
  557. package/src/fields/slug.ts +38 -0
  558. package/src/fields/text.ts +55 -0
  559. package/src/fields/textarea.ts +52 -0
  560. package/src/fields/types.ts +64 -0
  561. package/src/i18n/config.ts +68 -0
  562. package/src/import/index.ts +90 -0
  563. package/src/import/menus.ts +436 -0
  564. package/src/import/registry.ts +111 -0
  565. package/src/import/sections.ts +103 -0
  566. package/src/import/settings.ts +281 -0
  567. package/src/import/sources/wordpress-plugin.ts +641 -0
  568. package/src/import/sources/wordpress-rest.ts +191 -0
  569. package/src/import/sources/wxr.ts +330 -0
  570. package/src/import/ssrf.ts +260 -0
  571. package/src/import/types.ts +418 -0
  572. package/src/import/utils.ts +412 -0
  573. package/src/index.ts +481 -0
  574. package/src/loader.ts +770 -0
  575. package/src/mcp/server.ts +1463 -0
  576. package/src/media/index.ts +32 -0
  577. package/src/media/local-runtime.ts +213 -0
  578. package/src/media/local.ts +46 -0
  579. package/src/media/normalize.ts +190 -0
  580. package/src/media/placeholder.ts +150 -0
  581. package/src/media/provider-loader.ts +78 -0
  582. package/src/media/types.ts +279 -0
  583. package/src/menus/index.ts +324 -0
  584. package/src/menus/types.ts +112 -0
  585. package/src/page/context.ts +93 -0
  586. package/src/page/fragments.ts +89 -0
  587. package/src/page/index.ts +58 -0
  588. package/src/page/jsonld.ts +94 -0
  589. package/src/page/metadata.ts +185 -0
  590. package/src/page/seo-contributions.ts +136 -0
  591. package/src/plugin-utils.ts +80 -0
  592. package/src/plugins/adapt-sandbox-entry.ts +207 -0
  593. package/src/plugins/context.ts +833 -0
  594. package/src/plugins/cron.ts +361 -0
  595. package/src/plugins/define-plugin.ts +259 -0
  596. package/src/plugins/email-console.ts +73 -0
  597. package/src/plugins/email.ts +209 -0
  598. package/src/plugins/hooks.ts +1273 -0
  599. package/src/plugins/index.ts +193 -0
  600. package/src/plugins/manager.ts +595 -0
  601. package/src/plugins/manifest-schema.ts +230 -0
  602. package/src/plugins/marketplace.ts +460 -0
  603. package/src/plugins/request-meta.ts +139 -0
  604. package/src/plugins/routes.ts +302 -0
  605. package/src/plugins/sandbox/index.ts +18 -0
  606. package/src/plugins/sandbox/noop.ts +76 -0
  607. package/src/plugins/sandbox/types.ts +173 -0
  608. package/src/plugins/scheduler/node.ts +122 -0
  609. package/src/plugins/scheduler/piggyback.ts +71 -0
  610. package/src/plugins/scheduler/types.ts +27 -0
  611. package/src/plugins/state.ts +208 -0
  612. package/src/plugins/storage-indexes.ts +326 -0
  613. package/src/plugins/storage-query.ts +240 -0
  614. package/src/plugins/types.ts +1284 -0
  615. package/src/preview/helpers.ts +27 -0
  616. package/src/preview/index.ts +40 -0
  617. package/src/preview/tokens.ts +279 -0
  618. package/src/preview/urls.ts +118 -0
  619. package/src/query.ts +674 -0
  620. package/src/redirects/patterns.ts +224 -0
  621. package/src/request-context.ts +67 -0
  622. package/src/runtime.ts +21 -0
  623. package/src/schema/index.ts +29 -0
  624. package/src/schema/query.ts +44 -0
  625. package/src/schema/registry.ts +965 -0
  626. package/src/schema/types.ts +276 -0
  627. package/src/schema/zod-generator.ts +413 -0
  628. package/src/search/fts-manager.ts +452 -0
  629. package/src/search/index.ts +26 -0
  630. package/src/search/query.ts +396 -0
  631. package/src/search/text-extraction.ts +162 -0
  632. package/src/search/types.ts +114 -0
  633. package/src/sections/index.ts +226 -0
  634. package/src/sections/types.ts +86 -0
  635. package/src/seed/apply.ts +1141 -0
  636. package/src/seed/default.ts +86 -0
  637. package/src/seed/index.ts +28 -0
  638. package/src/seed/load.ts +35 -0
  639. package/src/seed/types.ts +341 -0
  640. package/src/seed/validate.ts +642 -0
  641. package/src/seo/index.ts +179 -0
  642. package/src/settings/index.ts +203 -0
  643. package/src/settings/types.ts +58 -0
  644. package/src/storage/index.ts +28 -0
  645. package/src/storage/local.ts +253 -0
  646. package/src/storage/s3.ts +271 -0
  647. package/src/storage/types.ts +204 -0
  648. package/src/taxonomies/index.ts +309 -0
  649. package/src/taxonomies/types.ts +61 -0
  650. package/src/ui.ts +75 -0
  651. package/src/utils/base64.ts +73 -0
  652. package/src/utils/hash.ts +36 -0
  653. package/src/utils/sanitize.ts +20 -0
  654. package/src/utils/slugify.ts +29 -0
  655. package/src/utils/url.ts +48 -0
  656. package/src/virtual-modules.d.ts +111 -0
  657. package/src/visual-editing/editable.ts +108 -0
  658. package/src/visual-editing/toolbar.ts +1229 -0
  659. package/src/widgets/components.ts +105 -0
  660. package/src/widgets/index.ts +131 -0
  661. package/src/widgets/types.ts +81 -0
@@ -0,0 +1,1229 @@
1
+ /**
2
+ * EmDash Visual Editing Toolbar
3
+ *
4
+ * A floating pill injected via middleware for authenticated editors.
5
+ * Renders as a plain HTML string with inline styles and a <script> tag.
6
+ * No dependencies — works on any page with a </body> tag.
7
+ */
8
+
9
+ interface ToolbarConfig {
10
+ editMode: boolean;
11
+ isPreview: boolean;
12
+ }
13
+
14
+ export function renderToolbar(config: ToolbarConfig): string {
15
+ const { editMode, isPreview } = config;
16
+
17
+ return `
18
+ <!-- EmDash Visual Editing Toolbar -->
19
+ <div id="emdash-toolbar" data-edit-mode="${editMode}" data-preview="${isPreview}">
20
+ <div class="emdash-tb-inner">
21
+ <span class="emdash-tb-logo">EmDash</span>
22
+
23
+ <div class="emdash-tb-divider"></div>
24
+
25
+ <label class="emdash-tb-toggle" title="Toggle edit mode">
26
+ <input type="checkbox" id="emdash-edit-toggle" ${editMode ? "checked" : ""} />
27
+ <span class="emdash-tb-toggle-track">
28
+ <span class="emdash-tb-toggle-thumb"></span>
29
+ </span>
30
+ <span class="emdash-tb-toggle-label">Edit</span>
31
+ </label>
32
+
33
+ <span class="emdash-tb-status" id="emdash-tb-status"></span>
34
+
35
+ <span class="emdash-tb-save-status" id="emdash-tb-save-status"></span>
36
+
37
+ <a class="emdash-tb-admin" id="emdash-tb-admin" href="#" target="emdash-admin" style="display:none" title="Open in admin">
38
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6"/><polyline points="15 3 21 3 21 9"/><line x1="10" y1="14" x2="21" y2="3"/></svg>
39
+ </a>
40
+
41
+ <button class="emdash-tb-publish" id="emdash-tb-publish" style="display:none">Publish</button>
42
+ </div>
43
+ </div>
44
+
45
+ <style>
46
+ #emdash-toolbar {
47
+ position: fixed;
48
+ bottom: 16px;
49
+ left: 50%;
50
+ transform: translateX(-50%);
51
+ z-index: 999999;
52
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
53
+ font-size: 13px;
54
+ line-height: 1;
55
+ -webkit-font-smoothing: antialiased;
56
+ }
57
+
58
+ .emdash-tb-inner {
59
+ display: flex;
60
+ align-items: center;
61
+ gap: 10px;
62
+ padding: 8px 16px;
63
+ background: #1a1a1a;
64
+ color: #e0e0e0;
65
+ border-radius: 999px;
66
+ box-shadow: 0 4px 24px rgba(0,0,0,0.3), 0 0 0 1px rgba(255,255,255,0.08);
67
+ white-space: nowrap;
68
+ user-select: none;
69
+ }
70
+
71
+ .emdash-tb-logo {
72
+ font-weight: 600;
73
+ font-size: 12px;
74
+ letter-spacing: 0.02em;
75
+ color: #fff;
76
+ opacity: 0.7;
77
+ }
78
+
79
+ .emdash-tb-divider {
80
+ width: 1px;
81
+ height: 16px;
82
+ background: rgba(255,255,255,0.15);
83
+ }
84
+
85
+ /* Toggle switch */
86
+ .emdash-tb-toggle {
87
+ display: flex;
88
+ align-items: center;
89
+ gap: 6px;
90
+ cursor: pointer;
91
+ }
92
+
93
+ .emdash-tb-toggle input {
94
+ position: absolute;
95
+ opacity: 0;
96
+ width: 0;
97
+ height: 0;
98
+ }
99
+
100
+ .emdash-tb-toggle-track {
101
+ position: relative;
102
+ width: 32px;
103
+ height: 18px;
104
+ background: #444;
105
+ border-radius: 9px;
106
+ transition: background 0.2s;
107
+ }
108
+
109
+ .emdash-tb-toggle input:checked + .emdash-tb-toggle-track {
110
+ background: #3b82f6;
111
+ }
112
+
113
+ .emdash-tb-toggle-thumb {
114
+ position: absolute;
115
+ top: 2px;
116
+ left: 2px;
117
+ width: 14px;
118
+ height: 14px;
119
+ background: #fff;
120
+ border-radius: 50%;
121
+ transition: transform 0.2s;
122
+ }
123
+
124
+ .emdash-tb-toggle input:checked + .emdash-tb-toggle-track .emdash-tb-toggle-thumb {
125
+ transform: translateX(14px);
126
+ }
127
+
128
+ .emdash-tb-toggle-label {
129
+ font-size: 12px;
130
+ color: #aaa;
131
+ }
132
+
133
+ .emdash-tb-toggle input:checked ~ .emdash-tb-toggle-label {
134
+ color: #fff;
135
+ }
136
+
137
+ /* Status area — flex for multiple badges */
138
+ .emdash-tb-status {
139
+ display: inline-flex;
140
+ gap: 6px;
141
+ align-items: center;
142
+ }
143
+
144
+ /* Badges */
145
+ .emdash-tb-badge {
146
+ display: inline-flex;
147
+ align-items: center;
148
+ padding: 3px 8px;
149
+ border-radius: 999px;
150
+ font-size: 11px;
151
+ font-weight: 600;
152
+ letter-spacing: 0.02em;
153
+ text-transform: uppercase;
154
+ }
155
+
156
+ .emdash-tb-badge--preview {
157
+ background: rgba(139,92,246,0.2);
158
+ color: #a78bfa;
159
+ }
160
+
161
+ .emdash-tb-badge--draft {
162
+ background: rgba(245,158,11,0.2);
163
+ color: #fbbf24;
164
+ }
165
+
166
+ .emdash-tb-badge--published {
167
+ background: rgba(34,197,94,0.2);
168
+ color: #4ade80;
169
+ }
170
+
171
+ .emdash-tb-badge--pending {
172
+ background: rgba(59,130,246,0.2);
173
+ color: #60a5fa;
174
+ }
175
+
176
+ .emdash-tb-badge--unsaved {
177
+ background: rgba(245,158,11,0.2);
178
+ color: #fbbf24;
179
+ }
180
+
181
+ .emdash-tb-badge--saving {
182
+ background: rgba(148,163,184,0.2);
183
+ color: #94a3b8;
184
+ }
185
+
186
+ .emdash-tb-badge--saved {
187
+ background: rgba(34,197,94,0.2);
188
+ color: #4ade80;
189
+ transition: opacity 0.3s;
190
+ }
191
+
192
+ .emdash-tb-badge--error {
193
+ background: rgba(239,68,68,0.2);
194
+ color: #f87171;
195
+ }
196
+
197
+ /* Admin link */
198
+ .emdash-tb-admin {
199
+ display: inline-flex;
200
+ align-items: center;
201
+ justify-content: center;
202
+ color: #888;
203
+ text-decoration: none;
204
+ padding: 2px;
205
+ border-radius: 4px;
206
+ transition: color 0.15s;
207
+ }
208
+
209
+ .emdash-tb-admin:hover {
210
+ color: #fff;
211
+ }
212
+
213
+ /* Publish button */
214
+ .emdash-tb-publish {
215
+ padding: 4px 12px;
216
+ background: #3b82f6;
217
+ color: #fff;
218
+ border: none;
219
+ border-radius: 999px;
220
+ font-size: 12px;
221
+ font-weight: 600;
222
+ cursor: pointer;
223
+ transition: background 0.15s;
224
+ font-family: inherit;
225
+ }
226
+
227
+ .emdash-tb-publish:hover {
228
+ background: #2563eb;
229
+ }
230
+
231
+ .emdash-tb-publish:disabled {
232
+ opacity: 0.5;
233
+ cursor: not-allowed;
234
+ }
235
+
236
+ /* Edit mode: editable hover styles — uses :has() to check toolbar state */
237
+ body:has(#emdash-toolbar[data-edit-mode="true"]) [data-emdash-ref] {
238
+ transition: box-shadow 0.15s, background-color 0.15s;
239
+ }
240
+
241
+ body:has(#emdash-toolbar[data-edit-mode="true"]) [data-emdash-ref]:hover {
242
+ box-shadow: 0 0 0 2px rgba(59,130,246,0.5);
243
+ border-radius: 4px;
244
+ background-color: rgba(59,130,246,0.04);
245
+ cursor: text;
246
+ }
247
+
248
+ /* Active editing state — override hover pencil cursor */
249
+ [data-emdash-editing] {
250
+ box-shadow: 0 0 0 2px #3b82f6 !important;
251
+ border-radius: 4px !important;
252
+ background-color: rgba(59,130,246,0.04) !important;
253
+ cursor: text !important;
254
+ }
255
+
256
+ /* Suppress browser focus ring on contenteditable and tiptap editor */
257
+ [data-emdash-editing]:focus,
258
+ [data-emdash-ref] .tiptap:focus,
259
+ [data-emdash-ref] .ProseMirror:focus {
260
+ outline: none !important;
261
+ }
262
+
263
+ /* Image editor popover */
264
+ .emdash-img-popover {
265
+ position: fixed;
266
+ z-index: 1000000;
267
+ background: #1a1a1a;
268
+ border-radius: 12px;
269
+ box-shadow: 0 8px 32px rgba(0,0,0,0.4), 0 0 0 1px rgba(255,255,255,0.08);
270
+ color: #e0e0e0;
271
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif;
272
+ font-size: 13px;
273
+ width: 320px;
274
+ overflow: hidden;
275
+ animation: emdash-img-fadein 0.15s ease-out;
276
+ }
277
+
278
+ @keyframes emdash-img-fadein {
279
+ from { opacity: 0; transform: translateY(4px); }
280
+ to { opacity: 1; transform: translateY(0); }
281
+ }
282
+
283
+ .emdash-img-popover-header {
284
+ display: flex;
285
+ align-items: center;
286
+ justify-content: space-between;
287
+ padding: 10px 12px;
288
+ border-bottom: 1px solid rgba(255,255,255,0.08);
289
+ }
290
+
291
+ .emdash-img-popover-title {
292
+ font-weight: 600;
293
+ font-size: 12px;
294
+ text-transform: uppercase;
295
+ letter-spacing: 0.04em;
296
+ color: #999;
297
+ }
298
+
299
+ .emdash-img-popover-close {
300
+ background: none;
301
+ border: none;
302
+ color: #666;
303
+ cursor: pointer;
304
+ padding: 2px;
305
+ line-height: 1;
306
+ font-size: 16px;
307
+ border-radius: 4px;
308
+ transition: color 0.15s;
309
+ }
310
+
311
+ .emdash-img-popover-close:hover {
312
+ color: #fff;
313
+ }
314
+
315
+ .emdash-img-popover-body {
316
+ padding: 12px;
317
+ display: flex;
318
+ flex-direction: column;
319
+ gap: 10px;
320
+ }
321
+
322
+ .emdash-img-preview {
323
+ width: 100%;
324
+ max-height: 160px;
325
+ object-fit: contain;
326
+ border-radius: 6px;
327
+ background: #111;
328
+ }
329
+
330
+ .emdash-img-empty {
331
+ display: flex;
332
+ align-items: center;
333
+ justify-content: center;
334
+ height: 80px;
335
+ border: 2px dashed rgba(255,255,255,0.15);
336
+ border-radius: 6px;
337
+ color: #666;
338
+ font-size: 12px;
339
+ }
340
+
341
+ .emdash-img-field {
342
+ display: flex;
343
+ flex-direction: column;
344
+ gap: 4px;
345
+ }
346
+
347
+ .emdash-img-field label {
348
+ font-size: 11px;
349
+ font-weight: 600;
350
+ color: #888;
351
+ text-transform: uppercase;
352
+ letter-spacing: 0.04em;
353
+ }
354
+
355
+ .emdash-img-field input[type="text"] {
356
+ background: #111;
357
+ border: 1px solid rgba(255,255,255,0.12);
358
+ border-radius: 6px;
359
+ color: #e0e0e0;
360
+ padding: 6px 8px;
361
+ font-size: 13px;
362
+ font-family: inherit;
363
+ outline: none;
364
+ transition: border-color 0.15s;
365
+ }
366
+
367
+ .emdash-img-field input[type="text"]:focus {
368
+ border-color: #3b82f6;
369
+ }
370
+
371
+ .emdash-img-actions {
372
+ display: flex;
373
+ gap: 6px;
374
+ }
375
+
376
+ .emdash-img-btn {
377
+ flex: 1;
378
+ padding: 6px 10px;
379
+ border: 1px solid rgba(255,255,255,0.12);
380
+ border-radius: 6px;
381
+ background: #222;
382
+ color: #e0e0e0;
383
+ font-size: 12px;
384
+ font-family: inherit;
385
+ cursor: pointer;
386
+ transition: background 0.15s, border-color 0.15s;
387
+ text-align: center;
388
+ white-space: nowrap;
389
+ }
390
+
391
+ .emdash-img-btn:hover {
392
+ background: #333;
393
+ border-color: rgba(255,255,255,0.2);
394
+ }
395
+
396
+ .emdash-img-btn--primary {
397
+ background: #3b82f6;
398
+ border-color: #3b82f6;
399
+ color: #fff;
400
+ }
401
+
402
+ .emdash-img-btn--primary:hover {
403
+ background: #2563eb;
404
+ border-color: #2563eb;
405
+ }
406
+
407
+ .emdash-img-btn--danger {
408
+ color: #f87171;
409
+ border-color: rgba(248,113,113,0.3);
410
+ }
411
+
412
+ .emdash-img-btn--danger:hover {
413
+ background: rgba(248,113,113,0.1);
414
+ border-color: rgba(248,113,113,0.5);
415
+ }
416
+
417
+ /* Media browser within the popover */
418
+ .emdash-img-browser {
419
+ border-top: 1px solid rgba(255,255,255,0.08);
420
+ padding: 12px;
421
+ }
422
+
423
+ .emdash-img-browser-header {
424
+ display: flex;
425
+ align-items: center;
426
+ justify-content: space-between;
427
+ margin-bottom: 8px;
428
+ }
429
+
430
+ .emdash-img-browser-title {
431
+ font-size: 12px;
432
+ font-weight: 600;
433
+ color: #999;
434
+ }
435
+
436
+ .emdash-img-browser-back {
437
+ background: none;
438
+ border: none;
439
+ color: #3b82f6;
440
+ cursor: pointer;
441
+ font-size: 12px;
442
+ font-family: inherit;
443
+ padding: 2px 4px;
444
+ }
445
+
446
+ .emdash-img-browser-back:hover {
447
+ text-decoration: underline;
448
+ }
449
+
450
+ .emdash-img-grid {
451
+ display: grid;
452
+ grid-template-columns: repeat(3, 1fr);
453
+ gap: 6px;
454
+ max-height: 240px;
455
+ overflow-y: auto;
456
+ }
457
+
458
+ .emdash-img-grid-item {
459
+ aspect-ratio: 1;
460
+ border-radius: 4px;
461
+ overflow: hidden;
462
+ cursor: pointer;
463
+ border: 2px solid transparent;
464
+ transition: border-color 0.15s;
465
+ background: #111;
466
+ }
467
+
468
+ .emdash-img-grid-item:hover {
469
+ border-color: rgba(59,130,246,0.5);
470
+ }
471
+
472
+ .emdash-img-grid-item--selected {
473
+ border-color: #3b82f6;
474
+ }
475
+
476
+ .emdash-img-grid-item img {
477
+ width: 100%;
478
+ height: 100%;
479
+ object-fit: cover;
480
+ }
481
+
482
+ .emdash-img-loading {
483
+ display: flex;
484
+ align-items: center;
485
+ justify-content: center;
486
+ height: 80px;
487
+ color: #666;
488
+ font-size: 12px;
489
+ }
490
+
491
+ .emdash-img-drop {
492
+ border: 2px dashed #3b82f6;
493
+ background: rgba(59,130,246,0.05);
494
+ }
495
+
496
+ .emdash-img-uploading {
497
+ display: flex;
498
+ align-items: center;
499
+ gap: 8px;
500
+ padding: 8px 0;
501
+ color: #999;
502
+ font-size: 12px;
503
+ }
504
+
505
+ .emdash-img-popover-backdrop {
506
+ position: fixed;
507
+ inset: 0;
508
+ z-index: 999999;
509
+ }
510
+ </style>
511
+
512
+ <script>
513
+ (function() {
514
+ var toolbar = document.getElementById("emdash-toolbar");
515
+ var toggle = document.getElementById("emdash-edit-toggle");
516
+ var statusEl = document.getElementById("emdash-tb-status");
517
+ var saveStatusEl = document.getElementById("emdash-tb-save-status");
518
+ var publishBtn = document.getElementById("emdash-tb-publish");
519
+ if (!toolbar || !toggle || !statusEl || !publishBtn || !saveStatusEl) return;
520
+
521
+ var isEditMode = toolbar.getAttribute("data-edit-mode") === "true";
522
+
523
+ // CSRF-protected fetch — adds X-EmDash-Request header to all API calls
524
+ function ecFetch(url, init) {
525
+ init = init || {};
526
+ init.headers = Object.assign({ "X-EmDash-Request": "1" }, init.headers || {});
527
+ return fetch(url, init);
528
+ }
529
+
530
+ // --- Save status tracking ---
531
+ var saveState = "idle"; // idle | unsaved | saving | saved | error
532
+ var saveHideTimer = null;
533
+
534
+ function setSaveState(state) {
535
+ saveState = state;
536
+ clearTimeout(saveHideTimer);
537
+
538
+ switch (state) {
539
+ case "unsaved":
540
+ saveStatusEl.innerHTML = '<span class="emdash-tb-badge emdash-tb-badge--unsaved">Unsaved</span>';
541
+ break;
542
+ case "saving":
543
+ saveStatusEl.innerHTML = '<span class="emdash-tb-badge emdash-tb-badge--saving">Saving\u2026</span>';
544
+ break;
545
+ case "saved":
546
+ saveStatusEl.innerHTML = '<span class="emdash-tb-badge emdash-tb-badge--saved">Saved</span>';
547
+ saveHideTimer = setTimeout(function() {
548
+ saveStatusEl.innerHTML = "";
549
+ saveState = "idle";
550
+ }, 2000);
551
+ break;
552
+ case "error":
553
+ saveStatusEl.innerHTML = '<span class="emdash-tb-badge emdash-tb-badge--error">Save failed</span>';
554
+ saveHideTimer = setTimeout(function() {
555
+ saveStatusEl.innerHTML = "";
556
+ saveState = "idle";
557
+ }, 3000);
558
+ break;
559
+ default:
560
+ saveStatusEl.innerHTML = "";
561
+ }
562
+ }
563
+
564
+ // Listen for save events from inline editors (e.g. PT editor)
565
+ document.addEventListener("emdash:save", function(e) {
566
+ var detail = e.detail || {};
567
+ if (detail.state) {
568
+ setSaveState(detail.state);
569
+ }
570
+ });
571
+
572
+ document.addEventListener("emdash:content-changed", function(e) {
573
+ var detail = e.detail || {};
574
+ if (detail.collection && detail.id) {
575
+ showUnpublishedChanges(detail.collection, detail.id);
576
+ }
577
+ });
578
+
579
+ // --- Entry status ---
580
+ var entryRef = null;
581
+
582
+ function updateStatus() {
583
+ if (!isEditMode) {
584
+ statusEl.innerHTML = "";
585
+ publishBtn.style.display = "none";
586
+ return;
587
+ }
588
+
589
+ var first = document.querySelector("[data-emdash-ref]");
590
+ if (!first) {
591
+ statusEl.innerHTML = "";
592
+ publishBtn.style.display = "none";
593
+ return;
594
+ }
595
+
596
+ try {
597
+ var ref = JSON.parse(first.getAttribute("data-emdash-ref"));
598
+ entryRef = ref;
599
+ if (!ref.status) return;
600
+
601
+ // Show admin link
602
+ var adminLink = document.getElementById("emdash-tb-admin");
603
+ if (adminLink) {
604
+ adminLink.href = "/_emdash/admin/content/" + encodeURIComponent(ref.collection) + "/" + encodeURIComponent(ref.id);
605
+ adminLink.style.display = "";
606
+ }
607
+
608
+ if (ref.status === "draft") {
609
+ statusEl.innerHTML = '<span class="emdash-tb-badge emdash-tb-badge--draft">Draft</span>';
610
+ publishBtn.style.display = "";
611
+ publishBtn.onclick = function() { publish(ref.collection, ref.id); };
612
+ } else if (ref.status === "published" && ref.hasDraft) {
613
+ statusEl.innerHTML = '<span class="emdash-tb-badge emdash-tb-badge--pending">Unpublished changes</span>';
614
+ publishBtn.style.display = "";
615
+ publishBtn.onclick = function() { publish(ref.collection, ref.id); };
616
+ } else if (ref.status === "published") {
617
+ statusEl.innerHTML = '<span class="emdash-tb-badge emdash-tb-badge--published">Published</span>';
618
+ publishBtn.style.display = "none";
619
+ }
620
+ } catch (e) {
621
+ // ignore parse errors
622
+ }
623
+ }
624
+
625
+ // Publish action
626
+ function publish(collection, id) {
627
+ publishBtn.disabled = true;
628
+ publishBtn.textContent = "Publishing\u2026";
629
+
630
+ ecFetch("/_emdash/api/content/" + encodeURIComponent(collection) + "/" + encodeURIComponent(id) + "/publish", {
631
+ method: "POST",
632
+ credentials: "same-origin",
633
+ })
634
+ .then(function(res) {
635
+ if (res.ok) {
636
+ if (document.startViewTransition) {
637
+ document.startViewTransition(function() { location.reload(); });
638
+ } else {
639
+ location.reload();
640
+ }
641
+ } else {
642
+ publishBtn.disabled = false;
643
+ publishBtn.textContent = "Publish";
644
+ console.error("Publish failed:", res.status);
645
+ }
646
+ })
647
+ .catch(function(err) {
648
+ publishBtn.disabled = false;
649
+ publishBtn.textContent = "Publish";
650
+ console.error("Publish failed:", err);
651
+ });
652
+ }
653
+
654
+ // Edit mode toggle
655
+ toggle.addEventListener("change", function() {
656
+ if (toggle.checked) {
657
+ document.cookie = "emdash-edit-mode=true;path=/;samesite=lax";
658
+ } else {
659
+ document.cookie = "emdash-edit-mode=;path=/;expires=Thu, 01 Jan 1970 00:00:00 GMT";
660
+ }
661
+
662
+ if (document.startViewTransition) {
663
+ document.startViewTransition(function() { location.replace(location.href); });
664
+ } else {
665
+ location.replace(location.href);
666
+ }
667
+ });
668
+
669
+ // --- Inline editing ---
670
+
671
+ // Cached manifest (fetched once on first edit click)
672
+ var manifestCache = null;
673
+ var manifestPromise = null;
674
+
675
+ function fetchManifest() {
676
+ if (manifestCache) return Promise.resolve(manifestCache);
677
+ if (manifestPromise) return manifestPromise;
678
+ manifestPromise = ecFetch("/_emdash/api/manifest", { credentials: "same-origin" })
679
+ .then(function(r) { return r.json(); })
680
+ .then(function(m) { manifestCache = m; return m; });
681
+ return manifestPromise;
682
+ }
683
+
684
+ function getFieldKind(manifest, collection, field) {
685
+ var col = manifest.collections && manifest.collections[collection];
686
+ if (!col || !col.fields) return null;
687
+ var f = col.fields[field];
688
+ return f ? f.kind : null;
689
+ }
690
+
691
+ // Save a single field value
692
+ function saveField(collection, id, field, value) {
693
+ setSaveState("saving");
694
+ return ecFetch("/_emdash/api/content/" + encodeURIComponent(collection) + "/" + encodeURIComponent(id), {
695
+ method: "PUT",
696
+ credentials: "same-origin",
697
+ headers: { "Content-Type": "application/json" },
698
+ body: JSON.stringify({ data: { [field]: value } }),
699
+ })
700
+ .then(function(res) {
701
+ if (res.ok) {
702
+ setSaveState("saved");
703
+ // A save creates/updates a draft — show unpublished changes
704
+ showUnpublishedChanges(collection, id);
705
+ } else {
706
+ setSaveState("error");
707
+ console.error("Save failed:", res.status);
708
+ }
709
+ })
710
+ .catch(function(err) {
711
+ setSaveState("error");
712
+ console.error("Save failed:", err);
713
+ });
714
+ }
715
+
716
+ function showUnpublishedChanges(collection, id) {
717
+ statusEl.innerHTML = '<span class="emdash-tb-badge emdash-tb-badge--pending">Unpublished changes</span>';
718
+ publishBtn.style.display = "";
719
+ publishBtn.disabled = false;
720
+ publishBtn.textContent = "Publish";
721
+ publishBtn.onclick = function() { publish(collection, id); };
722
+ }
723
+
724
+ // Plain text inline editing (contenteditable)
725
+ var currentlyEditing = null;
726
+
727
+ function startTextEdit(element, annotation) {
728
+ if (currentlyEditing === element) return;
729
+ if (currentlyEditing) endCurrentEdit();
730
+
731
+ currentlyEditing = element;
732
+ var originalText = element.textContent || "";
733
+
734
+ element.setAttribute("data-emdash-editing", "");
735
+ element.contentEditable = "plaintext-only";
736
+ element.focus();
737
+
738
+ // Select all text
739
+ var range = document.createRange();
740
+ range.selectNodeContents(element);
741
+ var sel = window.getSelection();
742
+ sel.removeAllRanges();
743
+ sel.addRange(range);
744
+
745
+ // Track dirty state via input events
746
+ function handleInput() {
747
+ var current = (element.textContent || "").trim();
748
+ if (current !== originalText.trim()) {
749
+ setSaveState("unsaved");
750
+ } else {
751
+ setSaveState("idle");
752
+ }
753
+ }
754
+
755
+ function handleBlur() {
756
+ element.removeEventListener("blur", handleBlur);
757
+ element.removeEventListener("keydown", handleKeydown);
758
+ element.removeEventListener("input", handleInput);
759
+ element.contentEditable = "false";
760
+ element.removeAttribute("data-emdash-editing");
761
+ currentlyEditing = null;
762
+
763
+ var newValue = (element.textContent || "").trim();
764
+ if (newValue !== originalText.trim()) {
765
+ saveField(annotation.collection, annotation.id, annotation.field, newValue);
766
+ } else {
767
+ setSaveState("idle");
768
+ }
769
+ }
770
+
771
+ function handleKeydown(e) {
772
+ if (e.key === "Enter" && !e.shiftKey) {
773
+ e.preventDefault();
774
+ element.blur();
775
+ }
776
+ if (e.key === "Escape") {
777
+ element.textContent = originalText;
778
+ setSaveState("idle");
779
+ element.blur();
780
+ }
781
+ }
782
+
783
+ element.addEventListener("input", handleInput);
784
+ element.addEventListener("blur", handleBlur);
785
+ element.addEventListener("keydown", handleKeydown);
786
+ }
787
+
788
+ function endCurrentEdit() {
789
+ if (currentlyEditing) {
790
+ currentlyEditing.blur();
791
+ }
792
+ }
793
+
794
+ // Fallback: open admin
795
+ function openAdmin(annotation) {
796
+ var url = "/_emdash/admin/content/" + encodeURIComponent(annotation.collection) + "/" + encodeURIComponent(annotation.id);
797
+ if (annotation.field) {
798
+ url += "?field=" + encodeURIComponent(annotation.field);
799
+ }
800
+ window.open(url, "emdash-admin");
801
+ }
802
+
803
+ // --- Inline image editing ---
804
+ var activeImagePopover = null;
805
+
806
+ function closeImagePopover() {
807
+ if (activeImagePopover) {
808
+ activeImagePopover.backdrop.remove();
809
+ activeImagePopover.popover.remove();
810
+ if (activeImagePopover.escapeHandler) {
811
+ document.removeEventListener("keydown", activeImagePopover.escapeHandler);
812
+ }
813
+ activeImagePopover = null;
814
+ }
815
+ }
816
+
817
+ function startImageEdit(element, annotation) {
818
+ closeImagePopover();
819
+
820
+ // Find the current image value by fetching the entry
821
+ var collection = annotation.collection;
822
+ var id = annotation.id;
823
+ var field = annotation.field;
824
+
825
+ // Find img element inside the annotated container (or the element itself if it's an img)
826
+ var imgEl = element.tagName === "IMG" ? element : element.querySelector("img");
827
+
828
+ // Fetch current field value from the content API
829
+ ecFetch("/_emdash/api/content/" + encodeURIComponent(collection) + "/" + encodeURIComponent(id), {
830
+ credentials: "same-origin"
831
+ })
832
+ .then(function(r) { return r.json(); })
833
+ .then(function(entry) {
834
+ var currentValue = entry.data && entry.data[field];
835
+ showImagePopover(element, imgEl, annotation, currentValue);
836
+ })
837
+ .catch(function() {
838
+ // If fetch fails, still show popover with what we can infer from DOM
839
+ showImagePopover(element, imgEl, annotation, null);
840
+ });
841
+ }
842
+
843
+ function showImagePopover(element, imgEl, annotation, currentValue) {
844
+ closeImagePopover();
845
+
846
+ var collection = annotation.collection;
847
+ var id = annotation.id;
848
+ var field = annotation.field;
849
+
850
+ // Position near the element
851
+ var rect = element.getBoundingClientRect();
852
+ var viewportH = window.innerHeight;
853
+ var viewportW = window.innerWidth;
854
+
855
+ // Create backdrop for click-outside-to-close
856
+ var backdrop = document.createElement("div");
857
+ backdrop.className = "emdash-img-popover-backdrop";
858
+ backdrop.addEventListener("click", function(e) {
859
+ if (e.target === backdrop) closeImagePopover();
860
+ });
861
+
862
+ // Create popover
863
+ var popover = document.createElement("div");
864
+ popover.className = "emdash-img-popover";
865
+
866
+ var currentSrc = currentValue ? (currentValue.previewUrl || currentValue.src) : (imgEl ? imgEl.src : null);
867
+ var currentAlt = currentValue ? (currentValue.alt || "") : (imgEl ? (imgEl.alt || "") : "");
868
+
869
+ // Build popover HTML
870
+ var html = '';
871
+ html += '<div class="emdash-img-popover-header">';
872
+ html += ' <span class="emdash-img-popover-title">Image</span>';
873
+ html += ' <button class="emdash-img-popover-close" data-action="close">&times;</button>';
874
+ html += '</div>';
875
+ html += '<div class="emdash-img-popover-body" id="emdash-img-main">';
876
+
877
+ if (currentSrc) {
878
+ html += '<img class="emdash-img-preview" src="' + escapeAttr(currentSrc) + '" alt="" />';
879
+ } else {
880
+ html += '<div class="emdash-img-empty">No image selected</div>';
881
+ }
882
+
883
+ html += '<div class="emdash-img-field">';
884
+ html += ' <label for="emdash-img-alt">Alt text</label>';
885
+ html += ' <input type="text" id="emdash-img-alt" value="' + escapeAttr(currentAlt) + '" placeholder="Describe the image" />';
886
+ html += '</div>';
887
+
888
+ html += '<div class="emdash-img-actions">';
889
+ html += ' <button class="emdash-img-btn emdash-img-btn--primary" data-action="browse">Replace</button>';
890
+ html += ' <label class="emdash-img-btn" style="cursor:pointer">';
891
+ html += ' Upload';
892
+ html += ' <input type="file" accept="image/*" id="emdash-img-upload" style="display:none" />';
893
+ html += ' </label>';
894
+ if (currentSrc) {
895
+ html += ' <button class="emdash-img-btn emdash-img-btn--danger" data-action="remove">Remove</button>';
896
+ }
897
+ html += '</div>';
898
+ html += '</div>';
899
+
900
+ popover.innerHTML = html;
901
+
902
+ backdrop.appendChild(popover);
903
+ document.body.appendChild(backdrop);
904
+
905
+ // Position the popover
906
+ positionPopover(popover, rect, viewportW, viewportH);
907
+
908
+ // Escape key handler
909
+ function handleEscape(e) {
910
+ if (e.key === "Escape") {
911
+ closeImagePopover();
912
+ document.removeEventListener("keydown", handleEscape);
913
+ }
914
+ }
915
+ document.addEventListener("keydown", handleEscape);
916
+
917
+ activeImagePopover = {
918
+ backdrop: backdrop,
919
+ popover: popover,
920
+ annotation: annotation,
921
+ currentValue: currentValue,
922
+ element: element,
923
+ imgEl: imgEl,
924
+ escapeHandler: handleEscape
925
+ };
926
+
927
+ // Event handlers
928
+ popover.querySelector('[data-action="close"]').addEventListener("click", closeImagePopover);
929
+
930
+ popover.querySelector('[data-action="browse"]').addEventListener("click", function() {
931
+ showMediaBrowser(popover, annotation, currentValue, element, imgEl);
932
+ });
933
+
934
+ var uploadInput = popover.querySelector("#emdash-img-upload");
935
+ uploadInput.addEventListener("change", function(e) {
936
+ var file = e.target.files && e.target.files[0];
937
+ if (file) handleImageUpload(file, popover, annotation, element, imgEl);
938
+ });
939
+
940
+ var removeBtn = popover.querySelector('[data-action="remove"]');
941
+ if (removeBtn) {
942
+ removeBtn.addEventListener("click", function() {
943
+ saveField(collection, id, field, null).then(function() {
944
+ if (imgEl) {
945
+ imgEl.style.display = "none";
946
+ }
947
+ closeImagePopover();
948
+ });
949
+ });
950
+ }
951
+
952
+ // Save alt text on change (debounced)
953
+ var altInput = popover.querySelector("#emdash-img-alt");
954
+ var altTimer = null;
955
+ altInput.addEventListener("input", function() {
956
+ clearTimeout(altTimer);
957
+ altTimer = setTimeout(function() {
958
+ var newAlt = altInput.value;
959
+ if (currentValue) {
960
+ var updated = Object.assign({}, currentValue, { alt: newAlt });
961
+ saveField(collection, id, field, updated);
962
+ if (imgEl) imgEl.alt = newAlt;
963
+ }
964
+ }, 500);
965
+ });
966
+
967
+ // Handle drag and drop on the popover body
968
+ var body = popover.querySelector(".emdash-img-popover-body");
969
+ body.addEventListener("dragover", function(e) {
970
+ e.preventDefault();
971
+ body.classList.add("emdash-img-drop");
972
+ });
973
+ body.addEventListener("dragleave", function() {
974
+ body.classList.remove("emdash-img-drop");
975
+ });
976
+ body.addEventListener("drop", function(e) {
977
+ e.preventDefault();
978
+ body.classList.remove("emdash-img-drop");
979
+ var file = e.dataTransfer && e.dataTransfer.files && e.dataTransfer.files[0];
980
+ if (file && file.type.startsWith("image/")) {
981
+ handleImageUpload(file, popover, annotation, element, imgEl);
982
+ }
983
+ });
984
+ }
985
+
986
+ function positionPopover(popover, targetRect, viewportW, viewportH) {
987
+ var popoverW = 320;
988
+ var gap = 8;
989
+
990
+ // Try to place to the right of the element
991
+ var left = targetRect.right + gap;
992
+ var top = targetRect.top;
993
+
994
+ // If it overflows right, place to the left
995
+ if (left + popoverW > viewportW - 16) {
996
+ left = targetRect.left - popoverW - gap;
997
+ }
998
+ // If it still overflows (narrow viewport), center below
999
+ if (left < 16) {
1000
+ left = Math.max(16, (viewportW - popoverW) / 2);
1001
+ top = targetRect.bottom + gap;
1002
+ }
1003
+ // Clamp vertically
1004
+ if (top + 400 > viewportH - 80) { // 80 for toolbar
1005
+ top = Math.max(16, viewportH - 480);
1006
+ }
1007
+ if (top < 16) top = 16;
1008
+
1009
+ popover.style.left = left + "px";
1010
+ popover.style.top = top + "px";
1011
+ }
1012
+
1013
+ function escapeAttr(str) {
1014
+ return String(str || "").replace(/&/g, "&amp;").replace(/"/g, "&quot;").replace(/</g, "&lt;").replace(/>/g, "&gt;");
1015
+ }
1016
+
1017
+ function showMediaBrowser(popover, annotation, currentValue, element, imgEl) {
1018
+ var mainBody = popover.querySelector("#emdash-img-main");
1019
+ if (mainBody) mainBody.style.display = "none";
1020
+
1021
+ // Remove existing browser if any
1022
+ var existing = popover.querySelector(".emdash-img-browser");
1023
+ if (existing) existing.remove();
1024
+
1025
+ var browser = document.createElement("div");
1026
+ browser.className = "emdash-img-browser";
1027
+
1028
+ browser.innerHTML = '<div class="emdash-img-browser-header">' +
1029
+ '<span class="emdash-img-browser-title">Media Library</span>' +
1030
+ '<button class="emdash-img-browser-back">Back</button>' +
1031
+ '</div>' +
1032
+ '<div class="emdash-img-loading">Loading\u2026</div>';
1033
+
1034
+ popover.appendChild(browser);
1035
+
1036
+ browser.querySelector(".emdash-img-browser-back").addEventListener("click", function() {
1037
+ browser.remove();
1038
+ if (mainBody) mainBody.style.display = "";
1039
+ });
1040
+
1041
+ // Fetch media
1042
+ ecFetch("/_emdash/api/media?mimeType=image/&limit=30", { credentials: "same-origin" })
1043
+ .then(function(r) { return r.json(); })
1044
+ .then(function(data) {
1045
+ var items = data.items || [];
1046
+ var loadingEl = browser.querySelector(".emdash-img-loading");
1047
+ if (loadingEl) loadingEl.remove();
1048
+
1049
+ if (items.length === 0) {
1050
+ var empty = document.createElement("div");
1051
+ empty.className = "emdash-img-loading";
1052
+ empty.textContent = "No images found";
1053
+ browser.appendChild(empty);
1054
+ return;
1055
+ }
1056
+
1057
+ var grid = document.createElement("div");
1058
+ grid.className = "emdash-img-grid";
1059
+
1060
+ items.forEach(function(item) {
1061
+ var thumb = document.createElement("div");
1062
+ thumb.className = "emdash-img-grid-item";
1063
+ if (currentValue && currentValue.id === item.id) {
1064
+ thumb.classList.add("emdash-img-grid-item--selected");
1065
+ }
1066
+ var thumbUrl = item.url || item.previewUrl || ("/_emdash/api/media/file/" + item.storageKey);
1067
+ thumb.innerHTML = '<img src="' + escapeAttr(thumbUrl) + '" alt="' + escapeAttr(item.alt || item.filename || "") + '" loading="lazy" />';
1068
+
1069
+ thumb.addEventListener("click", function() {
1070
+ selectMediaItem(item, annotation, element, imgEl);
1071
+ });
1072
+
1073
+ grid.appendChild(thumb);
1074
+ });
1075
+
1076
+ browser.appendChild(grid);
1077
+ })
1078
+ .catch(function(err) {
1079
+ var loadingEl = browser.querySelector(".emdash-img-loading");
1080
+ if (loadingEl) loadingEl.textContent = "Failed to load media";
1081
+ console.error("Media fetch error:", err);
1082
+ });
1083
+ }
1084
+
1085
+ function selectMediaItem(item, annotation, element, imgEl) {
1086
+ var collection = annotation.collection;
1087
+ var id = annotation.id;
1088
+ var field = annotation.field;
1089
+
1090
+ var isLocal = !item.provider || item.provider === "local";
1091
+ var itemUrl = item.url || item.previewUrl || ("/_emdash/api/media/file/" + item.storageKey);
1092
+
1093
+ var newValue = {
1094
+ id: item.id,
1095
+ provider: item.provider || "local",
1096
+ src: isLocal ? itemUrl : undefined,
1097
+ previewUrl: isLocal ? undefined : itemUrl,
1098
+ alt: item.alt || "",
1099
+ width: item.width,
1100
+ height: item.height,
1101
+ meta: item.meta
1102
+ };
1103
+
1104
+ // Clean undefined fields
1105
+ Object.keys(newValue).forEach(function(k) {
1106
+ if (newValue[k] === undefined) delete newValue[k];
1107
+ });
1108
+
1109
+ saveField(collection, id, field, newValue).then(function() {
1110
+ // Update the image in the DOM
1111
+ if (imgEl) {
1112
+ imgEl.src = itemUrl;
1113
+ imgEl.alt = item.alt || "";
1114
+ imgEl.style.display = "";
1115
+ }
1116
+ closeImagePopover();
1117
+ });
1118
+ }
1119
+
1120
+ function handleImageUpload(file, popover, annotation, element, imgEl) {
1121
+ var collection = annotation.collection;
1122
+ var id = annotation.id;
1123
+ var field = annotation.field;
1124
+
1125
+ // Show uploading state
1126
+ var mainBody = popover.querySelector("#emdash-img-main");
1127
+ var browserEl = popover.querySelector(".emdash-img-browser");
1128
+ if (browserEl) browserEl.remove();
1129
+ if (mainBody) {
1130
+ mainBody.innerHTML = '<div class="emdash-img-uploading">' +
1131
+ '<span>Uploading ' + escapeAttr(file.name) + '\u2026</span>' +
1132
+ '</div>';
1133
+ mainBody.style.display = "";
1134
+ }
1135
+
1136
+ // Detect dimensions before upload
1137
+ var dimPromise = new Promise(function(resolve) {
1138
+ if (!file.type.startsWith("image/")) return resolve({});
1139
+ var img = new Image();
1140
+ img.onload = function() {
1141
+ resolve({ width: img.naturalWidth, height: img.naturalHeight });
1142
+ URL.revokeObjectURL(img.src);
1143
+ };
1144
+ img.onerror = function() {
1145
+ resolve({});
1146
+ URL.revokeObjectURL(img.src);
1147
+ };
1148
+ img.src = URL.createObjectURL(file);
1149
+ });
1150
+
1151
+ dimPromise.then(function(dims) {
1152
+ var formData = new FormData();
1153
+ formData.append("file", file);
1154
+ if (dims.width) formData.append("width", String(dims.width));
1155
+ if (dims.height) formData.append("height", String(dims.height));
1156
+
1157
+ return ecFetch("/_emdash/api/media", {
1158
+ method: "POST",
1159
+ credentials: "same-origin",
1160
+ body: formData
1161
+ });
1162
+ })
1163
+ .then(function(r) { return r.json(); })
1164
+ .then(function(data) {
1165
+ if (!data.item) throw new Error("Upload failed");
1166
+ var item = data.item;
1167
+ selectMediaItem(item, annotation, element, imgEl);
1168
+ })
1169
+ .catch(function(err) {
1170
+ console.error("Upload error:", err);
1171
+ setSaveState("error");
1172
+ closeImagePopover();
1173
+ });
1174
+ }
1175
+
1176
+ // Click handler for edit mode
1177
+ if (isEditMode) {
1178
+ document.addEventListener("click", function(e) {
1179
+ var target = e.target;
1180
+
1181
+ // Don't intercept clicks on elements currently being edited
1182
+ if (target.hasAttribute && target.hasAttribute("data-emdash-editing")) return;
1183
+
1184
+ // Walk up to find annotated element
1185
+ while (target && target !== document.body) {
1186
+ if (target.hasAttribute && target.hasAttribute("data-emdash-editing")) return;
1187
+
1188
+ var ref = target.getAttribute && target.getAttribute("data-emdash-ref");
1189
+ if (ref) {
1190
+ e.preventDefault();
1191
+ e.stopPropagation();
1192
+
1193
+ try {
1194
+ var annotation = JSON.parse(ref);
1195
+
1196
+ // Entry-level annotation (no field) — ignore, it's a container
1197
+ if (!annotation.field) return;
1198
+
1199
+ // Fetch manifest to determine field type, then dispatch
1200
+ fetchManifest().then(function(manifest) {
1201
+ var kind = getFieldKind(manifest, annotation.collection, annotation.field);
1202
+
1203
+ // Close any open image popover before starting a new edit
1204
+ closeImagePopover();
1205
+
1206
+ if (kind === "string" || kind === "text") {
1207
+ startTextEdit(target, annotation);
1208
+ } else if (kind === "image") {
1209
+ startImageEdit(target, annotation);
1210
+ } else {
1211
+ // Fallback: open admin for unsupported types
1212
+ openAdmin(annotation);
1213
+ }
1214
+ });
1215
+ } catch (err) {
1216
+ console.error("Failed to parse emdash ref:", err);
1217
+ }
1218
+ return;
1219
+ }
1220
+ target = target.parentElement;
1221
+ }
1222
+ }, true);
1223
+ }
1224
+
1225
+ updateStatus();
1226
+ })();
1227
+ </script>
1228
+ `;
1229
+ }