markupr 2.1.8

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 (299) hide show
  1. package/.claude/commands/review-feedback.md +47 -0
  2. package/.eslintrc.json +35 -0
  3. package/.github/CODEOWNERS +16 -0
  4. package/.github/FUNDING.yml +1 -0
  5. package/.github/ISSUE_TEMPLATE/bug_report.md +56 -0
  6. package/.github/ISSUE_TEMPLATE/feature_request.md +54 -0
  7. package/.github/PULL_REQUEST_TEMPLATE.md +89 -0
  8. package/.github/dependabot.yml +70 -0
  9. package/.github/workflows/ci.yml +184 -0
  10. package/.github/workflows/deploy-landing.yml +134 -0
  11. package/.github/workflows/nightly.yml +288 -0
  12. package/.github/workflows/release.yml +318 -0
  13. package/CHANGELOG.md +127 -0
  14. package/CLAUDE.md +137 -0
  15. package/CODE_OF_CONDUCT.md +9 -0
  16. package/CONTRIBUTING.md +390 -0
  17. package/LICENSE +21 -0
  18. package/PRODUCT_VISION.md +277 -0
  19. package/README.md +517 -0
  20. package/SECURITY.md +51 -0
  21. package/SIGNING_INSTRUCTIONS.md +284 -0
  22. package/assets/DMG_BACKGROUND_INSTRUCTIONS.md +130 -0
  23. package/assets/svg-source/dmg-background.svg +70 -0
  24. package/assets/svg-source/icon.svg +20 -0
  25. package/assets/svg-source/tray-icon-processing.svg +7 -0
  26. package/assets/svg-source/tray-icon-recording.svg +7 -0
  27. package/assets/svg-source/tray-icon.svg +6 -0
  28. package/assets/tray-complete.png +0 -0
  29. package/assets/tray-complete@2x.png +0 -0
  30. package/assets/tray-completeTemplate.png +0 -0
  31. package/assets/tray-completeTemplate@2x.png +0 -0
  32. package/assets/tray-error.png +0 -0
  33. package/assets/tray-error@2x.png +0 -0
  34. package/assets/tray-errorTemplate.png +0 -0
  35. package/assets/tray-errorTemplate@2x.png +0 -0
  36. package/assets/tray-icon-processing.png +0 -0
  37. package/assets/tray-icon-processing@2x.png +0 -0
  38. package/assets/tray-icon-processingTemplate.png +0 -0
  39. package/assets/tray-icon-processingTemplate@2x.png +0 -0
  40. package/assets/tray-icon-recording.png +0 -0
  41. package/assets/tray-icon-recording@2x.png +0 -0
  42. package/assets/tray-icon-recordingTemplate.png +0 -0
  43. package/assets/tray-icon-recordingTemplate@2x.png +0 -0
  44. package/assets/tray-icon.png +0 -0
  45. package/assets/tray-icon@2x.png +0 -0
  46. package/assets/tray-iconTemplate.png +0 -0
  47. package/assets/tray-iconTemplate@2x.png +0 -0
  48. package/assets/tray-idle.png +0 -0
  49. package/assets/tray-idle@2x.png +0 -0
  50. package/assets/tray-idleTemplate.png +0 -0
  51. package/assets/tray-idleTemplate@2x.png +0 -0
  52. package/assets/tray-processing-0.png +0 -0
  53. package/assets/tray-processing-0@2x.png +0 -0
  54. package/assets/tray-processing-0Template.png +0 -0
  55. package/assets/tray-processing-0Template@2x.png +0 -0
  56. package/assets/tray-processing-1.png +0 -0
  57. package/assets/tray-processing-1@2x.png +0 -0
  58. package/assets/tray-processing-1Template.png +0 -0
  59. package/assets/tray-processing-1Template@2x.png +0 -0
  60. package/assets/tray-processing-2.png +0 -0
  61. package/assets/tray-processing-2@2x.png +0 -0
  62. package/assets/tray-processing-2Template.png +0 -0
  63. package/assets/tray-processing-2Template@2x.png +0 -0
  64. package/assets/tray-processing-3.png +0 -0
  65. package/assets/tray-processing-3@2x.png +0 -0
  66. package/assets/tray-processing-3Template.png +0 -0
  67. package/assets/tray-processing-3Template@2x.png +0 -0
  68. package/assets/tray-processing.png +0 -0
  69. package/assets/tray-processing@2x.png +0 -0
  70. package/assets/tray-processingTemplate.png +0 -0
  71. package/assets/tray-processingTemplate@2x.png +0 -0
  72. package/assets/tray-recording.png +0 -0
  73. package/assets/tray-recording@2x.png +0 -0
  74. package/assets/tray-recordingTemplate.png +0 -0
  75. package/assets/tray-recordingTemplate@2x.png +0 -0
  76. package/build/DMG_BACKGROUND_SPEC.md +50 -0
  77. package/build/dmg-background.png +0 -0
  78. package/build/dmg-background@2x.png +0 -0
  79. package/build/entitlements.mac.inherit.plist +27 -0
  80. package/build/entitlements.mac.plist +41 -0
  81. package/build/favicon-16.png +0 -0
  82. package/build/favicon-180.png +0 -0
  83. package/build/favicon-192.png +0 -0
  84. package/build/favicon-32.png +0 -0
  85. package/build/favicon-48.png +0 -0
  86. package/build/favicon-512.png +0 -0
  87. package/build/favicon-64.png +0 -0
  88. package/build/icon-128.png +0 -0
  89. package/build/icon-16.png +0 -0
  90. package/build/icon-24.png +0 -0
  91. package/build/icon-256.png +0 -0
  92. package/build/icon-32.png +0 -0
  93. package/build/icon-48.png +0 -0
  94. package/build/icon-64.png +0 -0
  95. package/build/icon.icns +0 -0
  96. package/build/icon.ico +0 -0
  97. package/build/icon.iconset/icon_128x128.png +0 -0
  98. package/build/icon.iconset/icon_128x128@2x.png +0 -0
  99. package/build/icon.iconset/icon_16x16.png +0 -0
  100. package/build/icon.iconset/icon_16x16@2x.png +0 -0
  101. package/build/icon.iconset/icon_256x256.png +0 -0
  102. package/build/icon.iconset/icon_256x256@2x.png +0 -0
  103. package/build/icon.iconset/icon_32x32.png +0 -0
  104. package/build/icon.iconset/icon_32x32@2x.png +0 -0
  105. package/build/icon.iconset/icon_512x512.png +0 -0
  106. package/build/icon.iconset/icon_512x512@2x.png +0 -0
  107. package/build/icon.png +0 -0
  108. package/build/installer-header.bmp +0 -0
  109. package/build/installer-header.png +0 -0
  110. package/build/installer-sidebar.bmp +0 -0
  111. package/build/installer-sidebar.png +0 -0
  112. package/build/installer.nsh +45 -0
  113. package/build/overlay-processing.png +0 -0
  114. package/build/overlay-recording.png +0 -0
  115. package/build/toolbar-record.png +0 -0
  116. package/build/toolbar-screenshot.png +0 -0
  117. package/build/toolbar-settings.png +0 -0
  118. package/build/toolbar-stop.png +0 -0
  119. package/dist/main/index.mjs +12612 -0
  120. package/dist/preload/index.mjs +907 -0
  121. package/dist/renderer/assets/index-CCmUjl9K.js +19495 -0
  122. package/dist/renderer/assets/index-CUqz_Gs6.css +2270 -0
  123. package/dist/renderer/index.html +27 -0
  124. package/docs/AI_AGENT_QUICKSTART.md +42 -0
  125. package/docs/AI_PIPELINE_DESIGN.md +595 -0
  126. package/docs/API.md +514 -0
  127. package/docs/ARCHITECTURE.md +460 -0
  128. package/docs/CONFIGURATION.md +336 -0
  129. package/docs/DEVELOPMENT.md +508 -0
  130. package/docs/EXPORT_FORMATS.md +451 -0
  131. package/docs/GETTING_STARTED.md +236 -0
  132. package/docs/KEYBOARD_SHORTCUTS.md +334 -0
  133. package/docs/TROUBLESHOOTING.md +418 -0
  134. package/docs/landing/index.html +672 -0
  135. package/docs/landing/script.js +342 -0
  136. package/docs/landing/styles.css +1543 -0
  137. package/electron-builder.yml +140 -0
  138. package/electron.vite.config.ts +63 -0
  139. package/package.json +108 -0
  140. package/railway.json +12 -0
  141. package/scripts/build.mjs +51 -0
  142. package/scripts/generate-icons.mjs +314 -0
  143. package/scripts/generate-installer-images.cjs +253 -0
  144. package/scripts/generate-tray-icons.mjs +258 -0
  145. package/scripts/notarize.cjs +180 -0
  146. package/scripts/one-click-clean-test.sh +147 -0
  147. package/scripts/postinstall.mjs +36 -0
  148. package/scripts/setup-markupr.sh +55 -0
  149. package/setup +17 -0
  150. package/site/index.html +1835 -0
  151. package/site/package.json +11 -0
  152. package/site/railway.json +12 -0
  153. package/site/server.js +31 -0
  154. package/src/main/AutoUpdater.ts +392 -0
  155. package/src/main/CrashRecovery.ts +655 -0
  156. package/src/main/ErrorHandler.ts +703 -0
  157. package/src/main/HotkeyManager.ts +399 -0
  158. package/src/main/MenuManager.ts +529 -0
  159. package/src/main/PermissionManager.ts +420 -0
  160. package/src/main/SessionController.ts +1465 -0
  161. package/src/main/TrayManager.ts +540 -0
  162. package/src/main/ai/AIPipelineManager.ts +199 -0
  163. package/src/main/ai/ClaudeAnalyzer.ts +339 -0
  164. package/src/main/ai/ImageOptimizer.ts +176 -0
  165. package/src/main/ai/StructuredMarkdownBuilder.ts +379 -0
  166. package/src/main/ai/index.ts +16 -0
  167. package/src/main/ai/types.ts +258 -0
  168. package/src/main/analysis/ClarificationGenerator.ts +385 -0
  169. package/src/main/analysis/FeedbackAnalyzer.ts +531 -0
  170. package/src/main/analysis/index.ts +19 -0
  171. package/src/main/audio/AudioCapture.ts +978 -0
  172. package/src/main/audio/audioUtils.ts +100 -0
  173. package/src/main/audio/index.ts +20 -0
  174. package/src/main/capture/index.ts +1 -0
  175. package/src/main/index.ts +1693 -0
  176. package/src/main/ipc/captureHandlers.ts +272 -0
  177. package/src/main/ipc/index.ts +45 -0
  178. package/src/main/ipc/outputHandlers.ts +302 -0
  179. package/src/main/ipc/sessionHandlers.ts +56 -0
  180. package/src/main/ipc/settingsHandlers.ts +471 -0
  181. package/src/main/ipc/types.ts +56 -0
  182. package/src/main/ipc/windowHandlers.ts +277 -0
  183. package/src/main/output/ClipboardService.ts +369 -0
  184. package/src/main/output/ExportService.ts +539 -0
  185. package/src/main/output/FileManager.ts +416 -0
  186. package/src/main/output/MarkdownGenerator.ts +791 -0
  187. package/src/main/output/MarkdownPatcher.ts +299 -0
  188. package/src/main/output/index.ts +186 -0
  189. package/src/main/output/sessionAdapter.ts +207 -0
  190. package/src/main/output/templates/html-template.ts +553 -0
  191. package/src/main/pipeline/FrameExtractor.ts +330 -0
  192. package/src/main/pipeline/PostProcessor.ts +399 -0
  193. package/src/main/pipeline/TranscriptAnalyzer.ts +226 -0
  194. package/src/main/pipeline/index.ts +36 -0
  195. package/src/main/platform/WindowsTaskbar.ts +600 -0
  196. package/src/main/platform/index.ts +16 -0
  197. package/src/main/settings/SettingsManager.ts +730 -0
  198. package/src/main/settings/index.ts +19 -0
  199. package/src/main/transcription/ModelDownloadManager.ts +494 -0
  200. package/src/main/transcription/TierManager.ts +219 -0
  201. package/src/main/transcription/TranscriptionRecoveryService.ts +340 -0
  202. package/src/main/transcription/WhisperService.ts +748 -0
  203. package/src/main/transcription/index.ts +56 -0
  204. package/src/main/transcription/types.ts +135 -0
  205. package/src/main/windows/PopoverManager.ts +284 -0
  206. package/src/main/windows/TaskbarIntegration.ts +452 -0
  207. package/src/main/windows/index.ts +23 -0
  208. package/src/preload/index.ts +1047 -0
  209. package/src/renderer/App.tsx +515 -0
  210. package/src/renderer/AppWrapper.tsx +28 -0
  211. package/src/renderer/assets/logo-dark.svg +7 -0
  212. package/src/renderer/assets/logo.svg +7 -0
  213. package/src/renderer/audio/AudioCaptureRenderer.ts +454 -0
  214. package/src/renderer/capture/ScreenRecordingRenderer.ts +492 -0
  215. package/src/renderer/components/AnnotationOverlay.tsx +836 -0
  216. package/src/renderer/components/AudioWaveform.tsx +811 -0
  217. package/src/renderer/components/ClarificationQuestions.tsx +656 -0
  218. package/src/renderer/components/CountdownTimer.tsx +495 -0
  219. package/src/renderer/components/CrashRecoveryDialog.tsx +632 -0
  220. package/src/renderer/components/DonateButton.tsx +127 -0
  221. package/src/renderer/components/ErrorBoundary.tsx +308 -0
  222. package/src/renderer/components/ExportDialog.tsx +872 -0
  223. package/src/renderer/components/HotkeyHint.tsx +261 -0
  224. package/src/renderer/components/KeyboardShortcuts.tsx +787 -0
  225. package/src/renderer/components/ModelDownloadDialog.tsx +844 -0
  226. package/src/renderer/components/Onboarding.tsx +1830 -0
  227. package/src/renderer/components/ProcessingOverlay.tsx +157 -0
  228. package/src/renderer/components/RecordingOverlay.tsx +423 -0
  229. package/src/renderer/components/SessionHistory.tsx +1746 -0
  230. package/src/renderer/components/SessionReview.tsx +1321 -0
  231. package/src/renderer/components/SettingsPanel.tsx +217 -0
  232. package/src/renderer/components/Skeleton.tsx +347 -0
  233. package/src/renderer/components/StatusIndicator.tsx +86 -0
  234. package/src/renderer/components/ThemeProvider.tsx +429 -0
  235. package/src/renderer/components/Tooltip.tsx +370 -0
  236. package/src/renderer/components/TranscriptionPreview.tsx +183 -0
  237. package/src/renderer/components/TranscriptionTierSelector.tsx +640 -0
  238. package/src/renderer/components/UpdateNotification.tsx +377 -0
  239. package/src/renderer/components/WindowSelector.tsx +947 -0
  240. package/src/renderer/components/index.ts +99 -0
  241. package/src/renderer/components/primitives/ApiKeyInput.tsx +98 -0
  242. package/src/renderer/components/primitives/ColorPicker.tsx +65 -0
  243. package/src/renderer/components/primitives/DangerButton.tsx +45 -0
  244. package/src/renderer/components/primitives/DirectoryPicker.tsx +41 -0
  245. package/src/renderer/components/primitives/Dropdown.tsx +34 -0
  246. package/src/renderer/components/primitives/KeyRecorder.tsx +117 -0
  247. package/src/renderer/components/primitives/SettingsSection.tsx +32 -0
  248. package/src/renderer/components/primitives/Slider.tsx +43 -0
  249. package/src/renderer/components/primitives/Toggle.tsx +36 -0
  250. package/src/renderer/components/primitives/index.ts +10 -0
  251. package/src/renderer/components/settings/AdvancedTab.tsx +174 -0
  252. package/src/renderer/components/settings/AppearanceTab.tsx +77 -0
  253. package/src/renderer/components/settings/GeneralTab.tsx +40 -0
  254. package/src/renderer/components/settings/HotkeysTab.tsx +79 -0
  255. package/src/renderer/components/settings/RecordingTab.tsx +84 -0
  256. package/src/renderer/components/settings/index.ts +9 -0
  257. package/src/renderer/components/settings/settingsStyles.ts +673 -0
  258. package/src/renderer/components/settings/tabConfig.tsx +85 -0
  259. package/src/renderer/components/settings/useSettingsPanel.ts +447 -0
  260. package/src/renderer/contexts/ProcessingContext.tsx +227 -0
  261. package/src/renderer/contexts/RecordingContext.tsx +683 -0
  262. package/src/renderer/contexts/UIContext.tsx +326 -0
  263. package/src/renderer/contexts/index.ts +24 -0
  264. package/src/renderer/donateMessages.ts +69 -0
  265. package/src/renderer/hooks/index.ts +75 -0
  266. package/src/renderer/hooks/useAnimation.tsx +544 -0
  267. package/src/renderer/hooks/useTheme.ts +313 -0
  268. package/src/renderer/index.html +26 -0
  269. package/src/renderer/main.tsx +52 -0
  270. package/src/renderer/styles/animations.css +1093 -0
  271. package/src/renderer/styles/app-shell.css +662 -0
  272. package/src/renderer/styles/globals.css +515 -0
  273. package/src/renderer/styles/theme.ts +578 -0
  274. package/src/renderer/types/electron.d.ts +385 -0
  275. package/src/shared/hotkeys.ts +283 -0
  276. package/src/shared/types.ts +809 -0
  277. package/tests/clipboard.test.ts +228 -0
  278. package/tests/e2e/criticalPaths.test.ts +594 -0
  279. package/tests/feedbackAnalyzer.test.ts +303 -0
  280. package/tests/integration/sessionFlow.test.ts +583 -0
  281. package/tests/markdownGenerator.test.ts +418 -0
  282. package/tests/output.test.ts +96 -0
  283. package/tests/setup.ts +486 -0
  284. package/tests/unit/appIntegration.test.ts +676 -0
  285. package/tests/unit/appViewState.test.ts +281 -0
  286. package/tests/unit/audioIpcChannels.test.ts +17 -0
  287. package/tests/unit/exportService.test.ts +492 -0
  288. package/tests/unit/hotkeys.test.ts +92 -0
  289. package/tests/unit/navigationPreload.test.ts +94 -0
  290. package/tests/unit/onboardingFlow.test.ts +345 -0
  291. package/tests/unit/permissionManager.test.ts +175 -0
  292. package/tests/unit/permissionManagerExpanded.test.ts +296 -0
  293. package/tests/unit/screenRecordingRenderer.test.ts +368 -0
  294. package/tests/unit/sessionController.test.ts +515 -0
  295. package/tests/unit/tierManager.test.ts +61 -0
  296. package/tests/unit/tierManagerExpanded.test.ts +142 -0
  297. package/tests/unit/transcriptAnalyzer.test.ts +64 -0
  298. package/tsconfig.json +25 -0
  299. package/vitest.config.ts +46 -0
