videowright 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 (306) hide show
  1. package/README.md +91 -0
  2. package/dist/cli/argv.d.ts +28 -0
  3. package/dist/cli/argv.d.ts.map +1 -0
  4. package/dist/cli/argv.js +115 -0
  5. package/dist/cli/argv.js.map +1 -0
  6. package/dist/cli/bin.d.ts +7 -0
  7. package/dist/cli/bin.d.ts.map +1 -0
  8. package/dist/cli/bin.js +10 -0
  9. package/dist/cli/bin.js.map +1 -0
  10. package/dist/cli/dev.d.ts +19 -0
  11. package/dist/cli/dev.d.ts.map +1 -0
  12. package/dist/cli/dev.js +104 -0
  13. package/dist/cli/dev.js.map +1 -0
  14. package/dist/cli/discover.d.ts +29 -0
  15. package/dist/cli/discover.d.ts.map +1 -0
  16. package/dist/cli/discover.js +104 -0
  17. package/dist/cli/discover.js.map +1 -0
  18. package/dist/cli/discover_project.d.ts +29 -0
  19. package/dist/cli/discover_project.d.ts.map +1 -0
  20. package/dist/cli/discover_project.js +108 -0
  21. package/dist/cli/discover_project.js.map +1 -0
  22. package/dist/cli/errors.d.ts +10 -0
  23. package/dist/cli/errors.d.ts.map +1 -0
  24. package/dist/cli/errors.js +13 -0
  25. package/dist/cli/errors.js.map +1 -0
  26. package/dist/cli/ffmpeg.d.ts +57 -0
  27. package/dist/cli/ffmpeg.d.ts.map +1 -0
  28. package/dist/cli/ffmpeg.js +122 -0
  29. package/dist/cli/ffmpeg.js.map +1 -0
  30. package/dist/cli/index.d.ts +7 -0
  31. package/dist/cli/index.d.ts.map +1 -0
  32. package/dist/cli/index.js +152 -0
  33. package/dist/cli/index.js.map +1 -0
  34. package/dist/cli/playwright_check.d.ts +44 -0
  35. package/dist/cli/playwright_check.d.ts.map +1 -0
  36. package/dist/cli/playwright_check.js +20 -0
  37. package/dist/cli/playwright_check.js.map +1 -0
  38. package/dist/cli/prompt.d.ts +13 -0
  39. package/dist/cli/prompt.d.ts.map +1 -0
  40. package/dist/cli/prompt.js +47 -0
  41. package/dist/cli/prompt.js.map +1 -0
  42. package/dist/cli/render.d.ts +60 -0
  43. package/dist/cli/render.d.ts.map +1 -0
  44. package/dist/cli/render.js +471 -0
  45. package/dist/cli/render.js.map +1 -0
  46. package/dist/cli/script_cmd.d.ts +26 -0
  47. package/dist/cli/script_cmd.d.ts.map +1 -0
  48. package/dist/cli/script_cmd.js +88 -0
  49. package/dist/cli/script_cmd.js.map +1 -0
  50. package/dist/cli/time_shim.d.ts +44 -0
  51. package/dist/cli/time_shim.d.ts.map +1 -0
  52. package/dist/cli/time_shim.js +390 -0
  53. package/dist/cli/time_shim.js.map +1 -0
  54. package/dist/cli/ts_loader.d.ts +28 -0
  55. package/dist/cli/ts_loader.d.ts.map +1 -0
  56. package/dist/cli/ts_loader.js +95 -0
  57. package/dist/cli/ts_loader.js.map +1 -0
  58. package/dist/cli/vite_helpers.d.ts +62 -0
  59. package/dist/cli/vite_helpers.d.ts.map +1 -0
  60. package/dist/cli/vite_helpers.js +273 -0
  61. package/dist/cli/vite_helpers.js.map +1 -0
  62. package/dist/index.d.ts +11 -0
  63. package/dist/index.d.ts.map +1 -0
  64. package/dist/index.js +14 -0
  65. package/dist/index.js.map +1 -0
  66. package/dist/player/hash_router.d.ts +23 -0
  67. package/dist/player/hash_router.d.ts.map +1 -0
  68. package/dist/player/hash_router.js +49 -0
  69. package/dist/player/hash_router.js.map +1 -0
  70. package/dist/player/hud.d.ts +33 -0
  71. package/dist/player/hud.d.ts.map +1 -0
  72. package/dist/player/hud.js +357 -0
  73. package/dist/player/hud.js.map +1 -0
  74. package/dist/player/index.d.ts +123 -0
  75. package/dist/player/index.d.ts.map +1 -0
  76. package/dist/player/index.js +848 -0
  77. package/dist/player/index.js.map +1 -0
  78. package/dist/player/input.d.ts +14 -0
  79. package/dist/player/input.d.ts.map +1 -0
  80. package/dist/player/input.js +90 -0
  81. package/dist/player/input.js.map +1 -0
  82. package/dist/player/slot.d.ts +22 -0
  83. package/dist/player/slot.d.ts.map +1 -0
  84. package/dist/player/slot.js +43 -0
  85. package/dist/player/slot.js.map +1 -0
  86. package/dist/player/transitions/cut.d.ts +7 -0
  87. package/dist/player/transitions/cut.d.ts.map +1 -0
  88. package/dist/player/transitions/cut.js +9 -0
  89. package/dist/player/transitions/cut.js.map +1 -0
  90. package/dist/player/transitions/fade.d.ts +7 -0
  91. package/dist/player/transitions/fade.d.ts.map +1 -0
  92. package/dist/player/transitions/fade.js +18 -0
  93. package/dist/player/transitions/fade.js.map +1 -0
  94. package/dist/player/transitions/index.d.ts +4 -0
  95. package/dist/player/transitions/index.d.ts.map +1 -0
  96. package/dist/player/transitions/index.js +4 -0
  97. package/dist/player/transitions/index.js.map +1 -0
  98. package/dist/player/transitions/slide.d.ts +6 -0
  99. package/dist/player/transitions/slide.d.ts.map +1 -0
  100. package/dist/player/transitions/slide.js +35 -0
  101. package/dist/player/transitions/slide.js.map +1 -0
  102. package/dist/script/index.d.ts +2 -0
  103. package/dist/script/index.d.ts.map +1 -0
  104. package/dist/script/index.js +2 -0
  105. package/dist/script/index.js.map +1 -0
  106. package/dist/script/script.d.ts +10 -0
  107. package/dist/script/script.d.ts.map +1 -0
  108. package/dist/script/script.js +41 -0
  109. package/dist/script/script.js.map +1 -0
  110. package/dist/segment/SegmentRunner.d.ts +52 -0
  111. package/dist/segment/SegmentRunner.d.ts.map +1 -0
  112. package/dist/segment/SegmentRunner.js +187 -0
  113. package/dist/segment/SegmentRunner.js.map +1 -0
  114. package/dist/segment/defineConfig.d.ts +6 -0
  115. package/dist/segment/defineConfig.d.ts.map +1 -0
  116. package/dist/segment/defineConfig.js +7 -0
  117. package/dist/segment/defineConfig.js.map +1 -0
  118. package/dist/segment/defineSegment.d.ts +7 -0
  119. package/dist/segment/defineSegment.d.ts.map +1 -0
  120. package/dist/segment/defineSegment.js +25 -0
  121. package/dist/segment/defineSegment.js.map +1 -0
  122. package/dist/segment/index.d.ts +5 -0
  123. package/dist/segment/index.d.ts.map +1 -0
  124. package/dist/segment/index.js +4 -0
  125. package/dist/segment/index.js.map +1 -0
  126. package/dist/timeline/index.d.ts +73 -0
  127. package/dist/timeline/index.d.ts.map +1 -0
  128. package/dist/timeline/index.js +142 -0
  129. package/dist/timeline/index.js.map +1 -0
  130. package/dist/timeline/loadAudioTrack.d.ts +18 -0
  131. package/dist/timeline/loadAudioTrack.d.ts.map +1 -0
  132. package/dist/timeline/loadAudioTrack.js +44 -0
  133. package/dist/timeline/loadAudioTrack.js.map +1 -0
  134. package/dist/timeline/loadVoiceover.d.ts +18 -0
  135. package/dist/timeline/loadVoiceover.d.ts.map +1 -0
  136. package/dist/timeline/loadVoiceover.js +38 -0
  137. package/dist/timeline/loadVoiceover.js.map +1 -0
  138. package/dist/timeline/resolveTiming.d.ts +28 -0
  139. package/dist/timeline/resolveTiming.d.ts.map +1 -0
  140. package/dist/timeline/resolveTiming.js +63 -0
  141. package/dist/timeline/resolveTiming.js.map +1 -0
  142. package/dist/timeline/validateTiming.d.ts +29 -0
  143. package/dist/timeline/validateTiming.d.ts.map +1 -0
  144. package/dist/timeline/validateTiming.js +62 -0
  145. package/dist/timeline/validateTiming.js.map +1 -0
  146. package/dist/types.d.ts +216 -0
  147. package/dist/types.d.ts.map +1 -0
  148. package/dist/types.js +6 -0
  149. package/dist/types.js.map +1 -0
  150. package/package.json +47 -0
  151. package/skill/SKILL.md +64 -0
  152. package/skill/assets/hello_world/PLAN.md +31 -0
  153. package/skill/assets/hello_world/README.md +27 -0
  154. package/skill/assets/hello_world/audio/audio_plan.md +14 -0
  155. package/skill/assets/hello_world/segments/hello_intro.ts +69 -0
  156. package/skill/assets/hello_world/segments/hello_outro.ts +71 -0
  157. package/skill/assets/hello_world/timeline.ts +15 -0
  158. package/skill/assets/hello_world/voiceover_script/script.md +10 -0
  159. package/skill/assets/install/package.json +10 -0
  160. package/skill/assets/install/tsconfig.json +23 -0
  161. package/skill/assets/styles/editorial-mono/STYLE.md +124 -0
  162. package/skill/assets/styles/editorial-mono/brand.md +85 -0
  163. package/skill/assets/styles/editorial-mono/reference/animations.jsx +752 -0
  164. package/skill/assets/styles/editorial-mono/reference/scenes.html +563 -0
  165. package/skill/assets/styles/editorial-mono/sample/bullet.ts +101 -0
  166. package/skill/assets/styles/editorial-mono/sample/content.ts +104 -0
  167. package/skill/assets/styles/editorial-mono/sample/cta.ts +113 -0
  168. package/skill/assets/styles/editorial-mono/sample/feature.ts +111 -0
  169. package/skill/assets/styles/editorial-mono/sample/grid.ts +97 -0
  170. package/skill/assets/styles/editorial-mono/sample/kinetic.ts +96 -0
  171. package/skill/assets/styles/editorial-mono/sample/section.ts +101 -0
  172. package/skill/assets/styles/editorial-mono/sample/stat.ts +128 -0
  173. package/skill/assets/styles/editorial-mono/sample/title.ts +97 -0
  174. package/skill/assets/styles/editorial-mono/sample/ui-showcase.ts +159 -0
  175. package/skill/assets/styles/editorial-mono/tokens.css +44 -0
  176. package/skill/assets/styles/iso-diagram/STYLE.md +109 -0
  177. package/skill/assets/styles/iso-diagram/brand.md +32 -0
  178. package/skill/assets/styles/iso-diagram/reference/animations.jsx +673 -0
  179. package/skill/assets/styles/iso-diagram/reference/scenes.html +427 -0
  180. package/skill/assets/styles/iso-diagram/sample/bullet.ts +144 -0
  181. package/skill/assets/styles/iso-diagram/sample/content.ts +192 -0
  182. package/skill/assets/styles/iso-diagram/sample/cta.ts +162 -0
  183. package/skill/assets/styles/iso-diagram/sample/feature.ts +205 -0
  184. package/skill/assets/styles/iso-diagram/sample/grid.ts +181 -0
  185. package/skill/assets/styles/iso-diagram/sample/kinetic.ts +102 -0
  186. package/skill/assets/styles/iso-diagram/sample/section.ts +149 -0
  187. package/skill/assets/styles/iso-diagram/sample/stat.ts +164 -0
  188. package/skill/assets/styles/iso-diagram/sample/title.ts +173 -0
  189. package/skill/assets/styles/iso-diagram/sample/ui-showcase.ts +162 -0
  190. package/skill/assets/styles/iso-diagram/tokens.css +40 -0
  191. package/skill/assets/styles/motion-engineering/STYLE.md +106 -0
  192. package/skill/assets/styles/motion-engineering/brand.md +29 -0
  193. package/skill/assets/styles/motion-engineering/reference/animations.jsx +673 -0
  194. package/skill/assets/styles/motion-engineering/reference/scenes.html +513 -0
  195. package/skill/assets/styles/motion-engineering/sample/bullet.ts +176 -0
  196. package/skill/assets/styles/motion-engineering/sample/content.ts +228 -0
  197. package/skill/assets/styles/motion-engineering/sample/cta.ts +209 -0
  198. package/skill/assets/styles/motion-engineering/sample/feature.ts +299 -0
  199. package/skill/assets/styles/motion-engineering/sample/grid.ts +190 -0
  200. package/skill/assets/styles/motion-engineering/sample/kinetic.ts +159 -0
  201. package/skill/assets/styles/motion-engineering/sample/section.ts +196 -0
  202. package/skill/assets/styles/motion-engineering/sample/stat.ts +230 -0
  203. package/skill/assets/styles/motion-engineering/sample/title.ts +219 -0
  204. package/skill/assets/styles/motion-engineering/sample/ui-showcase.ts +267 -0
  205. package/skill/assets/styles/motion-engineering/tokens.css +40 -0
  206. package/skill/assets/styles/neon-terminal/STYLE.md +105 -0
  207. package/skill/assets/styles/neon-terminal/brand.md +27 -0
  208. package/skill/assets/styles/neon-terminal/reference/animations.jsx +673 -0
  209. package/skill/assets/styles/neon-terminal/reference/scenes.html +387 -0
  210. package/skill/assets/styles/neon-terminal/sample/bullet.ts +113 -0
  211. package/skill/assets/styles/neon-terminal/sample/content.ts +117 -0
  212. package/skill/assets/styles/neon-terminal/sample/cta.ts +131 -0
  213. package/skill/assets/styles/neon-terminal/sample/feature.ts +112 -0
  214. package/skill/assets/styles/neon-terminal/sample/grid.ts +128 -0
  215. package/skill/assets/styles/neon-terminal/sample/kinetic.ts +105 -0
  216. package/skill/assets/styles/neon-terminal/sample/section.ts +96 -0
  217. package/skill/assets/styles/neon-terminal/sample/stat.ts +123 -0
  218. package/skill/assets/styles/neon-terminal/sample/title.ts +122 -0
  219. package/skill/assets/styles/neon-terminal/sample/ui-showcase.ts +127 -0
  220. package/skill/assets/styles/neon-terminal/tokens.css +39 -0
  221. package/skill/assets/styles/risograph/STYLE.md +110 -0
  222. package/skill/assets/styles/risograph/brand.md +26 -0
  223. package/skill/assets/styles/risograph/reference/animations.jsx +673 -0
  224. package/skill/assets/styles/risograph/reference/scenes.html +403 -0
  225. package/skill/assets/styles/risograph/sample/bullet.ts +124 -0
  226. package/skill/assets/styles/risograph/sample/content.ts +135 -0
  227. package/skill/assets/styles/risograph/sample/cta.ts +149 -0
  228. package/skill/assets/styles/risograph/sample/feature.ts +152 -0
  229. package/skill/assets/styles/risograph/sample/grid.ts +123 -0
  230. package/skill/assets/styles/risograph/sample/kinetic.ts +125 -0
  231. package/skill/assets/styles/risograph/sample/section.ts +130 -0
  232. package/skill/assets/styles/risograph/sample/stat.ts +145 -0
  233. package/skill/assets/styles/risograph/sample/title.ts +132 -0
  234. package/skill/assets/styles/risograph/sample/ui-showcase.ts +147 -0
  235. package/skill/assets/styles/risograph/tokens.css +39 -0
  236. package/skill/assets/styles/swiss-console/STYLE.md +107 -0
  237. package/skill/assets/styles/swiss-console/brand.md +37 -0
  238. package/skill/assets/styles/swiss-console/reference/animations.jsx +673 -0
  239. package/skill/assets/styles/swiss-console/reference/scenes.html +420 -0
  240. package/skill/assets/styles/swiss-console/sample/bullet.ts +122 -0
  241. package/skill/assets/styles/swiss-console/sample/content.ts +137 -0
  242. package/skill/assets/styles/swiss-console/sample/cta.ts +109 -0
  243. package/skill/assets/styles/swiss-console/sample/feature.ts +163 -0
  244. package/skill/assets/styles/swiss-console/sample/grid.ts +145 -0
  245. package/skill/assets/styles/swiss-console/sample/kinetic.ts +117 -0
  246. package/skill/assets/styles/swiss-console/sample/section.ts +127 -0
  247. package/skill/assets/styles/swiss-console/sample/stat.ts +148 -0
  248. package/skill/assets/styles/swiss-console/sample/title.ts +148 -0
  249. package/skill/assets/styles/swiss-console/sample/ui-showcase.ts +198 -0
  250. package/skill/assets/styles/swiss-console/tokens.css +39 -0
  251. package/skill/install/INSTALL.md +400 -0
  252. package/skill/references/audio/audio_plan.md +199 -0
  253. package/skill/references/audio/build.md +208 -0
  254. package/skill/references/audio/cue_template.md +219 -0
  255. package/skill/references/audio/ffmpeg_cookbook.md +267 -0
  256. package/skill/references/audio/music/music.md +171 -0
  257. package/skill/references/audio/music/providers/elevenlabs.md +170 -0
  258. package/skill/references/audio/music/providers/manual.md +140 -0
  259. package/skill/references/audio/music/providers/openverse.md +265 -0
  260. package/skill/references/audio/sfx/providers/elevenlabs.md +152 -0
  261. package/skill/references/audio/sfx/providers/manual.md +117 -0
  262. package/skill/references/audio/sfx/providers/openverse.md +243 -0
  263. package/skill/references/audio/sfx/sfx.md +149 -0
  264. package/skill/references/audio/styles.md +102 -0
  265. package/skill/references/audio/sync.md +237 -0
  266. package/skill/references/audio/voiceover/animation_sync.md +142 -0
  267. package/skill/references/audio/voiceover/provider_script.md +153 -0
  268. package/skill/references/audio/voiceover/providers/elevenlabs.md +288 -0
  269. package/skill/references/audio/voiceover/providers/manual.md +100 -0
  270. package/skill/references/audio/voiceover/script_writing.md +100 -0
  271. package/skill/references/audio/voiceover/style_intake.md +56 -0
  272. package/skill/references/audio/voiceover/sync_algorithm.md +167 -0
  273. package/skill/references/audio/voiceover.md +296 -0
  274. package/skill/references/audio.md +135 -0
  275. package/skill/references/authoring_segment.md +446 -0
  276. package/skill/references/create_or_edit_video.md +232 -0
  277. package/skill/references/dev_server.md +157 -0
  278. package/skill/references/export.md +145 -0
  279. package/skill/references/new_video.md +117 -0
  280. package/skill/references/project_structure.md +144 -0
  281. package/skill/references/setup.md +109 -0
  282. package/skill/references/setup_new_style.md +158 -0
  283. package/skill/references/styles.md +154 -0
  284. package/skill/references/testing.md +115 -0
  285. package/skill/references/types.md +240 -0
  286. package/src/cli/entry/components/copy_button.ts +42 -0
  287. package/src/cli/entry/components/download_modal.ts +204 -0
  288. package/src/cli/entry/components/empty_state.ts +55 -0
  289. package/src/cli/entry/components/hide_hud_tab.ts +37 -0
  290. package/src/cli/entry/components/icons.ts +31 -0
  291. package/src/cli/entry/components/top_bar.ts +69 -0
  292. package/src/cli/entry/components/video_card.ts +57 -0
  293. package/src/cli/entry/dev_frame.ts +189 -0
  294. package/src/cli/entry/entry_index.ts +16 -0
  295. package/src/cli/entry/entry_video.ts +24 -0
  296. package/src/cli/entry/index.html +12 -0
  297. package/src/cli/entry/parse_slug.ts +14 -0
  298. package/src/cli/entry/render.html +17 -0
  299. package/src/cli/entry/render_entry.ts +121 -0
  300. package/src/cli/entry/styles/base.css +45 -0
  301. package/src/cli/entry/styles/components.css +605 -0
  302. package/src/cli/entry/styles/tokens.css +44 -0
  303. package/src/cli/entry/video.html +22 -0
  304. package/src/cli/entry/views/homepage.ts +66 -0
  305. package/src/cli/entry/views/video_view.ts +286 -0
  306. package/src/cli/entry/virtual.d.ts +8 -0
