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,809 @@
1
+ /**
2
+ * Shared types for markupr
3
+ */
4
+
5
+ /**
6
+ * Represents a single screenshot captured during a feedback session
7
+ */
8
+ export interface Screenshot {
9
+ id: string;
10
+ timestamp: number;
11
+ imagePath: string;
12
+ base64?: string;
13
+ width: number;
14
+ height: number;
15
+ }
16
+
17
+ /**
18
+ * Represents a transcription segment from voice narration
19
+ */
20
+ export interface TranscriptionSegment {
21
+ id: string;
22
+ text: string;
23
+ startTime: number;
24
+ endTime: number;
25
+ confidence: number;
26
+ isFinal: boolean;
27
+ }
28
+
29
+ /**
30
+ * Represents a complete feedback session
31
+ */
32
+ export interface FeedbackSession {
33
+ id: string;
34
+ startedAt: number;
35
+ endedAt?: number;
36
+ screenshots: Screenshot[];
37
+ transcription: TranscriptionSegment[];
38
+ status: SessionStatus;
39
+ }
40
+
41
+ /**
42
+ * Session status enum (legacy, kept for compatibility)
43
+ */
44
+ export type SessionStatus = 'idle' | 'recording' | 'processing' | 'complete' | 'error';
45
+
46
+ /**
47
+ * Session state for bulletproof state machine.
48
+ * Every state except 'idle' has a maximum timeout for automatic recovery.
49
+ */
50
+ export type SessionState =
51
+ | 'idle' // Waiting for user action
52
+ | 'starting' // Initializing services (5s timeout)
53
+ | 'recording' // Active recording (30 min max)
54
+ | 'stopping' // Stopping services (3s timeout)
55
+ | 'processing' // Processing results (10s timeout)
56
+ | 'complete' // Session finished (30s timeout then auto-idle)
57
+ | 'error'; // Error occurred (5s timeout then auto-idle)
58
+
59
+ /**
60
+ * Tray icon visual states
61
+ */
62
+ export type TrayState = 'idle' | 'recording' | 'processing' | 'complete' | 'error';
63
+
64
+ /**
65
+ * Hotkey configuration for the application
66
+ */
67
+ export interface HotkeyConfig {
68
+ toggleRecording: string;
69
+ manualScreenshot: string;
70
+ pauseResume: string;
71
+ }
72
+
73
+ /**
74
+ * Default hotkey configuration
75
+ */
76
+ export const DEFAULT_HOTKEY_CONFIG: HotkeyConfig = {
77
+ toggleRecording: 'CommandOrControl+Shift+F',
78
+ manualScreenshot: 'CommandOrControl+Shift+S',
79
+ pauseResume: 'CommandOrControl+Shift+P',
80
+ };
81
+
82
+ /**
83
+ * Application settings (v2 - expanded schema)
84
+ *
85
+ * Note: API keys are NOT stored in settings.
86
+ * Use SettingsManager.getApiKey('<service>') for secure storage via keytar.
87
+ */
88
+ export interface AppSettings {
89
+ // General
90
+ outputDirectory: string;
91
+ launchAtLogin: boolean;
92
+ checkForUpdates: boolean;
93
+
94
+ // Recording
95
+ defaultCountdown: 0 | 3 | 5;
96
+ showTranscriptionPreview: boolean;
97
+ showAudioWaveform: boolean;
98
+
99
+ // Capture
100
+ pauseThreshold: number; // 500-3000ms
101
+ minTimeBetweenCaptures: number; // 300-2000ms
102
+ imageFormat: 'png' | 'jpeg';
103
+ imageQuality: number; // 1-100 for jpeg
104
+ maxImageWidth: number; // 800-2400
105
+
106
+ // Transcription
107
+ transcriptionService: 'openai';
108
+ language: string;
109
+ enableKeywordTriggers: boolean;
110
+
111
+ // Hotkeys
112
+ hotkeys: HotkeyConfig;
113
+
114
+ // Appearance
115
+ theme: 'dark' | 'light' | 'system';
116
+ accentColor: string;
117
+
118
+ // Audio
119
+ audioDeviceId: string | null;
120
+
121
+ // Advanced
122
+ debugMode: boolean;
123
+ keepAudioBackups: boolean;
124
+
125
+ // Onboarding
126
+ hasCompletedOnboarding: boolean;
127
+
128
+ // Legacy fields (for migration compatibility)
129
+ /** @deprecated Use imageQuality instead */
130
+ screenshotQuality?: number;
131
+ /** @deprecated Use pauseThreshold instead */
132
+ pauseThresholdMs?: number;
133
+ /** @deprecated Clipboard is always available, no setting needed */
134
+ autoClipboard?: boolean;
135
+ /** @deprecated Output format is always markdown */
136
+ outputFormat?: 'markdown' | 'json';
137
+ /** @deprecated Use audioDeviceId instead */
138
+ preferredAudioDevice?: string;
139
+ }
140
+
141
+ /**
142
+ * Result of validating a provider API key from the main process.
143
+ */
144
+ export interface ApiKeyValidationResult {
145
+ valid: boolean;
146
+ error?: string;
147
+ status?: number;
148
+ }
149
+
150
+ /**
151
+ * Default settings
152
+ */
153
+ export const DEFAULT_SETTINGS: AppSettings = {
154
+ // General
155
+ outputDirectory: '', // Set dynamically by SettingsManager
156
+ launchAtLogin: false,
157
+ checkForUpdates: true,
158
+
159
+ // Recording
160
+ defaultCountdown: 0,
161
+ showTranscriptionPreview: true,
162
+ showAudioWaveform: true,
163
+
164
+ // Capture
165
+ pauseThreshold: 1500,
166
+ minTimeBetweenCaptures: 500,
167
+ imageFormat: 'png',
168
+ imageQuality: 85,
169
+ maxImageWidth: 1920,
170
+
171
+ // Transcription
172
+ transcriptionService: 'openai',
173
+ language: 'en',
174
+ enableKeywordTriggers: false,
175
+
176
+ // Hotkeys
177
+ hotkeys: { ...DEFAULT_HOTKEY_CONFIG },
178
+
179
+ // Appearance
180
+ theme: 'system',
181
+ accentColor: '#3B82F6',
182
+
183
+ // Audio
184
+ audioDeviceId: null,
185
+
186
+ // Advanced
187
+ debugMode: false,
188
+ keepAudioBackups: false,
189
+
190
+ // Onboarding
191
+ hasCompletedOnboarding: false,
192
+ };
193
+
194
+ // =============================================================================
195
+ // IPC Channel Definitions
196
+ // =============================================================================
197
+
198
+ /**
199
+ * IPC channel names for main/renderer communication
200
+ * Namespaced with 'markupr:' prefix for clarity
201
+ */
202
+ export const IPC_CHANNELS = {
203
+ // ---------------------------------------------------------------------------
204
+ // Session Channels (Renderer -> Main)
205
+ // ---------------------------------------------------------------------------
206
+ SESSION_START: 'markupr:session:start',
207
+ SESSION_STOP: 'markupr:session:stop',
208
+ SESSION_PAUSE: 'markupr:session:pause',
209
+ SESSION_RESUME: 'markupr:session:resume',
210
+ SESSION_CANCEL: 'markupr:session:cancel',
211
+ SESSION_GET_STATUS: 'markupr:session:get-status',
212
+ SESSION_GET_CURRENT: 'markupr:session:get-current',
213
+
214
+ // ---------------------------------------------------------------------------
215
+ // Session Channels (Main -> Renderer)
216
+ // ---------------------------------------------------------------------------
217
+ SESSION_STATE_CHANGED: 'markupr:session:state-changed',
218
+ SESSION_STATUS: 'markupr:session:status-update',
219
+ SESSION_COMPLETE: 'markupr:session:complete',
220
+ SESSION_FEEDBACK_ITEM: 'markupr:session:feedback-item',
221
+ SESSION_VOICE_ACTIVITY: 'markupr:session:voice-activity',
222
+ SESSION_ERROR: 'markupr:session:error',
223
+
224
+ // ---------------------------------------------------------------------------
225
+ // Capture Channels (Renderer -> Main)
226
+ // ---------------------------------------------------------------------------
227
+ CAPTURE_GET_SOURCES: 'markupr:capture:get-sources',
228
+ CAPTURE_MANUAL_SCREENSHOT: 'markupr:capture:manual-screenshot',
229
+ SCREEN_RECORDING_START: 'markupr:screen-recording:start',
230
+ SCREEN_RECORDING_CHUNK: 'markupr:screen-recording:chunk',
231
+ SCREEN_RECORDING_STOP: 'markupr:screen-recording:stop',
232
+
233
+ // ---------------------------------------------------------------------------
234
+ // Capture Channels (Main -> Renderer)
235
+ // ---------------------------------------------------------------------------
236
+ SCREENSHOT_CAPTURED: 'markupr:capture:screenshot-taken',
237
+ MANUAL_SCREENSHOT: 'markupr:capture:manual-triggered',
238
+
239
+ // ---------------------------------------------------------------------------
240
+ // Display Channels (Main -> Renderer) - Multi-monitor support
241
+ // ---------------------------------------------------------------------------
242
+ DISPLAYS_CHANGED: 'markupr:displays:changed',
243
+ DISPLAY_DISCONNECTED: 'markupr:display:disconnected',
244
+
245
+ // ---------------------------------------------------------------------------
246
+ // Audio Channels (Renderer -> Main)
247
+ // ---------------------------------------------------------------------------
248
+ AUDIO_GET_DEVICES: 'markupr:audio:get-devices',
249
+ AUDIO_SET_DEVICE: 'markupr:audio:set-device',
250
+
251
+ // ---------------------------------------------------------------------------
252
+ // Audio Channels (Main -> Renderer) - Communication with audio capture
253
+ // ---------------------------------------------------------------------------
254
+ AUDIO_REQUEST_DEVICES: 'markupr:audio:request-devices',
255
+ AUDIO_START_CAPTURE: 'markupr:audio:start-capture',
256
+ AUDIO_STOP_CAPTURE: 'markupr:audio:stop-capture',
257
+ AUDIO_CHUNK: 'markupr:audio:chunk',
258
+ AUDIO_DEVICES_RESPONSE: 'markupr:audio:devices-response',
259
+ AUDIO_CAPTURE_ERROR: 'markupr:audio:capture-error',
260
+ AUDIO_CAPTURE_STARTED: 'markupr:audio:capture-started',
261
+ AUDIO_CAPTURE_STOPPED: 'markupr:audio:capture-stopped',
262
+ AUDIO_LEVEL: 'markupr:audio:level',
263
+ AUDIO_VOICE_ACTIVITY: 'markupr:audio:voice-activity',
264
+
265
+ // ---------------------------------------------------------------------------
266
+ // Transcription Channels (Main -> Renderer)
267
+ // ---------------------------------------------------------------------------
268
+ TRANSCRIPTION_UPDATE: 'markupr:transcript:chunk',
269
+ TRANSCRIPTION_FINAL: 'markupr:transcript:final',
270
+
271
+ // ---------------------------------------------------------------------------
272
+ // Transcription Control Channels (Renderer -> Main)
273
+ // ---------------------------------------------------------------------------
274
+ TRANSCRIPTION_GET_TIER_STATUSES: 'markupr:transcription:get-tier-statuses',
275
+ TRANSCRIPTION_GET_CURRENT_TIER: 'markupr:transcription:get-current-tier',
276
+ TRANSCRIPTION_SET_TIER: 'markupr:transcription:set-tier',
277
+
278
+ // ---------------------------------------------------------------------------
279
+ // Settings Channels (Renderer -> Main)
280
+ // ---------------------------------------------------------------------------
281
+ SETTINGS_GET: 'markupr:settings:get',
282
+ SETTINGS_GET_ALL: 'markupr:settings:get-all',
283
+ SETTINGS_SET: 'markupr:settings:set',
284
+ SETTINGS_GET_API_KEY: 'markupr:settings:get-api-key',
285
+ SETTINGS_SET_API_KEY: 'markupr:settings:set-api-key',
286
+ SETTINGS_DELETE_API_KEY: 'markupr:settings:delete-api-key',
287
+ SETTINGS_HAS_API_KEY: 'markupr:settings:has-api-key',
288
+ SETTINGS_TEST_API_KEY: 'markupr:settings:test-api-key',
289
+ SETTINGS_SELECT_DIRECTORY: 'markupr:settings:select-directory',
290
+ SETTINGS_CLEAR_ALL_DATA: 'markupr:settings:clear-all-data',
291
+ SETTINGS_EXPORT: 'markupr:settings:export',
292
+ SETTINGS_IMPORT: 'markupr:settings:import',
293
+
294
+ // ---------------------------------------------------------------------------
295
+ // Permissions Channels (Renderer -> Main)
296
+ // ---------------------------------------------------------------------------
297
+ PERMISSIONS_CHECK: 'markupr:permissions:check',
298
+ PERMISSIONS_REQUEST: 'markupr:permissions:request',
299
+ PERMISSIONS_GET_ALL: 'markupr:permissions:get-all',
300
+
301
+ // ---------------------------------------------------------------------------
302
+ // Output Channels (Renderer -> Main)
303
+ // ---------------------------------------------------------------------------
304
+ OUTPUT_SAVE: 'markupr:output:save',
305
+ OUTPUT_COPY_CLIPBOARD: 'markupr:output:copy-clipboard',
306
+ OUTPUT_OPEN_FOLDER: 'markupr:output:open-folder',
307
+ OUTPUT_EXPORT: 'markupr:output:export',
308
+
309
+ // Session History Browser
310
+ OUTPUT_LIST_SESSIONS: 'markupr:output:list-sessions',
311
+ OUTPUT_GET_SESSION_METADATA: 'markupr:output:get-session-metadata',
312
+ OUTPUT_DELETE_SESSION: 'markupr:output:delete-session',
313
+ OUTPUT_DELETE_SESSIONS: 'markupr:output:delete-sessions',
314
+ OUTPUT_EXPORT_SESSION: 'markupr:output:export-session',
315
+ OUTPUT_EXPORT_SESSIONS: 'markupr:output:export-sessions',
316
+
317
+ // ---------------------------------------------------------------------------
318
+ // Output Channels (Main -> Renderer)
319
+ // ---------------------------------------------------------------------------
320
+ OUTPUT_READY: 'markupr:output:ready',
321
+ OUTPUT_ERROR: 'markupr:output:error',
322
+
323
+ // ---------------------------------------------------------------------------
324
+ // Hotkey Channels
325
+ // ---------------------------------------------------------------------------
326
+ HOTKEY_TRIGGERED: 'markupr:hotkey:triggered',
327
+ HOTKEY_CONFIG: 'markupr:hotkey:config',
328
+ HOTKEY_UPDATE: 'markupr:hotkey:update',
329
+
330
+ // ---------------------------------------------------------------------------
331
+ // Clipboard (Legacy - kept for compatibility)
332
+ // ---------------------------------------------------------------------------
333
+ COPY_TO_CLIPBOARD: 'markupr:clipboard:copy',
334
+
335
+ // ---------------------------------------------------------------------------
336
+ // Window Control Channels
337
+ // ---------------------------------------------------------------------------
338
+ WINDOW_MINIMIZE: 'markupr:window:minimize',
339
+ WINDOW_CLOSE: 'markupr:window:close',
340
+ WINDOW_HIDE: 'markupr:window:hide',
341
+
342
+ // ---------------------------------------------------------------------------
343
+ // Update Channels (Renderer -> Main)
344
+ // ---------------------------------------------------------------------------
345
+ UPDATE_CHECK: 'markupr:update:check',
346
+ UPDATE_DOWNLOAD: 'markupr:update:download',
347
+ UPDATE_INSTALL: 'markupr:update:install',
348
+
349
+ // ---------------------------------------------------------------------------
350
+ // Update Channels (Main -> Renderer)
351
+ // ---------------------------------------------------------------------------
352
+ UPDATE_STATUS: 'markupr:update:status',
353
+
354
+ // ---------------------------------------------------------------------------
355
+ // Crash Recovery Channels (Renderer -> Main)
356
+ // ---------------------------------------------------------------------------
357
+ CRASH_RECOVERY_CHECK: 'markupr:crash-recovery:check',
358
+ CRASH_RECOVERY_RECOVER: 'markupr:crash-recovery:recover',
359
+ CRASH_RECOVERY_DISCARD: 'markupr:crash-recovery:discard',
360
+ CRASH_RECOVERY_GET_LOGS: 'markupr:crash-recovery:get-logs',
361
+ CRASH_RECOVERY_CLEAR_LOGS: 'markupr:crash-recovery:clear-logs',
362
+ CRASH_RECOVERY_UPDATE_SETTINGS: 'markupr:crash-recovery:update-settings',
363
+
364
+ // ---------------------------------------------------------------------------
365
+ // Crash Recovery Channels (Main -> Renderer)
366
+ // ---------------------------------------------------------------------------
367
+ CRASH_RECOVERY_FOUND: 'markupr:crash-recovery:found',
368
+
369
+ // ---------------------------------------------------------------------------
370
+ // Taskbar Channels (Renderer -> Main) - Windows-specific
371
+ // ---------------------------------------------------------------------------
372
+ TASKBAR_SET_PROGRESS: 'markupr:taskbar:setProgress',
373
+ TASKBAR_FLASH_FRAME: 'markupr:taskbar:flashFrame',
374
+ TASKBAR_SET_OVERLAY: 'markupr:taskbar:setOverlay',
375
+
376
+ // ---------------------------------------------------------------------------
377
+ // Whisper Model Channels (Renderer -> Main)
378
+ // ---------------------------------------------------------------------------
379
+ WHISPER_CHECK_MODEL: 'markupr:whisper:check-model',
380
+ WHISPER_DOWNLOAD_MODEL: 'markupr:whisper:download-model',
381
+ WHISPER_CANCEL_DOWNLOAD: 'markupr:whisper:cancel-download',
382
+ WHISPER_GET_AVAILABLE_MODELS: 'markupr:whisper:get-available-models',
383
+ WHISPER_HAS_TRANSCRIPTION_CAPABILITY: 'markupr:whisper:has-transcription-capability',
384
+
385
+ // ---------------------------------------------------------------------------
386
+ // Whisper Model Channels (Main -> Renderer)
387
+ // ---------------------------------------------------------------------------
388
+ WHISPER_DOWNLOAD_PROGRESS: 'markupr:whisper:download-progress',
389
+ WHISPER_DOWNLOAD_COMPLETE: 'markupr:whisper:download-complete',
390
+ WHISPER_DOWNLOAD_ERROR: 'markupr:whisper:download-error',
391
+
392
+ // ---------------------------------------------------------------------------
393
+ // Processing Pipeline Channels (Main -> Renderer)
394
+ // ---------------------------------------------------------------------------
395
+ PROCESSING_PROGRESS: 'markupr:processing:progress',
396
+ PROCESSING_COMPLETE: 'markupr:processing:complete',
397
+
398
+ // ---------------------------------------------------------------------------
399
+ // Session Events (Main -> Renderer, non-standard)
400
+ // ---------------------------------------------------------------------------
401
+ SESSION_RECOVERED: 'markupr:session:recovered',
402
+ SESSION_WARNING: 'markupr:session:warning',
403
+
404
+ // ---------------------------------------------------------------------------
405
+ // Navigation Channels (Main -> Renderer)
406
+ // ---------------------------------------------------------------------------
407
+ SHOW_ONBOARDING: 'markupr:show-onboarding',
408
+ SHOW_SETTINGS: 'markupr:show-settings',
409
+ SHOW_HISTORY: 'markupr:show-history',
410
+ SHOW_EXPORT: 'markupr:show-export',
411
+ SHOW_SHORTCUTS: 'markupr:show-shortcuts',
412
+ SHOW_WINDOW_SELECTOR: 'markupr:show-window-selector',
413
+ OPEN_SESSION_DIALOG: 'markupr:open-session-dialog',
414
+ OPEN_SESSION: 'markupr:open-session',
415
+
416
+ // ---------------------------------------------------------------------------
417
+ // App Channels (Renderer -> Main)
418
+ // ---------------------------------------------------------------------------
419
+ APP_VERSION: 'markupr:app:version',
420
+
421
+ // ---------------------------------------------------------------------------
422
+ // Popover Channels (Renderer -> Main)
423
+ // ---------------------------------------------------------------------------
424
+ POPOVER_RESIZE: 'markupr:popover:resize',
425
+ POPOVER_RESIZE_TO_STATE: 'markupr:popover:resize-to-state',
426
+ POPOVER_SHOW: 'markupr:popover:show',
427
+ POPOVER_HIDE: 'markupr:popover:hide',
428
+ POPOVER_TOGGLE: 'markupr:popover:toggle',
429
+
430
+ // ---------------------------------------------------------------------------
431
+ // Error Channels (Main -> Renderer)
432
+ // ---------------------------------------------------------------------------
433
+ NETWORK_ERROR: 'markupr:network-error',
434
+ NETWORK_RESTORED: 'markupr:network-restored',
435
+ CAPTURE_WARNING: 'markupr:capture-warning',
436
+ AUDIO_ERROR: 'markupr:audio-error',
437
+ TRANSCRIPTION_ERROR: 'markupr:transcription-error',
438
+ NOTIFICATION: 'markupr:notification',
439
+
440
+ // ---------------------------------------------------------------------------
441
+ // Legacy channels (backwards compatibility)
442
+ // ---------------------------------------------------------------------------
443
+ START_SESSION: 'session:start',
444
+ STOP_SESSION: 'session:stop',
445
+ GET_SETTINGS: 'settings:get',
446
+ SET_SETTINGS: 'settings:set',
447
+ } as const;
448
+
449
+ /**
450
+ * Type for IPC channel names
451
+ */
452
+ export type IPCChannel = (typeof IPC_CHANNELS)[keyof typeof IPC_CHANNELS];
453
+
454
+ // =============================================================================
455
+ // IPC Payload Types
456
+ // =============================================================================
457
+
458
+ /**
459
+ * Session status update payload
460
+ */
461
+ export interface SessionStatusPayload {
462
+ state: SessionState;
463
+ duration: number;
464
+ feedbackCount: number;
465
+ screenshotCount: number;
466
+ isPaused: boolean;
467
+ }
468
+
469
+ /**
470
+ * Feedback item payload (without Buffer)
471
+ */
472
+ export interface FeedbackItemPayload {
473
+ id: string;
474
+ timestamp: number;
475
+ text: string;
476
+ confidence: number;
477
+ hasScreenshot: boolean;
478
+ screenshotId?: string;
479
+ }
480
+
481
+ /**
482
+ * Transcript chunk payload
483
+ */
484
+ export interface TranscriptChunkPayload {
485
+ text: string;
486
+ timestamp: number;
487
+ confidence: number;
488
+ isFinal: boolean;
489
+ words?: Array<{
490
+ word: string;
491
+ start: number;
492
+ end: number;
493
+ confidence: number;
494
+ }>;
495
+ }
496
+
497
+ /**
498
+ * Screenshot captured payload
499
+ */
500
+ export interface ScreenshotCapturedPayload {
501
+ id: string;
502
+ timestamp: number;
503
+ count: number;
504
+ width?: number;
505
+ height?: number;
506
+ trigger?: 'pause' | 'manual' | 'voice-command';
507
+ }
508
+
509
+ /**
510
+ * Processing pipeline progress payload
511
+ */
512
+ export interface ProcessingProgressPayload {
513
+ percent: number;
514
+ step: string;
515
+ }
516
+
517
+ /**
518
+ * Transcription tier identifiers used by UI and IPC.
519
+ */
520
+ export type TranscriptionTier = 'auto' | 'whisper' | 'timer-only';
521
+
522
+ /**
523
+ * Runtime availability status for a transcription tier.
524
+ */
525
+ export interface TranscriptionTierStatus {
526
+ tier: Exclude<TranscriptionTier, 'auto'>;
527
+ available: boolean;
528
+ reason?: string;
529
+ }
530
+
531
+ // =============================================================================
532
+ // Review Session Types (renderer-compatible mirrors of MarkdownGenerator types)
533
+ // =============================================================================
534
+
535
+ /**
536
+ * Feedback category labels for the review UI
537
+ */
538
+ export type ReviewFeedbackCategory = 'Bug' | 'UX Issue' | 'Suggestion' | 'Performance' | 'Question' | 'General';
539
+
540
+ /**
541
+ * Feedback severity levels for the review UI
542
+ */
543
+ export type ReviewFeedbackSeverity = 'Critical' | 'High' | 'Medium' | 'Low';
544
+
545
+ /**
546
+ * A single feedback item in the review session (renderer-safe)
547
+ */
548
+ export interface ReviewFeedbackItem {
549
+ id: string;
550
+ transcription: string;
551
+ timestamp: number;
552
+ screenshots: Screenshot[];
553
+ title?: string;
554
+ keywords?: string[];
555
+ category?: ReviewFeedbackCategory;
556
+ severity?: ReviewFeedbackSeverity;
557
+ }
558
+
559
+ /**
560
+ * Review session metadata
561
+ */
562
+ export interface ReviewSessionMetadata {
563
+ os?: string;
564
+ sourceName?: string;
565
+ sourceType?: 'screen' | 'window';
566
+ /** Epoch ms when video recording started, for computing video offsets */
567
+ videoStartTime?: number;
568
+ }
569
+
570
+ /**
571
+ * Complete review session for SessionReview component (renderer-safe)
572
+ */
573
+ export interface ReviewSession {
574
+ id: string;
575
+ startTime: number;
576
+ endTime?: number;
577
+ feedbackItems: ReviewFeedbackItem[];
578
+ metadata?: ReviewSessionMetadata;
579
+ }
580
+
581
+ /**
582
+ * Output ready payload
583
+ */
584
+ export interface OutputReadyPayload {
585
+ markdown: string;
586
+ sessionId: string;
587
+ path: string;
588
+ reportPath?: string;
589
+ sessionDir?: string;
590
+ recordingPath?: string;
591
+ audioPath?: string;
592
+ audioDurationMs?: number;
593
+ /** Epoch ms when video recording started, for computing video offsets from transcript timestamps */
594
+ videoStartTime?: number;
595
+ /** The full review session for SessionReview component */
596
+ reviewSession?: ReviewSession;
597
+ }
598
+
599
+ /**
600
+ * Whisper model download progress payload
601
+ */
602
+ export interface WhisperDownloadProgressPayload {
603
+ model: string;
604
+ downloadedBytes: number;
605
+ totalBytes: number;
606
+ percent: number;
607
+ speedBps: number;
608
+ estimatedSecondsRemaining: number;
609
+ }
610
+
611
+ /**
612
+ * Whisper model info payload
613
+ */
614
+ export interface WhisperModelInfoPayload {
615
+ name: string;
616
+ filename: string;
617
+ sizeMB: number;
618
+ ramRequired: string;
619
+ quality: string;
620
+ isDownloaded: boolean;
621
+ }
622
+
623
+ /**
624
+ * Whisper model check result
625
+ */
626
+ export interface WhisperModelCheckResult {
627
+ hasAnyModel: boolean;
628
+ defaultModel: string | null;
629
+ downloadedModels: string[];
630
+ recommendedModel: string;
631
+ recommendedModelSizeMB: number;
632
+ }
633
+
634
+ /**
635
+ * Permission types
636
+ */
637
+ export type PermissionType = 'microphone' | 'screen' | 'accessibility';
638
+
639
+ /**
640
+ * Permission status
641
+ */
642
+ export interface PermissionStatus {
643
+ microphone: boolean;
644
+ screen: boolean;
645
+ accessibility: boolean;
646
+ }
647
+
648
+ /**
649
+ * Save result
650
+ */
651
+ export interface SaveResult {
652
+ success: boolean;
653
+ path?: string;
654
+ error?: string;
655
+ }
656
+
657
+ // =============================================================================
658
+ // Update Types
659
+ // =============================================================================
660
+
661
+ /**
662
+ * Update status types for auto-updater
663
+ */
664
+ export type UpdateStatusType =
665
+ | 'idle'
666
+ | 'checking'
667
+ | 'available'
668
+ | 'not-available'
669
+ | 'downloading'
670
+ | 'ready'
671
+ | 'error';
672
+
673
+ /**
674
+ * Update status payload from main process
675
+ */
676
+ export interface UpdateStatusPayload {
677
+ status: UpdateStatusType;
678
+ version?: string;
679
+ releaseNotes?: string | null;
680
+ releaseDate?: string;
681
+ percent?: number;
682
+ bytesPerSecond?: number;
683
+ total?: number;
684
+ transferred?: number;
685
+ message?: string;
686
+ }
687
+
688
+ // =============================================================================
689
+ // Audio Types
690
+ // =============================================================================
691
+
692
+ /**
693
+ * Audio device representation
694
+ */
695
+ export interface AudioDevice {
696
+ id: string;
697
+ name: string;
698
+ isDefault: boolean;
699
+ }
700
+
701
+ /**
702
+ * Audio chunk for streaming to transcription
703
+ */
704
+ export interface AudioChunk {
705
+ buffer: Buffer;
706
+ timestamp: number;
707
+ duration: number;
708
+ sampleRate: number;
709
+ }
710
+
711
+ /**
712
+ * Audio capture configuration
713
+ */
714
+ export interface AudioCaptureConfig {
715
+ sampleRate: number;
716
+ channels: number;
717
+ chunkDurationMs: number;
718
+ vadThreshold: number;
719
+ vadSilenceMs: number;
720
+ recoveryBufferMinutes: number;
721
+ }
722
+
723
+ /**
724
+ * Default audio configuration (optimized for speech transcription)
725
+ */
726
+ export const DEFAULT_AUDIO_CONFIG: AudioCaptureConfig = {
727
+ sampleRate: 16000,
728
+ channels: 1,
729
+ chunkDurationMs: 100,
730
+ vadThreshold: 0.01,
731
+ vadSilenceMs: 300,
732
+ recoveryBufferMinutes: 5,
733
+ };
734
+
735
+ // =============================================================================
736
+ // Capture Types
737
+ // =============================================================================
738
+
739
+ /**
740
+ * Output document structure
741
+ */
742
+ export interface OutputDocument {
743
+ sessionId: string;
744
+ generatedAt: number;
745
+ markdown: string;
746
+ screenshots: Screenshot[];
747
+ }
748
+
749
+ /**
750
+ * Display information with layout positioning for multi-monitor support
751
+ */
752
+ export interface DisplayInfo {
753
+ id: number;
754
+ label: string;
755
+ bounds: { x: number; y: number; width: number; height: number };
756
+ workArea: { x: number; y: number; width: number; height: number };
757
+ scaleFactor: number;
758
+ isPrimary: boolean;
759
+ rotation: 0 | 90 | 180 | 270;
760
+ internal: boolean;
761
+ }
762
+
763
+ /**
764
+ * Capture source for window/screen selection
765
+ */
766
+ export interface CaptureSource {
767
+ id: string;
768
+ name: string;
769
+ type: 'screen' | 'window';
770
+ thumbnail?: string;
771
+ appIcon?: string;
772
+ /** Display info for screen sources (multi-monitor support) */
773
+ display?: DisplayInfo;
774
+ }
775
+
776
+ // =============================================================================
777
+ // Session Types (for IPC)
778
+ // =============================================================================
779
+
780
+ /**
781
+ * Session metadata for IPC transport
782
+ */
783
+ export interface SessionMetadata {
784
+ sourceId: string;
785
+ sourceName?: string;
786
+ windowTitle?: string;
787
+ appName?: string;
788
+ recordingPath?: string;
789
+ recordingMimeType?: string;
790
+ recordingBytes?: number;
791
+ audioPath?: string;
792
+ audioBytes?: number;
793
+ audioDurationMs?: number;
794
+ /** Epoch ms when video recording started, for computing video offsets from transcript timestamps */
795
+ videoStartTime?: number;
796
+ }
797
+
798
+ /**
799
+ * Complete session for IPC (excludes Buffer data)
800
+ */
801
+ export interface SessionPayload {
802
+ id: string;
803
+ startTime: number;
804
+ endTime?: number;
805
+ state: SessionState;
806
+ sourceId: string;
807
+ feedbackItems: FeedbackItemPayload[];
808
+ metadata: SessionMetadata;
809
+ }