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,632 @@
1
+ /**
2
+ * CrashRecoveryDialog - Recovery UI for incomplete sessions
3
+ *
4
+ * Shows when markupr detects an incomplete session from a previous
5
+ * crash or abnormal exit. Offers the user the choice to recover or discard.
6
+ */
7
+
8
+ import React, { useState, useEffect, useCallback } from 'react';
9
+ import { useTheme } from '../hooks/useTheme';
10
+
11
+ // ============================================================================
12
+ // Types
13
+ // ============================================================================
14
+
15
+ export interface RecoverableSession {
16
+ id: string;
17
+ startTime: number;
18
+ lastSaveTime: number;
19
+ feedbackItems: RecoverableFeedbackItem[];
20
+ transcriptionBuffer: string;
21
+ sourceId: string;
22
+ sourceName: string;
23
+ screenshotCount: number;
24
+ metadata?: {
25
+ appVersion: string;
26
+ platform: string;
27
+ sessionDurationMs: number;
28
+ };
29
+ }
30
+
31
+ export interface RecoverableFeedbackItem {
32
+ id: string;
33
+ timestamp: number;
34
+ text: string;
35
+ confidence: number;
36
+ hasScreenshot: boolean;
37
+ screenshotId?: string;
38
+ }
39
+
40
+ export interface CrashRecoveryDialogProps {
41
+ session: RecoverableSession;
42
+ onRecover: (session: RecoverableSession) => void;
43
+ onDiscard: () => void;
44
+ }
45
+
46
+ // ============================================================================
47
+ // Helper Functions
48
+ // ============================================================================
49
+
50
+ function formatTimeSince(ms: number): string {
51
+ const seconds = Math.floor(ms / 1000);
52
+ const minutes = Math.floor(seconds / 60);
53
+ const hours = Math.floor(minutes / 60);
54
+ const days = Math.floor(hours / 24);
55
+
56
+ if (days > 0) {
57
+ return `${days} day${days !== 1 ? 's' : ''}`;
58
+ }
59
+ if (hours > 0) {
60
+ return `${hours} hour${hours !== 1 ? 's' : ''}`;
61
+ }
62
+ if (minutes > 0) {
63
+ return `${minutes} minute${minutes !== 1 ? 's' : ''}`;
64
+ }
65
+ return 'just now';
66
+ }
67
+
68
+ function formatDuration(ms: number): string {
69
+ const seconds = Math.floor(ms / 1000);
70
+ const minutes = Math.floor(seconds / 60);
71
+ const remainingSeconds = seconds % 60;
72
+
73
+ if (minutes > 0) {
74
+ return `${minutes}m ${remainingSeconds}s`;
75
+ }
76
+ return `${seconds}s`;
77
+ }
78
+
79
+ function formatDate(timestamp: number): string {
80
+ const date = new Date(timestamp);
81
+ return date.toLocaleString(undefined, {
82
+ weekday: 'short',
83
+ month: 'short',
84
+ day: 'numeric',
85
+ hour: 'numeric',
86
+ minute: '2-digit',
87
+ });
88
+ }
89
+
90
+ // ============================================================================
91
+ // Component
92
+ // ============================================================================
93
+
94
+ export function CrashRecoveryDialog({
95
+ session,
96
+ onRecover,
97
+ onDiscard,
98
+ }: CrashRecoveryDialogProps): React.ReactElement {
99
+ const [isRecovering, setIsRecovering] = useState(false);
100
+ const [isDiscarding, setIsDiscarding] = useState(false);
101
+ const [showDetails, setShowDetails] = useState(false);
102
+ const [hoveredBtn, setHoveredBtn] = useState<'recover' | 'discard' | null>(null);
103
+ const { colors } = useTheme();
104
+
105
+ const timeSince = Date.now() - session.lastSaveTime;
106
+ const formattedTime = formatTimeSince(timeSince);
107
+ const sessionDuration = session.metadata?.sessionDurationMs ||
108
+ (session.lastSaveTime - session.startTime);
109
+
110
+ const handleRecover = useCallback(async () => {
111
+ setIsRecovering(true);
112
+ try {
113
+ onRecover(session);
114
+ } catch (error) {
115
+ console.error('Recovery failed:', error);
116
+ setIsRecovering(false);
117
+ }
118
+ }, [onRecover, session]);
119
+
120
+ const handleDiscard = useCallback(async () => {
121
+ setIsDiscarding(true);
122
+ try {
123
+ onDiscard();
124
+ } catch (error) {
125
+ console.error('Discard failed:', error);
126
+ setIsDiscarding(false);
127
+ }
128
+ }, [onDiscard]);
129
+
130
+ // Keyboard shortcuts
131
+ useEffect(() => {
132
+ const handleKeyDown = (e: KeyboardEvent) => {
133
+ if (isRecovering || isDiscarding) return;
134
+
135
+ if (e.key === 'Enter' || e.key === 'r' || e.key === 'R') {
136
+ e.preventDefault();
137
+ handleRecover();
138
+ } else if (e.key === 'Escape' || e.key === 'd' || e.key === 'D') {
139
+ e.preventDefault();
140
+ handleDiscard();
141
+ }
142
+ };
143
+
144
+ window.addEventListener('keydown', handleKeyDown);
145
+ return () => window.removeEventListener('keydown', handleKeyDown);
146
+ }, [isRecovering, isDiscarding, handleRecover, handleDiscard]);
147
+
148
+ const spinnerSvg = (
149
+ <svg style={{ width: 16, height: 16, animation: 'spin 1s linear infinite' }} fill="none" viewBox="0 0 24 24">
150
+ <circle
151
+ style={{ opacity: 0.25 }}
152
+ cx="12"
153
+ cy="12"
154
+ r="10"
155
+ stroke="currentColor"
156
+ strokeWidth="4"
157
+ />
158
+ <path
159
+ style={{ opacity: 0.75 }}
160
+ fill="currentColor"
161
+ d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"
162
+ />
163
+ </svg>
164
+ );
165
+
166
+ return (
167
+ <div style={{
168
+ position: 'fixed',
169
+ inset: 0,
170
+ backgroundColor: colors.bg.overlay,
171
+ backdropFilter: 'blur(8px)',
172
+ display: 'flex',
173
+ alignItems: 'center',
174
+ justifyContent: 'center',
175
+ zIndex: 50,
176
+ padding: 12,
177
+ overflowY: 'auto',
178
+ }}>
179
+ {/* spin keyframe provided by animations.css */}
180
+
181
+ {/* Dialog Container */}
182
+ <div
183
+ style={{
184
+ backgroundColor: colors.bg.secondary,
185
+ border: `1px solid ${colors.border.default}`,
186
+ borderRadius: 16,
187
+ padding: 24,
188
+ maxWidth: 448,
189
+ width: '100%',
190
+ maxHeight: 'calc(100vh - 24px)',
191
+ overflowY: 'auto',
192
+ boxShadow: '0 25px 50px -12px rgba(0, 0, 0, 0.5)',
193
+ }}
194
+ role="dialog"
195
+ aria-labelledby="recovery-title"
196
+ aria-describedby="recovery-description"
197
+ >
198
+ {/* Icon */}
199
+ <div style={{
200
+ width: 64,
201
+ height: 64,
202
+ margin: '0 auto 16px',
203
+ borderRadius: '50%',
204
+ backgroundColor: colors.status.warningSubtle,
205
+ display: 'flex',
206
+ alignItems: 'center',
207
+ justifyContent: 'center',
208
+ }}>
209
+ <svg
210
+ style={{ width: 32, height: 32, color: colors.status.warning }}
211
+ fill="none"
212
+ viewBox="0 0 24 24"
213
+ stroke="currentColor"
214
+ aria-hidden="true"
215
+ >
216
+ <path
217
+ strokeLinecap="round"
218
+ strokeLinejoin="round"
219
+ strokeWidth={2}
220
+ d="M12 9v2m0 4h.01m-6.938 4h13.856c1.54 0 2.502-1.667 1.732-3L13.732 4c-.77-1.333-2.694-1.333-3.464 0L3.34 16c-.77 1.333.192 3 1.732 3z"
221
+ />
222
+ </svg>
223
+ </div>
224
+
225
+ {/* Title */}
226
+ <h2
227
+ id="recovery-title"
228
+ style={{
229
+ fontSize: 20,
230
+ fontWeight: 600,
231
+ color: colors.text.primary,
232
+ textAlign: 'center',
233
+ marginBottom: 8,
234
+ }}
235
+ >
236
+ Recover Previous Session?
237
+ </h2>
238
+
239
+ {/* Description */}
240
+ <p
241
+ id="recovery-description"
242
+ style={{
243
+ color: colors.text.secondary,
244
+ textAlign: 'center',
245
+ marginBottom: 16,
246
+ fontSize: 14,
247
+ }}
248
+ >
249
+ markupr found an incomplete session from{' '}
250
+ <span style={{ color: colors.text.primary, fontWeight: 500 }}>{formattedTime} ago</span>.
251
+ </p>
252
+
253
+ {/* Session Info Card */}
254
+ <div style={{
255
+ backgroundColor: colors.bg.tertiary,
256
+ borderRadius: 8,
257
+ padding: 16,
258
+ marginBottom: 16,
259
+ }}>
260
+ {[
261
+ { label: 'Source', value: session.sourceName || 'Unknown' },
262
+ { label: 'Feedback items', value: String(session.feedbackItems.length) },
263
+ { label: 'Screenshots', value: String(session.screenshotCount) },
264
+ { label: 'Duration', value: formatDuration(sessionDuration) },
265
+ ].map((row, i, arr) => (
266
+ <div key={row.label} style={{
267
+ display: 'flex',
268
+ justifyContent: 'space-between',
269
+ fontSize: 14,
270
+ marginBottom: i < arr.length - 1 ? 8 : 0,
271
+ }}>
272
+ <span style={{ color: colors.text.secondary }}>{row.label}:</span>
273
+ <span style={{
274
+ color: colors.text.primary,
275
+ fontWeight: 500,
276
+ overflow: 'hidden',
277
+ textOverflow: 'ellipsis',
278
+ whiteSpace: 'nowrap',
279
+ marginLeft: 8,
280
+ maxWidth: 200,
281
+ }}>
282
+ {row.value}
283
+ </span>
284
+ </div>
285
+ ))}
286
+ </div>
287
+
288
+ {/* Expandable Details */}
289
+ <button
290
+ onClick={() => setShowDetails(!showDetails)}
291
+ style={{
292
+ width: '100%',
293
+ fontSize: 14,
294
+ color: colors.text.secondary,
295
+ display: 'flex',
296
+ alignItems: 'center',
297
+ justifyContent: 'center',
298
+ gap: 4,
299
+ marginBottom: 16,
300
+ background: 'none',
301
+ border: 'none',
302
+ cursor: 'pointer',
303
+ transition: 'color 150ms ease',
304
+ }}
305
+ >
306
+ <svg
307
+ style={{
308
+ width: 16,
309
+ height: 16,
310
+ transition: 'transform 200ms ease',
311
+ transform: showDetails ? 'rotate(180deg)' : 'rotate(0deg)',
312
+ }}
313
+ fill="none"
314
+ viewBox="0 0 24 24"
315
+ stroke="currentColor"
316
+ >
317
+ <path
318
+ strokeLinecap="round"
319
+ strokeLinejoin="round"
320
+ strokeWidth={2}
321
+ d="M19 9l-7 7-7-7"
322
+ />
323
+ </svg>
324
+ {showDetails ? 'Hide details' : 'Show details'}
325
+ </button>
326
+
327
+ {/* Expandable Content */}
328
+ {showDetails && (
329
+ <div style={{
330
+ backgroundColor: colors.surface.inset,
331
+ borderRadius: 8,
332
+ padding: 12,
333
+ marginBottom: 16,
334
+ fontSize: 12,
335
+ maxHeight: 192,
336
+ overflowY: 'auto',
337
+ display: 'grid',
338
+ gap: 8,
339
+ }}>
340
+ {[
341
+ { label: 'Session ID', value: `${session.id.slice(0, 8)}...`, mono: true },
342
+ { label: 'Started', value: formatDate(session.startTime) },
343
+ { label: 'Last saved', value: formatDate(session.lastSaveTime) },
344
+ ...(session.metadata ? [
345
+ { label: 'App version', value: session.metadata.appVersion },
346
+ { label: 'Platform', value: session.metadata.platform },
347
+ ] : []),
348
+ ].map((row) => (
349
+ <div key={row.label} style={{ display: 'flex', justifyContent: 'space-between' }}>
350
+ <span style={{ color: colors.text.tertiary }}>{row.label}:</span>
351
+ <span style={{
352
+ color: colors.text.secondary,
353
+ ...(row.mono ? { fontFamily: "'SF Mono', Menlo, Monaco, monospace" } : {}),
354
+ }}>
355
+ {row.value}
356
+ </span>
357
+ </div>
358
+ ))}
359
+
360
+ {/* Preview first few feedback items */}
361
+ {session.feedbackItems.length > 0 && (
362
+ <div style={{
363
+ marginTop: 12,
364
+ paddingTop: 12,
365
+ borderTop: `1px solid ${colors.border.default}`,
366
+ }}>
367
+ <p style={{ color: colors.text.tertiary, marginBottom: 8 }}>Recent feedback:</p>
368
+ <ul style={{ listStyle: 'none', padding: 0, margin: 0, display: 'grid', gap: 4 }}>
369
+ {session.feedbackItems.slice(0, 3).map((item, index) => (
370
+ <li key={item.id || index} style={{
371
+ color: colors.text.secondary,
372
+ overflow: 'hidden',
373
+ textOverflow: 'ellipsis',
374
+ whiteSpace: 'nowrap',
375
+ }}>
376
+ {item.hasScreenshot && (
377
+ <span style={{ color: colors.text.link, marginRight: 4 }} title="Has screenshot">
378
+ [img]
379
+ </span>
380
+ )}
381
+ {item.text || '[No text]'}
382
+ </li>
383
+ ))}
384
+ {session.feedbackItems.length > 3 && (
385
+ <li style={{ color: colors.text.tertiary, fontStyle: 'italic' }}>
386
+ ...and {session.feedbackItems.length - 3} more
387
+ </li>
388
+ )}
389
+ </ul>
390
+ </div>
391
+ )}
392
+ </div>
393
+ )}
394
+
395
+ {/* Warning about data loss */}
396
+ <div style={{
397
+ display: 'flex',
398
+ alignItems: 'flex-start',
399
+ gap: 8,
400
+ fontSize: 12,
401
+ color: colors.status.warning,
402
+ marginBottom: 24,
403
+ backgroundColor: colors.status.warningSubtle,
404
+ borderRadius: 8,
405
+ padding: 12,
406
+ }}>
407
+ <svg
408
+ style={{ width: 16, height: 16, flexShrink: 0, marginTop: 2 }}
409
+ fill="none"
410
+ viewBox="0 0 24 24"
411
+ stroke="currentColor"
412
+ >
413
+ <path
414
+ strokeLinecap="round"
415
+ strokeLinejoin="round"
416
+ strokeWidth={2}
417
+ d="M13 16h-1v-4h-1m1-4h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z"
418
+ />
419
+ </svg>
420
+ <span>
421
+ Discarding will permanently delete this session data.
422
+ Recovery will restore your feedback items for review.
423
+ </span>
424
+ </div>
425
+
426
+ {/* Actions */}
427
+ <div style={{ display: 'flex', gap: 12 }}>
428
+ <button
429
+ onClick={handleDiscard}
430
+ disabled={isRecovering || isDiscarding}
431
+ onMouseEnter={() => setHoveredBtn('discard')}
432
+ onMouseLeave={() => setHoveredBtn(null)}
433
+ style={{
434
+ flex: 1,
435
+ padding: '10px 16px',
436
+ borderRadius: 8,
437
+ fontWeight: 500,
438
+ fontSize: 14,
439
+ border: 'none',
440
+ cursor: isDiscarding ? 'wait' : (isRecovering ? 'not-allowed' : 'pointer'),
441
+ opacity: (isRecovering || isDiscarding) ? 0.5 : 1,
442
+ transition: 'all 200ms ease',
443
+ backgroundColor: isDiscarding
444
+ ? colors.bg.tertiary
445
+ : (hoveredBtn === 'discard' ? colors.bg.tertiary : 'transparent'),
446
+ color: isDiscarding
447
+ ? colors.text.secondary
448
+ : (hoveredBtn === 'discard' ? colors.text.primary : colors.text.secondary),
449
+ }}
450
+ >
451
+ {isDiscarding ? (
452
+ <span style={{ display: 'flex', alignItems: 'center', justifyContent: 'center', gap: 8 }}>
453
+ {spinnerSvg}
454
+ Discarding...
455
+ </span>
456
+ ) : (
457
+ <>
458
+ Discard
459
+ <span style={{ fontSize: 12, color: colors.text.tertiary, marginLeft: 4 }}>(D)</span>
460
+ </>
461
+ )}
462
+ </button>
463
+
464
+ <button
465
+ onClick={handleRecover}
466
+ disabled={isRecovering || isDiscarding}
467
+ onMouseEnter={() => setHoveredBtn('recover')}
468
+ onMouseLeave={() => setHoveredBtn(null)}
469
+ style={{
470
+ flex: 1,
471
+ padding: '10px 16px',
472
+ borderRadius: 8,
473
+ fontWeight: 500,
474
+ fontSize: 14,
475
+ border: 'none',
476
+ cursor: isRecovering ? 'wait' : (isDiscarding ? 'not-allowed' : 'pointer'),
477
+ opacity: (isRecovering || isDiscarding) ? 0.5 : 1,
478
+ transition: 'all 200ms ease',
479
+ backgroundColor: isRecovering
480
+ ? colors.accent.hover
481
+ : (hoveredBtn === 'recover' ? colors.accent.hover : colors.accent.default),
482
+ color: colors.text.inverse,
483
+ }}
484
+ >
485
+ {isRecovering ? (
486
+ <span style={{ display: 'flex', alignItems: 'center', justifyContent: 'center', gap: 8 }}>
487
+ {spinnerSvg}
488
+ Recovering...
489
+ </span>
490
+ ) : (
491
+ <>
492
+ Recover Session
493
+ <span style={{ fontSize: 12, marginLeft: 4, opacity: 0.8 }}>(R)</span>
494
+ </>
495
+ )}
496
+ </button>
497
+ </div>
498
+
499
+ {/* Keyboard hint */}
500
+ <p style={{ textAlign: 'center', fontSize: 12, color: colors.text.tertiary, marginTop: 16 }}>
501
+ Press <kbd style={{
502
+ padding: '2px 6px',
503
+ backgroundColor: colors.bg.tertiary,
504
+ borderRadius: 4,
505
+ color: colors.text.secondary,
506
+ }}>Enter</kbd> to recover or{' '}
507
+ <kbd style={{
508
+ padding: '2px 6px',
509
+ backgroundColor: colors.bg.tertiary,
510
+ borderRadius: 4,
511
+ color: colors.text.secondary,
512
+ }}>Esc</kbd> to discard
513
+ </p>
514
+ </div>
515
+ </div>
516
+ );
517
+ }
518
+
519
+ // ============================================================================
520
+ // Hook for managing crash recovery state
521
+ // ============================================================================
522
+
523
+ export interface UseCrashRecoveryReturn {
524
+ incompleteSession: RecoverableSession | null;
525
+ isCheckingRecovery: boolean;
526
+ recoverSession: () => void;
527
+ discardSession: () => void;
528
+ clearRecoveryState: () => void;
529
+ }
530
+
531
+ export function useCrashRecovery(): UseCrashRecoveryReturn {
532
+ const [incompleteSession, setIncompleteSession] =
533
+ useState<RecoverableSession | null>(null);
534
+ const [isCheckingRecovery, setIsCheckingRecovery] = useState(true);
535
+
536
+ useEffect(() => {
537
+ let unsubscribe: (() => void) | undefined;
538
+
539
+ // Check for incomplete session on mount
540
+ const checkForIncompleteSession = async () => {
541
+ try {
542
+ // Listen for crash recovery notification from main process
543
+ unsubscribe = window.markupr?.crashRecovery.onIncompleteFound(
544
+ (data) => {
545
+ if (data.session) {
546
+ // Map from IPC type to local type
547
+ setIncompleteSession({
548
+ id: data.session.id,
549
+ startTime: data.session.startTime,
550
+ lastSaveTime: data.session.lastSaveTime,
551
+ feedbackItems: data.session.feedbackItems.map((item) => ({
552
+ id: item.id,
553
+ timestamp: item.timestamp,
554
+ text: item.text,
555
+ confidence: item.confidence,
556
+ hasScreenshot: item.hasScreenshot,
557
+ })),
558
+ transcriptionBuffer: '',
559
+ sourceId: '',
560
+ sourceName: data.session.sourceName,
561
+ screenshotCount: data.session.screenshotCount,
562
+ });
563
+ }
564
+ }
565
+ );
566
+
567
+ // Request check from main process
568
+ const result = await window.markupr?.crashRecovery.check();
569
+
570
+ if (result?.session) {
571
+ // Map from IPC type to local type
572
+ setIncompleteSession({
573
+ id: result.session.id,
574
+ startTime: result.session.startTime,
575
+ lastSaveTime: result.session.lastSaveTime,
576
+ feedbackItems: result.session.feedbackItems.map((item) => ({
577
+ id: item.id,
578
+ timestamp: item.timestamp,
579
+ text: item.text,
580
+ confidence: item.confidence,
581
+ hasScreenshot: item.hasScreenshot,
582
+ screenshotId: item.screenshotId,
583
+ })),
584
+ transcriptionBuffer: '',
585
+ sourceId: '',
586
+ sourceName: result.session.sourceName,
587
+ screenshotCount: result.session.screenshotCount,
588
+ metadata: result.session.metadata,
589
+ });
590
+ }
591
+ } catch (error) {
592
+ console.error('Failed to check for incomplete session:', error);
593
+ } finally {
594
+ setIsCheckingRecovery(false);
595
+ }
596
+ };
597
+
598
+ checkForIncompleteSession();
599
+
600
+ // Cleanup listener on unmount
601
+ return () => {
602
+ unsubscribe?.();
603
+ };
604
+ }, []);
605
+
606
+ const recoverSession = async () => {
607
+ if (incompleteSession) {
608
+ await window.markupr?.crashRecovery.recover(incompleteSession.id);
609
+ setIncompleteSession(null);
610
+ }
611
+ };
612
+
613
+ const discardSession = async () => {
614
+ await window.markupr?.crashRecovery.discard();
615
+ setIncompleteSession(null);
616
+ };
617
+
618
+ const clearRecoveryState = () => {
619
+ setIncompleteSession(null);
620
+ };
621
+
622
+ return {
623
+ incompleteSession,
624
+ isCheckingRecovery,
625
+ recoverSession,
626
+ discardSession,
627
+ clearRecoveryState,
628
+ };
629
+ }
630
+
631
+ // Default export for convenience
632
+ export default CrashRecoveryDialog;