@@ -0,0 +1,385 @@
1
+ /**
2
+ * markupr - Electron Type Declarations for Renderer
3
+ *
4
+ * This file provides TypeScript type support for the window.markupr API
5
+ * exposed by the preload script via contextBridge.
6
+ */
7
+
8
+ import type {
9
+ AppSettings,
10
+ CaptureSource,
11
+ AudioDevice,
12
+ PermissionType,
13
+ PermissionStatus,
14
+ SessionStatusPayload,
15
+ SessionPayload,
16
+ FeedbackItemPayload,
17
+ TranscriptChunkPayload,
18
+ ScreenshotCapturedPayload,
19
+ OutputReadyPayload,
20
+ SaveResult,
21
+ HotkeyConfig,
22
+ SessionState,
23
+ TranscriptionTier,
24
+ TranscriptionTierStatus,
25
+ UpdateStatusPayload,
26
+ WhisperDownloadProgressPayload,
27
+ WhisperModelInfoPayload,
28
+ WhisperModelCheckResult,
29
+ ApiKeyValidationResult,
30
+ ProcessingProgressPayload,
31
+ } from '../../shared/types';
32
+
33
+ type Unsubscribe = () => void;
34
+
35
+ /**
36
+ * Session API
37
+ */
38
+ interface SessionAPI {
39
+ start: (
40
+ sourceId?: string,
41
+ sourceName?: string
42
+ ) => Promise<{ success: boolean; sessionId?: string; error?: string }>;
43
+ stop: () => Promise<{ success: boolean; session?: SessionPayload; error?: string }>;
44
+ pause: () => Promise<{ success: boolean; error?: string }>;
45
+ resume: () => Promise<{ success: boolean; error?: string }>;
46
+ cancel: () => Promise<{ success: boolean }>;
47
+ getStatus: () => Promise<SessionStatusPayload>;
48
+ getCurrent: () => Promise<SessionPayload | null>;
49
+ onStateChange: (callback: (data: { state: SessionState; session: SessionPayload | null }) => void) => Unsubscribe;
50
+ onStatusUpdate: (callback: (data: SessionStatusPayload) => void) => Unsubscribe;
51
+ onComplete: (callback: (data: SessionPayload) => void) => Unsubscribe;
52
+ onFeedbackItem: (callback: (data: FeedbackItemPayload) => void) => Unsubscribe;
53
+ onVoiceActivity: (callback: (data: { active: boolean }) => void) => Unsubscribe;
54
+ onError: (callback: (data: { message: string }) => void) => Unsubscribe;
55
+ }
56
+
57
+ /**
58
+ * Capture API
59
+ */
60
+ interface CaptureAPI {
61
+ getSources: () => Promise<CaptureSource[]>;
62
+ manualScreenshot: () => Promise<{ success: boolean; error?: string }>;
63
+ onScreenshot: (callback: (data: ScreenshotCapturedPayload) => void) => Unsubscribe;
64
+ onManualTrigger: (callback: (data: { timestamp: number }) => void) => Unsubscribe;
65
+ }
66
+
67
+ /**
68
+ * Audio API
69
+ */
70
+ interface AudioAPI {
71
+ getDevices: () => Promise<AudioDevice[]>;
72
+ setDevice: (deviceId: string) => Promise<{ success: boolean }>;
73
+ onLevel: (callback: (level: number) => void) => Unsubscribe;
74
+ onVoiceActivity: (callback: (active: boolean) => void) => Unsubscribe;
75
+
76
+ // Audio capture bridge
77
+ onRequestDevices: (callback: () => void) => Unsubscribe;
78
+ sendDevices: (devices: AudioDevice[]) => void;
79
+ onStartCapture: (callback: (config: {
80
+ deviceId: string | null;
81
+ sampleRate: number;
82
+ channels: number;
83
+ chunkDurationMs: number;
84
+ }) => void) => Unsubscribe;
85
+ onStopCapture: (callback: () => void) => Unsubscribe;
86
+ onSetDevice: (callback: (deviceId: string) => void) => Unsubscribe;
87
+ sendAudioChunk: (data: {
88
+ timestamp: number;
89
+ duration: number;
90
+ samples?: number[];
91
+ encodedChunk?: Uint8Array;
92
+ mimeType?: string;
93
+ audioLevel?: number;
94
+ rms?: number;
95
+ }) => void;
96
+ notifyCaptureStarted: () => void;
97
+ notifyCaptureStopped: () => void;
98
+ sendCaptureError: (error: string) => void;
99
+ }
100
+
101
+ /**
102
+ * Screen recording persistence API
103
+ */
104
+ interface ScreenRecordingAPI {
105
+ start: (
106
+ sessionId: string,
107
+ mimeType: string,
108
+ startTime?: number
109
+ ) => Promise<{ success: boolean; path?: string; error?: string }>;
110
+ appendChunk: (sessionId: string, chunk: Uint8Array) => Promise<{ success: boolean; error?: string }>;
111
+ stop: (
112
+ sessionId: string
113
+ ) => Promise<{ success: boolean; path?: string; bytes?: number; mimeType?: string; error?: string }>;
114
+ }
115
+
116
+ /**
117
+ * Transcript API
118
+ */
119
+ interface TranscriptAPI {
120
+ onChunk: (callback: (data: TranscriptChunkPayload) => void) => Unsubscribe;
121
+ onFinal: (callback: (data: { text: string; confidence: number; timestamp: number }) => void) => Unsubscribe;
122
+ }
123
+
124
+ /**
125
+ * Transcription tier and model control API
126
+ */
127
+ interface TranscriptionControlAPI {
128
+ getTierStatuses: () => Promise<TranscriptionTierStatus[]>;
129
+ getCurrentTier: () => Promise<TranscriptionTier | null>;
130
+ setTier: (tier: TranscriptionTier) => Promise<{ success: boolean; error?: string }>;
131
+ downloadModel: (model: string) => Promise<{ success: boolean; error?: string }>;
132
+ cancelDownload: (model: string) => Promise<{ success: boolean }>;
133
+ onModelProgress: (callback: (data: WhisperDownloadProgressPayload) => void) => Unsubscribe;
134
+ }
135
+
136
+ /**
137
+ * Processing Pipeline API (post-recording progress events)
138
+ */
139
+ interface ProcessingAPI {
140
+ onProgress: (callback: (data: ProcessingProgressPayload) => void) => Unsubscribe;
141
+ onComplete: (callback: (data: OutputReadyPayload) => void) => Unsubscribe;
142
+ }
143
+
144
+ /**
145
+ * Settings API
146
+ */
147
+ interface SettingsAPI {
148
+ get: <K extends keyof AppSettings>(key: K) => Promise<AppSettings[K]>;
149
+ getAll: () => Promise<AppSettings>;
150
+ set: <K extends keyof AppSettings>(key: K, value: AppSettings[K]) => Promise<AppSettings>;
151
+ getApiKey: (service: string) => Promise<string | null>;
152
+ setApiKey: (service: string, key: string) => Promise<boolean>;
153
+ deleteApiKey: (service: string) => Promise<boolean>;
154
+ hasApiKey: (service: string) => Promise<boolean>;
155
+ testApiKey: (service: 'openai' | 'anthropic', key: string) => Promise<ApiKeyValidationResult>;
156
+ selectDirectory: () => Promise<string | null>;
157
+ clearAllData: () => Promise<void>;
158
+ export: () => Promise<void>;
159
+ import: () => Promise<AppSettings | null>;
160
+ }
161
+
162
+ /**
163
+ * Hotkey API
164
+ */
165
+ interface HotkeyAPI {
166
+ getConfig: () => Promise<HotkeyConfig>;
167
+ updateConfig: (config: Partial<HotkeyConfig>) => Promise<{ config: HotkeyConfig; results: unknown[] }>;
168
+ onTriggered: (callback: (data: { action: string; accelerator: string }) => void) => Unsubscribe;
169
+ }
170
+
171
+ /**
172
+ * Permissions API
173
+ */
174
+ interface PermissionsAPI {
175
+ check: (type: PermissionType) => Promise<boolean>;
176
+ request: (type: PermissionType) => Promise<boolean>;
177
+ getAll: () => Promise<PermissionStatus>;
178
+ }
179
+
180
+ /**
181
+ * Session history metadata for display
182
+ */
183
+ interface SessionHistoryItem {
184
+ id: string;
185
+ startTime: number;
186
+ endTime: number;
187
+ itemCount: number;
188
+ screenshotCount: number;
189
+ sourceName: string;
190
+ firstThumbnail?: string;
191
+ folder: string;
192
+ transcriptionPreview?: string;
193
+ }
194
+
195
+ /**
196
+ * Output API
197
+ */
198
+ interface OutputAPI {
199
+ save: (session?: SessionPayload) => Promise<SaveResult>;
200
+ copyClipboard: () => Promise<boolean>;
201
+ openFolder: (sessionDir?: string) => Promise<{ success: boolean; error?: string }>;
202
+ onReady: (callback: (data: OutputReadyPayload) => void) => Unsubscribe;
203
+ onError: (callback: (data: { message: string }) => void) => Unsubscribe;
204
+
205
+ // Session History Browser API
206
+ listSessions: () => Promise<SessionHistoryItem[]>;
207
+ getSessionMetadata: (sessionId: string) => Promise<SessionHistoryItem | null>;
208
+ deleteSession: (sessionId: string) => Promise<{ success: boolean; error?: string }>;
209
+ deleteSessions: (sessionIds: string[]) => Promise<{ success: boolean; deleted: string[]; failed: string[] }>;
210
+ exportSession: (sessionId: string, format?: 'markdown' | 'json' | 'pdf') => Promise<{ success: boolean; path?: string; error?: string }>;
211
+ exportSessions: (sessionIds: string[], format?: 'markdown' | 'json' | 'pdf') => Promise<{ success: boolean; path?: string; error?: string }>;
212
+ }
213
+
214
+ /**
215
+ * Complete markupr API
216
+ */
217
+ /**
218
+ * Navigation API (Main -> Renderer navigation events)
219
+ */
220
+ interface NavigationAPI {
221
+ onShowSettings: (callback: () => void) => Unsubscribe;
222
+ onShowHistory: (callback: () => void) => Unsubscribe;
223
+ onShowShortcuts: (callback: () => void) => Unsubscribe;
224
+ onShowOnboarding: (callback: () => void) => Unsubscribe;
225
+ onShowExport: (callback: () => void) => Unsubscribe;
226
+ onShowWindowSelector: (callback: () => void) => Unsubscribe;
227
+ }
228
+
229
+ /**
230
+ * Window control API
231
+ */
232
+ interface WindowAPI {
233
+ minimize: () => Promise<{ success: boolean }>;
234
+ hide: () => Promise<{ success: boolean }>;
235
+ close: () => Promise<{ success: boolean }>;
236
+ }
237
+
238
+ /**
239
+ * Popover control API
240
+ */
241
+ interface PopoverAPI {
242
+ resize: (width: number, height: number) => Promise<{ success: boolean }>;
243
+ resizeToState: (state: string) => Promise<{ success: boolean }>;
244
+ }
245
+
246
+ /**
247
+ * Crash Recovery API
248
+ */
249
+ interface CrashRecoveryAPI {
250
+ check: () => Promise<{
251
+ hasIncomplete: boolean;
252
+ session: {
253
+ id: string;
254
+ startTime: number;
255
+ lastSaveTime: number;
256
+ feedbackItems: Array<{
257
+ id: string;
258
+ timestamp: number;
259
+ text: string;
260
+ confidence: number;
261
+ hasScreenshot: boolean;
262
+ screenshotId?: string;
263
+ }>;
264
+ sourceName: string;
265
+ screenshotCount: number;
266
+ metadata?: { appVersion: string; platform: string; sessionDurationMs: number };
267
+ } | null;
268
+ }>;
269
+ recover: (sessionId: string) => Promise<{
270
+ success: boolean;
271
+ session?: {
272
+ id: string;
273
+ feedbackItems: Array<{
274
+ id: string;
275
+ timestamp: number;
276
+ text: string;
277
+ confidence: number;
278
+ hasScreenshot: boolean;
279
+ }>;
280
+ };
281
+ error?: string;
282
+ }>;
283
+ discard: () => Promise<{ success: boolean }>;
284
+ getLogs: (limit?: number) => Promise<Array<{
285
+ timestamp: string;
286
+ error: { name: string; message: string; stack?: string };
287
+ appVersion: string;
288
+ platform: string;
289
+ sessionId?: string;
290
+ }>>;
291
+ clearLogs: () => Promise<{ success: boolean }>;
292
+ updateSettings: (settings: {
293
+ enableAutoSave?: boolean;
294
+ autoSaveIntervalMs?: number;
295
+ enableCrashReporting?: boolean;
296
+ maxCrashLogs?: number;
297
+ }) => Promise<{ success: boolean }>;
298
+ onIncompleteFound: (callback: (data: {
299
+ session: {
300
+ id: string;
301
+ startTime: number;
302
+ lastSaveTime: number;
303
+ feedbackItems: Array<{
304
+ id: string;
305
+ timestamp: number;
306
+ text: string;
307
+ confidence: number;
308
+ hasScreenshot: boolean;
309
+ }>;
310
+ sourceName: string;
311
+ screenshotCount: number;
312
+ };
313
+ }) => void) => Unsubscribe;
314
+ }
315
+
316
+ /**
317
+ * Updates API
318
+ */
319
+ interface UpdatesAPI {
320
+ check: () => Promise<unknown>;
321
+ download: () => Promise<void>;
322
+ install: () => Promise<void>;
323
+ onStatus: (callback: (status: UpdateStatusPayload) => void) => Unsubscribe;
324
+ }
325
+
326
+ /**
327
+ * Whisper Model API
328
+ */
329
+ interface WhisperAPI {
330
+ checkModel: () => Promise<WhisperModelCheckResult>;
331
+ hasTranscriptionCapability: () => Promise<boolean>;
332
+ getAvailableModels: () => Promise<WhisperModelInfoPayload[]>;
333
+ downloadModel: (model: string) => Promise<{ success: boolean; error?: string }>;
334
+ cancelDownload: (model: string) => Promise<{ success: boolean }>;
335
+ onDownloadProgress: (callback: (data: WhisperDownloadProgressPayload) => void) => Unsubscribe;
336
+ onDownloadComplete: (callback: (data: { model: string; path: string }) => void) => Unsubscribe;
337
+ onDownloadError: (callback: (data: { model: string; error: string }) => void) => Unsubscribe;
338
+ }
339
+
340
+ /**
341
+ * Complete markupr API
342
+ */
343
+ export interface MarkuprAPI {
344
+ // Domain APIs
345
+ session: SessionAPI;
346
+ capture: CaptureAPI;
347
+ audio: AudioAPI;
348
+ screenRecording: ScreenRecordingAPI;
349
+ transcript: TranscriptAPI;
350
+ transcription: TranscriptionControlAPI;
351
+ processing: ProcessingAPI;
352
+ settings: SettingsAPI;
353
+ hotkeys: HotkeyAPI;
354
+ permissions: PermissionsAPI;
355
+ output: OutputAPI;
356
+ navigation: NavigationAPI;
357
+ window: WindowAPI;
358
+ popover: PopoverAPI;
359
+ crashRecovery: CrashRecoveryAPI;
360
+ updates: UpdatesAPI;
361
+ whisper: WhisperAPI;
362
+
363
+ // App version
364
+ version: () => Promise<string>;
365
+
366
+ // Legacy API (backwards compatibility)
367
+ startSession: () => Promise<{ success: boolean; sessionId?: string }>;
368
+ stopSession: () => Promise<{ success: boolean }>;
369
+ getSettings: () => Promise<AppSettings>;
370
+ setSettings: (settings: Partial<AppSettings>) => Promise<AppSettings>;
371
+ copyToClipboard: (text: string) => Promise<{ success: boolean }>;
372
+ onSessionStatus: (callback: (status: { action: string; status?: SessionStatusPayload }) => void) => Unsubscribe;
373
+ onTranscriptionUpdate: (callback: (data: { text: string; isFinal: boolean }) => void) => Unsubscribe;
374
+ onScreenshotCaptured: (callback: (data: { id: string; timestamp: number }) => void) => Unsubscribe;
375
+ onOutputReady: (callback: (data: OutputReadyPayload) => void) => Unsubscribe;
376
+ onOutputError: (callback: (error: { message: string }) => void) => Unsubscribe;
377
+ }
378
+
379
+ declare global {
380
+ interface Window {
381
+ markupr: MarkuprAPI;
382
+ }
383
+ }
384
+
385
+ export {};
@@ -0,0 +1,283 @@
1
+ /**
2
+ * Platform-aware hotkey definitions for markupr
3
+ *
4
+ * Provides consistent hotkey handling across macOS, Windows, and Linux.
5
+ * Uses Electron's accelerator format internally but converts to
6
+ * platform-appropriate display formats.
7
+ */
8
+
9
+ // ============================================================================
10
+ // Types
11
+ // ============================================================================
12
+
13
+ export interface HotkeyDefinition {
14
+ id: string;
15
+ label: string;
16
+ description: string;
17
+ macAccelerator: string;
18
+ winLinuxAccelerator: string;
19
+ }
20
+
21
+ export interface DisplayKey {
22
+ key: string;
23
+ symbol: string;
24
+ }
25
+
26
+ // ============================================================================
27
+ // Platform Detection
28
+ // ============================================================================
29
+
30
+ /**
31
+ * Detect if running on macOS
32
+ * Works in both main process (process.platform) and renderer (navigator.platform)
33
+ */
34
+ export function isMacOS(): boolean {
35
+ const nodeProcess = (globalThis as { process?: { platform?: string } }).process;
36
+ if (nodeProcess?.platform) {
37
+ return nodeProcess.platform === 'darwin';
38
+ }
39
+ if (typeof navigator !== 'undefined' && navigator.platform) {
40
+ return navigator.platform.toUpperCase().includes('MAC');
41
+ }
42
+ return false;
43
+ }
44
+
45
+ /**
46
+ * Detect if running on Windows
47
+ */
48
+ export function isWindows(): boolean {
49
+ const nodeProcess = (globalThis as { process?: { platform?: string } }).process;
50
+ if (nodeProcess?.platform) {
51
+ return nodeProcess.platform === 'win32';
52
+ }
53
+ if (typeof navigator !== 'undefined' && navigator.platform) {
54
+ return navigator.platform.toUpperCase().includes('WIN');
55
+ }
56
+ return false;
57
+ }
58
+
59
+ // ============================================================================
60
+ // Hotkey Definitions
61
+ // ============================================================================
62
+
63
+ /**
64
+ * All hotkey definitions for markupr
65
+ * Uses Electron accelerator format (CommandOrControl, Shift, Alt, etc.)
66
+ */
67
+ export const HOTKEYS: HotkeyDefinition[] = [
68
+ {
69
+ id: 'toggleRecording',
70
+ label: 'Toggle Recording',
71
+ description: 'Start or stop recording',
72
+ macAccelerator: 'Command+Shift+F',
73
+ winLinuxAccelerator: 'Ctrl+Shift+F',
74
+ },
75
+ {
76
+ id: 'manualScreenshot',
77
+ label: 'Manual Screenshot',
78
+ description: 'Capture screenshot immediately',
79
+ macAccelerator: 'Command+Shift+S',
80
+ winLinuxAccelerator: 'Ctrl+Shift+S',
81
+ },
82
+ {
83
+ id: 'pauseResume',
84
+ label: 'Pause/Resume',
85
+ description: 'Pause or resume recording',
86
+ macAccelerator: 'Command+Shift+P',
87
+ winLinuxAccelerator: 'Ctrl+Shift+P',
88
+ },
89
+ {
90
+ id: 'openSettings',
91
+ label: 'Open Settings',
92
+ description: 'Open settings panel',
93
+ macAccelerator: 'Command+,',
94
+ winLinuxAccelerator: 'Ctrl+,',
95
+ },
96
+ {
97
+ id: 'showHelp',
98
+ label: 'Show Help',
99
+ description: 'Show keyboard shortcuts',
100
+ macAccelerator: 'Command+?',
101
+ winLinuxAccelerator: 'Ctrl+Shift+/',
102
+ },
103
+ {
104
+ id: 'quit',
105
+ label: 'Quit',
106
+ description: 'Quit markupr',
107
+ macAccelerator: 'Command+Q',
108
+ winLinuxAccelerator: 'Alt+F4',
109
+ },
110
+ ];
111
+
112
+ // ============================================================================
113
+ // Key Display Mappings
114
+ // ============================================================================
115
+
116
+ /**
117
+ * macOS key display names (plain text for maximum rendering stability)
118
+ */
119
+ const MAC_SYMBOLS: Record<string, string> = {
120
+ command: 'Cmd',
121
+ cmd: 'Cmd',
122
+ control: 'Ctrl',
123
+ ctrl: 'Ctrl',
124
+ option: 'Option',
125
+ alt: 'Option',
126
+ shift: 'Shift',
127
+ enter: 'Enter',
128
+ return: 'Return',
129
+ delete: 'Delete',
130
+ backspace: 'Delete',
131
+ escape: 'Esc',
132
+ esc: 'Esc',
133
+ tab: 'Tab',
134
+ space: 'Space',
135
+ up: 'Up',
136
+ down: 'Down',
137
+ left: 'Left',
138
+ right: 'Right',
139
+ pageup: 'Page Up',
140
+ pagedown: 'Page Down',
141
+ home: 'Home',
142
+ end: 'End',
143
+ fn: 'Fn',
144
+ };
145
+
146
+ /**
147
+ * Windows/Linux key display names
148
+ */
149
+ const WIN_LINUX_NAMES: Record<string, string> = {
150
+ command: 'Ctrl',
151
+ cmd: 'Ctrl',
152
+ control: 'Ctrl',
153
+ ctrl: 'Ctrl',
154
+ option: 'Alt',
155
+ alt: 'Alt',
156
+ shift: 'Shift',
157
+ enter: 'Enter',
158
+ return: 'Enter',
159
+ delete: 'Del',
160
+ backspace: 'Backspace',
161
+ escape: 'Esc',
162
+ esc: 'Esc',
163
+ tab: 'Tab',
164
+ space: 'Space',
165
+ up: 'Up',
166
+ down: 'Down',
167
+ left: 'Left',
168
+ right: 'Right',
169
+ pageup: 'Page Up',
170
+ pagedown: 'Page Down',
171
+ home: 'Home',
172
+ end: 'End',
173
+ fn: 'Fn',
174
+ };
175
+
176
+ // ============================================================================
177
+ // Utility Functions
178
+ // ============================================================================
179
+
180
+ /**
181
+ * Get a hotkey definition by ID
182
+ */
183
+ export function getHotkeyById(id: string): HotkeyDefinition | undefined {
184
+ return HOTKEYS.find(h => h.id === id);
185
+ }
186
+
187
+ /**
188
+ * Get the Electron accelerator string for a hotkey
189
+ * Returns platform-appropriate accelerator
190
+ */
191
+ export function getAccelerator(hotkeyId: string): string {
192
+ const hotkey = getHotkeyById(hotkeyId);
193
+ if (!hotkey) return '';
194
+
195
+ return isMacOS() ? hotkey.macAccelerator : hotkey.winLinuxAccelerator;
196
+ }
197
+
198
+ /**
199
+ * Get the Electron accelerator from a custom hotkey string
200
+ * Handles both platform-specific and generic formats
201
+ */
202
+ export function normalizeAccelerator(accelerator: string): string {
203
+ // Replace CommandOrControl with platform-specific
204
+ if (isMacOS()) {
205
+ return accelerator
206
+ .replace(/CommandOrControl/gi, 'Command')
207
+ .replace(/CmdOrCtrl/gi, 'Cmd');
208
+ } else {
209
+ return accelerator
210
+ .replace(/CommandOrControl/gi, 'Control')
211
+ .replace(/CmdOrCtrl/gi, 'Ctrl')
212
+ .replace(/Command/gi, 'Control')
213
+ .replace(/Cmd/gi, 'Ctrl');
214
+ }
215
+ }
216
+
217
+ /**
218
+ * Parse an accelerator string into individual keys
219
+ */
220
+ export function parseAccelerator(accelerator: string): string[] {
221
+ return accelerator.split('+').map(k => k.trim());
222
+ }
223
+
224
+ /**
225
+ * Get display keys for an accelerator (for UI rendering)
226
+ * Returns keys formatted appropriately for the current platform
227
+ */
228
+ export function getDisplayKeys(accelerator: string): string[] {
229
+ const keys = parseAccelerator(accelerator);
230
+ const platform = isMacOS();
231
+
232
+ return keys.map(key => {
233
+ const lowerKey = key.toLowerCase();
234
+
235
+ if (platform) {
236
+ // macOS: Use symbols
237
+ return MAC_SYMBOLS[lowerKey] || key.toUpperCase();
238
+ } else {
239
+ // Windows/Linux: Use readable names
240
+ return WIN_LINUX_NAMES[lowerKey] || key.toUpperCase();
241
+ }
242
+ });
243
+ }
244
+
245
+ /**
246
+ * Get display keys for a hotkey by ID
247
+ */
248
+ export function getDisplayKeysById(hotkeyId: string): string[] {
249
+ const accelerator = getAccelerator(hotkeyId);
250
+ return getDisplayKeys(accelerator);
251
+ }
252
+
253
+ /**
254
+ * Format an accelerator for display
255
+ * Returns a single string like "Cmd+Shift+F" or "Ctrl+Shift+F"
256
+ */
257
+ export function formatAcceleratorForDisplay(accelerator: string): string {
258
+ const keys = getDisplayKeys(accelerator);
259
+ return keys.join('+');
260
+ }
261
+
262
+ /**
263
+ * Format a hotkey for display by ID
264
+ */
265
+ export function formatHotkeyForDisplay(hotkeyId: string): string {
266
+ const accelerator = getAccelerator(hotkeyId);
267
+ return formatAcceleratorForDisplay(accelerator);
268
+ }
269
+
270
+ /**
271
+ * Convert a HotkeyConfig-style accelerator to display format
272
+ * Handles "CommandOrControl" and other generic formats
273
+ */
274
+ export function formatHotkeyConfigForDisplay(accelerator: string): string {
275
+ const normalized = normalizeAccelerator(accelerator);
276
+ return formatAcceleratorForDisplay(normalized);
277
+ }
278
+
279
+ // ============================================================================
280
+ // Exports for types compatibility
281
+ // ============================================================================
282
+
283
+ export type HotkeyId = (typeof HOTKEYS)[number]['id'];