@@ -0,0 +1,605 @@
1
+ /**
2
+ * Component styles for the Videowright dev UI.
3
+ * Uses design tokens from tokens.css.
4
+ */
5
+
6
+ /* ---- Top Bar ---- */
7
+
8
+ .vw-top-bar {
9
+ display: flex;
10
+ align-items: center;
11
+ justify-content: space-between;
12
+ height: 56px;
13
+ padding: 0 24px;
14
+ background: var(--bg-surface);
15
+ border-bottom: 1px solid var(--border-subtle);
16
+ flex-shrink: 0;
17
+ }
18
+
19
+ .vw-top-bar__left {
20
+ display: flex;
21
+ align-items: center;
22
+ gap: 10px;
23
+ min-width: 0;
24
+ }
25
+
26
+ .vw-top-bar__wordmark {
27
+ font-family: var(--font-mono);
28
+ font-size: 14px;
29
+ font-weight: 500;
30
+ color: var(--text-primary);
31
+ text-decoration: none;
32
+ cursor: pointer;
33
+ flex-shrink: 0;
34
+ transition: color var(--duration) var(--ease-out);
35
+ }
36
+
37
+ .vw-top-bar__wordmark:hover {
38
+ color: var(--accent);
39
+ }
40
+
41
+ .vw-top-bar__sep {
42
+ color: var(--text-tertiary);
43
+ font-size: 14px;
44
+ flex-shrink: 0;
45
+ user-select: none;
46
+ }
47
+
48
+ .vw-top-bar__title {
49
+ font-size: 14px;
50
+ font-weight: 500;
51
+ color: var(--text-primary);
52
+ white-space: nowrap;
53
+ overflow: hidden;
54
+ text-overflow: ellipsis;
55
+ min-width: 0;
56
+ }
57
+
58
+ .vw-top-bar__right {
59
+ display: flex;
60
+ align-items: center;
61
+ gap: 12px;
62
+ flex-shrink: 0;
63
+ }
64
+
65
+ .vw-top-bar__project {
66
+ font-size: 13px;
67
+ color: var(--text-secondary);
68
+ }
69
+
70
+ .vw-top-bar__icon-btn {
71
+ display: flex;
72
+ align-items: center;
73
+ justify-content: center;
74
+ width: 32px;
75
+ height: 32px;
76
+ border: none;
77
+ border-radius: var(--radius-sm);
78
+ background: transparent;
79
+ color: var(--text-secondary);
80
+ cursor: pointer;
81
+ transition: background var(--duration) var(--ease-out), color var(--duration) var(--ease-out);
82
+ }
83
+
84
+ .vw-top-bar__icon-btn:hover {
85
+ background: var(--bg-surface-hover);
86
+ color: var(--accent);
87
+ }
88
+
89
+ /* ---- Video Card ---- */
90
+
91
+ .vw-card {
92
+ padding: 16px;
93
+ border-radius: var(--radius-md);
94
+ background: var(--bg-surface);
95
+ border: 1px solid var(--border-subtle);
96
+ cursor: pointer;
97
+ color: inherit;
98
+ text-decoration: none;
99
+ transition: transform var(--duration) var(--ease-out), border-color var(--duration)
100
+ var(--ease-out);
101
+ display: flex;
102
+ flex-direction: column;
103
+ gap: 6px;
104
+ }
105
+
106
+ .vw-card:hover {
107
+ transform: translateY(-1px);
108
+ border-color: var(--border-strong);
109
+ }
110
+
111
+ @media (prefers-reduced-motion: reduce) {
112
+ .vw-card:hover {
113
+ transform: none;
114
+ border-color: var(--border-strong);
115
+ }
116
+ }
117
+
118
+ .vw-card__header {
119
+ display: flex;
120
+ align-items: flex-start;
121
+ justify-content: space-between;
122
+ gap: 8px;
123
+ }
124
+
125
+ .vw-card__title {
126
+ font-size: 16px;
127
+ font-weight: 500;
128
+ color: var(--text-primary);
129
+ line-height: 1.3;
130
+ min-width: 0;
131
+ overflow: hidden;
132
+ text-overflow: ellipsis;
133
+ white-space: nowrap;
134
+ }
135
+
136
+ .vw-card__download {
137
+ display: flex;
138
+ align-items: center;
139
+ justify-content: center;
140
+ width: 32px;
141
+ height: 32px;
142
+ border: none;
143
+ border-radius: var(--radius-sm);
144
+ background: transparent;
145
+ color: var(--text-secondary);
146
+ cursor: pointer;
147
+ flex-shrink: 0;
148
+ transition: background var(--duration) var(--ease-out), color var(--duration) var(--ease-out);
149
+ }
150
+
151
+ .vw-card__download:hover {
152
+ background: var(--bg-surface-hover);
153
+ color: var(--accent);
154
+ }
155
+
156
+ .vw-card__slug {
157
+ font-family: var(--font-mono);
158
+ font-size: 13px;
159
+ color: var(--text-secondary);
160
+ line-height: 1.4;
161
+ }
162
+
163
+ .vw-card__badge {
164
+ display: inline-block;
165
+ align-self: flex-start;
166
+ font-family: var(--font-mono);
167
+ font-size: 11px;
168
+ color: var(--text-secondary);
169
+ border: 1px solid var(--border-subtle);
170
+ border-radius: var(--radius-sm);
171
+ padding: 2px 8px;
172
+ line-height: 1.4;
173
+ margin-top: 4px;
174
+ }
175
+
176
+ /* ---- Empty State ---- */
177
+
178
+ .vw-empty {
179
+ display: flex;
180
+ flex-direction: column;
181
+ align-items: center;
182
+ justify-content: center;
183
+ text-align: center;
184
+ max-width: 440px;
185
+ margin: 0 auto;
186
+ padding: 64px 32px;
187
+ min-height: calc(100vh - 56px - 80px);
188
+ }
189
+
190
+ .vw-empty__hero {
191
+ font-size: 32px;
192
+ font-weight: 600;
193
+ color: var(--text-primary);
194
+ line-height: 1.3;
195
+ margin-bottom: 12px;
196
+ }
197
+
198
+ .vw-empty__desc {
199
+ font-size: 14px;
200
+ color: var(--text-secondary);
201
+ line-height: 1.5;
202
+ margin-bottom: 20px;
203
+ }
204
+
205
+ .vw-empty__code {
206
+ display: flex;
207
+ align-items: center;
208
+ gap: 8px;
209
+ background: var(--bg-surface-elevated);
210
+ border: 1px solid var(--border-subtle);
211
+ border-radius: var(--radius-md);
212
+ padding: 12px 16px;
213
+ margin-bottom: 24px;
214
+ width: 100%;
215
+ max-width: 320px;
216
+ }
217
+
218
+ .vw-empty__code code {
219
+ font-family: var(--font-mono);
220
+ font-size: 13px;
221
+ color: var(--text-primary);
222
+ flex: 1;
223
+ text-align: left;
224
+ }
225
+
226
+ .vw-empty__docs {
227
+ font-size: 13px;
228
+ color: var(--text-secondary);
229
+ }
230
+
231
+ .vw-empty__docs-link {
232
+ color: var(--accent);
233
+ text-decoration: none;
234
+ transition: color var(--duration) var(--ease-out);
235
+ }
236
+
237
+ .vw-empty__docs-link:hover {
238
+ color: var(--text-primary);
239
+ }
240
+
241
+ /* ---- Copy Button ---- */
242
+
243
+ .vw-copy-btn {
244
+ display: flex;
245
+ align-items: center;
246
+ justify-content: center;
247
+ width: 28px;
248
+ height: 28px;
249
+ border: none;
250
+ border-radius: var(--radius-sm);
251
+ background: transparent;
252
+ color: var(--text-secondary);
253
+ cursor: pointer;
254
+ flex-shrink: 0;
255
+ transition: background var(--duration) var(--ease-out), color var(--duration) var(--ease-out);
256
+ }
257
+
258
+ .vw-copy-btn:hover {
259
+ background: var(--bg-surface-hover);
260
+ color: var(--text-primary);
261
+ }
262
+
263
+ .vw-copy-btn--copied {
264
+ color: var(--success);
265
+ }
266
+
267
+ .vw-copy-btn--copied:hover {
268
+ color: var(--success);
269
+ }
270
+
271
+ /* ---- Homepage Layout ---- */
272
+
273
+ .vw-homepage {
274
+ min-height: 100vh;
275
+ background: var(--bg-base);
276
+ }
277
+
278
+ .vw-homepage__content {
279
+ max-width: 1200px;
280
+ margin: 0 auto;
281
+ padding: 32px;
282
+ }
283
+
284
+ .vw-homepage__heading {
285
+ font-size: 24px;
286
+ font-weight: 600;
287
+ color: var(--text-primary);
288
+ line-height: 1.3;
289
+ margin: 0 0 4px 0;
290
+ }
291
+
292
+ .vw-homepage__subtitle {
293
+ font-size: 12px;
294
+ color: var(--text-secondary);
295
+ margin: 0 0 24px 0;
296
+ }
297
+
298
+ .vw-homepage__grid {
299
+ display: grid;
300
+ grid-template-columns: repeat(auto-fill, minmax(280px, 1fr));
301
+ gap: 16px;
302
+ }
303
+
304
+ /* ---- 404 Layout ---- */
305
+
306
+ .vw-not-found {
307
+ display: flex;
308
+ flex-direction: column;
309
+ min-height: 100vh;
310
+ background: var(--bg-base);
311
+ }
312
+
313
+ .vw-not-found__content {
314
+ display: flex;
315
+ flex-direction: column;
316
+ align-items: center;
317
+ justify-content: center;
318
+ flex: 1;
319
+ text-align: center;
320
+ padding: 32px;
321
+ }
322
+
323
+ .vw-not-found__code {
324
+ font-family: var(--font-mono);
325
+ font-size: 64px;
326
+ color: var(--text-tertiary);
327
+ line-height: 1;
328
+ margin-bottom: 16px;
329
+ }
330
+
331
+ .vw-not-found__heading {
332
+ font-size: 24px;
333
+ font-weight: 600;
334
+ color: var(--text-primary);
335
+ line-height: 1.3;
336
+ margin: 0 0 8px 0;
337
+ }
338
+
339
+ .vw-not-found__desc {
340
+ font-size: 14px;
341
+ color: var(--text-secondary);
342
+ margin: 0 0 24px 0;
343
+ }
344
+
345
+ .vw-not-found__link {
346
+ font-size: 14px;
347
+ color: var(--accent);
348
+ text-decoration: none;
349
+ cursor: pointer;
350
+ transition: color var(--duration) var(--ease-out);
351
+ }
352
+
353
+ .vw-not-found__link:hover {
354
+ color: var(--text-primary);
355
+ }
356
+
357
+ /* ---- Download Modal ---- */
358
+
359
+ .vw-modal-backdrop {
360
+ position: fixed;
361
+ inset: 0;
362
+ background: rgba(0, 0, 0, 0.6);
363
+ backdrop-filter: blur(8px);
364
+ display: flex;
365
+ align-items: center;
366
+ justify-content: center;
367
+ z-index: 10000;
368
+ animation: vw-backdrop-in 100ms ease-out;
369
+ }
370
+
371
+ @keyframes vw-backdrop-in {
372
+ from {
373
+ opacity: 0;
374
+ }
375
+ to {
376
+ opacity: 1;
377
+ }
378
+ }
379
+
380
+ .vw-modal {
381
+ background: var(--bg-surface-elevated);
382
+ border-radius: var(--radius-lg);
383
+ box-shadow: var(--shadow-modal);
384
+ max-width: 720px;
385
+ width: 90vw;
386
+ padding: 32px;
387
+ animation: vw-modal-in 200ms var(--ease-out);
388
+ }
389
+
390
+ @keyframes vw-modal-in {
391
+ from {
392
+ opacity: 0;
393
+ transform: translateY(8px);
394
+ }
395
+ to {
396
+ opacity: 1;
397
+ transform: translateY(0);
398
+ }
399
+ }
400
+
401
+ @media (prefers-reduced-motion: reduce) {
402
+ .vw-modal-backdrop {
403
+ animation: none;
404
+ }
405
+ .vw-modal {
406
+ animation: none;
407
+ }
408
+ }
409
+
410
+ .vw-modal__header {
411
+ display: flex;
412
+ align-items: flex-start;
413
+ justify-content: space-between;
414
+ margin-bottom: 24px;
415
+ }
416
+
417
+ .vw-modal__header-text {
418
+ min-width: 0;
419
+ }
420
+
421
+ .vw-modal__title {
422
+ font-size: 20px;
423
+ font-weight: 600;
424
+ color: var(--text-primary);
425
+ line-height: 1.3;
426
+ margin: 0;
427
+ }
428
+
429
+ .vw-modal__slug {
430
+ font-family: var(--font-mono);
431
+ font-size: 13px;
432
+ color: var(--text-secondary);
433
+ margin-top: 2px;
434
+ }
435
+
436
+ .vw-modal__close {
437
+ display: flex;
438
+ align-items: center;
439
+ justify-content: center;
440
+ width: 32px;
441
+ height: 32px;
442
+ border: none;
443
+ border-radius: var(--radius-sm);
444
+ background: transparent;
445
+ color: var(--text-secondary);
446
+ cursor: pointer;
447
+ flex-shrink: 0;
448
+ transition: background var(--duration) var(--ease-out), color var(--duration) var(--ease-out);
449
+ }
450
+
451
+ .vw-modal__close:hover {
452
+ background: var(--bg-surface-hover);
453
+ color: var(--text-primary);
454
+ }
455
+
456
+ .vw-modal__columns {
457
+ display: grid;
458
+ grid-template-columns: 1fr 1fr;
459
+ gap: 16px;
460
+ }
461
+
462
+ .vw-modal__column {
463
+ background: var(--bg-surface);
464
+ border: 1px solid var(--border-subtle);
465
+ border-radius: var(--radius-md);
466
+ padding: 20px;
467
+ }
468
+
469
+ .vw-modal__column-label {
470
+ font-size: 11px;
471
+ font-weight: 500;
472
+ letter-spacing: 0.05em;
473
+ text-transform: uppercase;
474
+ color: var(--text-secondary);
475
+ margin-bottom: 12px;
476
+ display: flex;
477
+ align-items: center;
478
+ gap: 8px;
479
+ }
480
+
481
+ .vw-modal__badge {
482
+ font-size: 10px;
483
+ font-weight: 500;
484
+ letter-spacing: 0.02em;
485
+ text-transform: uppercase;
486
+ color: var(--accent);
487
+ background: var(--accent-dim);
488
+ border-radius: var(--radius-sm);
489
+ padding: 2px 6px;
490
+ }
491
+
492
+ .vw-modal__desc {
493
+ font-size: 13px;
494
+ color: var(--text-secondary);
495
+ line-height: 1.5;
496
+ margin: 0 0 16px 0;
497
+ }
498
+
499
+ .vw-modal__code {
500
+ display: flex;
501
+ align-items: center;
502
+ gap: 8px;
503
+ background: var(--bg-surface-elevated);
504
+ border: 1px solid var(--border-subtle);
505
+ border-radius: var(--radius-sm);
506
+ padding: 12px;
507
+ margin-bottom: 12px;
508
+ }
509
+
510
+ .vw-modal__code code {
511
+ font-family: var(--font-mono);
512
+ font-size: 12px;
513
+ color: var(--text-primary);
514
+ flex: 1;
515
+ word-break: break-all;
516
+ }
517
+
518
+ .vw-modal__note {
519
+ font-size: 12px;
520
+ color: var(--text-tertiary);
521
+ line-height: 1.5;
522
+ margin: 0;
523
+ }
524
+
525
+ .vw-modal__tips {
526
+ list-style: none;
527
+ margin: 0;
528
+ padding: 0;
529
+ font-size: 13px;
530
+ color: var(--text-secondary);
531
+ line-height: 1.8;
532
+ }
533
+
534
+ .vw-modal__tips li::before {
535
+ content: "•";
536
+ margin-right: 8px;
537
+ color: var(--text-tertiary);
538
+ }
539
+
540
+ /* ---- Hide-HUD Tab ---- */
541
+
542
+ .vw-hide-hud-tab {
543
+ position: absolute;
544
+ top: -8px;
545
+ left: 50%;
546
+ transform: translateX(-50%);
547
+ width: 28px;
548
+ height: 8px;
549
+ border: none;
550
+ border-radius: var(--radius-sm) var(--radius-sm) 0 0;
551
+ background: var(--bg-surface-hover);
552
+ color: var(--text-tertiary);
553
+ cursor: pointer;
554
+ display: flex;
555
+ align-items: center;
556
+ justify-content: center;
557
+ padding: 0;
558
+ z-index: 10;
559
+ transition: background var(--duration) var(--ease-out), color var(--duration) var(--ease-out);
560
+ }
561
+
562
+ .vw-hide-hud-tab:hover {
563
+ background: var(--border-strong);
564
+ color: var(--text-secondary);
565
+ }
566
+
567
+ .vw-hide-hud-tab svg {
568
+ width: 10px;
569
+ height: 10px;
570
+ }
571
+
572
+ /* Focus styles for modal and tab */
573
+ .vw-modal__close:focus-visible,
574
+ .vw-hide-hud-tab:focus-visible {
575
+ outline: 2px solid var(--accent);
576
+ outline-offset: 2px;
577
+ }
578
+
579
+ /* ---- Responsive ---- */
580
+
581
+ @media (max-width: 640px) {
582
+ .vw-top-bar {
583
+ padding: 0 16px;
584
+ }
585
+
586
+ .vw-top-bar__project {
587
+ display: none;
588
+ }
589
+
590
+ .vw-homepage__content {
591
+ padding: 24px 16px;
592
+ }
593
+
594
+ .vw-homepage__grid {
595
+ grid-template-columns: 1fr;
596
+ }
597
+
598
+ .vw-modal__columns {
599
+ grid-template-columns: 1fr;
600
+ }
601
+
602
+ .vw-modal {
603
+ padding: 24px;
604
+ }
605
+ }
@@ -0,0 +1,44 @@
1
+ /**
2
+ * Design tokens for the Videowright dev UI.
3
+ * Dark mode only. See ui_design.md for rationale.
4
+ */
5
+
6
+ :root {
7
+ /* Backgrounds */
8
+ --bg-base: #0a0a0b;
9
+ --bg-surface: #131316;
10
+ --bg-surface-hover: #1a1a1f;
11
+ --bg-surface-elevated: #1d1d22;
12
+
13
+ /* Borders */
14
+ --border-subtle: #26262d;
15
+ --border-strong: #3a3a44;
16
+
17
+ /* Text */
18
+ --text-primary: #f5f5f7;
19
+ --text-secondary: #a0a0a8;
20
+ --text-tertiary: #6e6e78;
21
+
22
+ /* Accent */
23
+ --accent: #a78bfa;
24
+ --accent-dim: #7c3aed33;
25
+
26
+ /* Semantic */
27
+ --success: #34d399;
28
+ --error: #f87171;
29
+
30
+ /* Radius */
31
+ --radius-sm: 6px;
32
+ --radius-md: 10px;
33
+ --radius-lg: 14px;
34
+
35
+ /* Shadow */
36
+ --shadow-modal: 0 24px 48px rgba(0, 0, 0, 0.4), 0 0 0 1px var(--border-subtle);
37
+
38
+ /* Typography */
39
+ --font-mono: "JetBrains Mono", ui-monospace, "SF Mono", Menlo, Consolas, monospace;
40
+
41
+ /* Motion */
42
+ --ease-out: cubic-bezier(0.16, 1, 0.3, 1);
43
+ --duration: 150ms;
44
+ }
@@ -0,0 +1,22 @@
1
+ <!DOCTYPE html>
2
+ <html lang="en">
3
+ <head>
4
+ <meta charset="UTF-8" />
5
+ <meta name="viewport" content="width=device-width, initial-scale=1.0" />
6
+ <title>Videowright Dev</title>
7
+ <style>
8
+ /* Inline critical styles for HUD relocation (used by video_view.ts) */
9
+ #dev-hud-container .vw-hud { position: relative; inset: auto; }
10
+ #dev-hud-container .vw-hud-inner { position: relative; }
11
+ #dev-hud-container .vw-hud-ended {
12
+ position: relative; top: auto; right: auto;
13
+ display: inline-block; margin-bottom: 4px;
14
+ }
15
+ #dev-hud-container .vw-hud-error-overlay { position: relative; inset: auto; }
16
+ </style>
17
+ </head>
18
+ <body>
19
+ <div id="app"></div>
20
+ <script type="module" src="/entry_video.ts"></script>
21
+ </body>
22
+ </html>
@@ -0,0 +1,66 @@
1
+ /**
2
+ * Homepage view.
3
+ * Shows a grid of video cards, or the empty-state panel when no videos exist.
4
+ */
5
+
6
+ import type { ProjectInfo } from "../../../types.js";
7
+ import { renderDownloadModal } from "../components/download_modal.js";
8
+ import { renderEmptyState } from "../components/empty_state.js";
9
+ import { renderTopBar } from "../components/top_bar.js";
10
+ import { renderVideoCard } from "../components/video_card.js";
11
+
12
+ export function renderHomepage(projectInfo: ProjectInfo): HTMLElement {
13
+ const container = document.createElement("main");
14
+ container.className = "vw-homepage";
15
+
16
+ // Top bar
17
+ const topBar = renderTopBar({ projectName: projectInfo.projectName });
18
+ container.appendChild(topBar);
19
+
20
+ // Empty state
21
+ if (projectInfo.videos.length === 0) {
22
+ container.appendChild(renderEmptyState());
23
+ return container;
24
+ }
25
+
26
+ // Content area
27
+ const content = document.createElement("div");
28
+ content.className = "vw-homepage__content";
29
+
30
+ // Section heading
31
+ const heading = document.createElement("h1");
32
+ heading.className = "vw-homepage__heading";
33
+ heading.textContent = "Videos";
34
+ content.appendChild(heading);
35
+
36
+ const subtitle = document.createElement("p");
37
+ subtitle.className = "vw-homepage__subtitle";
38
+ const count = projectInfo.videos.length;
39
+ subtitle.textContent = `${count} ${count === 1 ? "video" : "videos"} in this project`;
40
+ content.appendChild(subtitle);
41
+
42
+ // Card grid
43
+ const grid = document.createElement("div");
44
+ grid.className = "vw-homepage__grid";
45
+
46
+ for (const video of projectInfo.videos) {
47
+ const card = renderVideoCard({
48
+ slug: video.slug,
49
+ title: video.title,
50
+ style: video.style,
51
+ onDownload: () => {
52
+ renderDownloadModal({
53
+ slug: video.slug,
54
+ title: video.title,
55
+ onClose: () => {},
56
+ });
57
+ },
58
+ });
59
+ grid.appendChild(card);
60
+ }
61
+
62
+ content.appendChild(grid);
63
+ container.appendChild(content);
64
+
65
+ return container;
66
+ }