appclean 1.8.0 → 2.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (211) hide show
  1. package/GUI_IMPLEMENTATION_STATUS.md +143 -0
  2. package/MD_Files/INDEX.md +51 -0
  3. package/MD_Files/PUBLICATION_SUCCESS_REPORT.md +227 -0
  4. package/PHASE2_COMPLETION.md +281 -0
  5. package/PHASE3_COMPLETION.md +364 -0
  6. package/README.md +446 -376
  7. package/assets/logo.svg +34 -0
  8. package/dist/core/appUpdateChecker.js +12 -16
  9. package/dist/core/appUpdateChecker.js.map +1 -1
  10. package/dist/core/detector.js +14 -18
  11. package/dist/core/detector.js.map +1 -1
  12. package/dist/core/duplicateFileFinder.js +12 -19
  13. package/dist/core/duplicateFileFinder.js.map +1 -1
  14. package/dist/core/orphanedDependencyDetector.js +19 -26
  15. package/dist/core/orphanedDependencyDetector.js.map +1 -1
  16. package/dist/core/performanceOptimizer.js +6 -10
  17. package/dist/core/performanceOptimizer.js.map +1 -1
  18. package/dist/core/permissionHandler.js +21 -25
  19. package/dist/core/permissionHandler.js.map +1 -1
  20. package/dist/core/pluginSystem.js +9 -13
  21. package/dist/core/pluginSystem.js.map +1 -1
  22. package/dist/core/removalRecorder.js +12 -19
  23. package/dist/core/removalRecorder.js.map +1 -1
  24. package/dist/core/remover.js +59 -66
  25. package/dist/core/remover.js.map +1 -1
  26. package/dist/core/reportGenerator.d.ts +1 -1
  27. package/dist/core/reportGenerator.d.ts.map +1 -1
  28. package/dist/core/reportGenerator.js +27 -34
  29. package/dist/core/reportGenerator.js.map +1 -1
  30. package/dist/core/scheduledCleanup.js +23 -30
  31. package/dist/core/scheduledCleanup.js.map +1 -1
  32. package/dist/core/serviceFileDetector.js +24 -31
  33. package/dist/core/serviceFileDetector.js.map +1 -1
  34. package/dist/core/verificationModule.js +10 -14
  35. package/dist/core/verificationModule.js.map +1 -1
  36. package/dist/index.js +190 -90
  37. package/dist/index.js.map +1 -1
  38. package/dist/managers/brewManager.d.ts.map +1 -1
  39. package/dist/managers/brewManager.js +35 -41
  40. package/dist/managers/brewManager.js.map +1 -1
  41. package/dist/managers/customManager.d.ts +2 -1
  42. package/dist/managers/customManager.d.ts.map +1 -1
  43. package/dist/managers/customManager.js +79 -53
  44. package/dist/managers/customManager.js.map +1 -1
  45. package/dist/managers/linuxManager.js +29 -36
  46. package/dist/managers/linuxManager.js.map +1 -1
  47. package/dist/managers/npmManager.js +27 -34
  48. package/dist/managers/npmManager.js.map +1 -1
  49. package/dist/types/index.js +1 -2
  50. package/dist/ui/client/api/client.d.ts +24 -0
  51. package/dist/ui/client/api/client.d.ts.map +1 -0
  52. package/dist/ui/client/api/client.js +96 -0
  53. package/dist/ui/client/api/client.js.map +1 -0
  54. package/dist/ui/client/app.d.ts +7 -0
  55. package/dist/ui/client/app.d.ts.map +1 -0
  56. package/dist/ui/client/app.js +71 -0
  57. package/dist/ui/client/app.js.map +1 -0
  58. package/dist/ui/client/index.html +107 -0
  59. package/dist/ui/client/pages/appDetails.d.ts +8 -0
  60. package/dist/ui/client/pages/appDetails.d.ts.map +1 -0
  61. package/dist/ui/client/pages/appDetails.js +287 -0
  62. package/dist/ui/client/pages/appDetails.js.map +1 -0
  63. package/dist/ui/client/pages/appSearch.d.ts +2 -0
  64. package/dist/ui/client/pages/appSearch.d.ts.map +1 -0
  65. package/dist/ui/client/pages/appSearch.js +210 -0
  66. package/dist/ui/client/pages/appSearch.js.map +1 -0
  67. package/dist/ui/client/pages/dashboard.d.ts +2 -0
  68. package/dist/ui/client/pages/dashboard.d.ts.map +1 -0
  69. package/dist/ui/client/pages/dashboard.js +154 -0
  70. package/dist/ui/client/pages/dashboard.js.map +1 -0
  71. package/dist/ui/client/pages/settings.d.ts +7 -0
  72. package/dist/ui/client/pages/settings.d.ts.map +1 -0
  73. package/dist/ui/client/pages/settings.js +279 -0
  74. package/dist/ui/client/pages/settings.js.map +1 -0
  75. package/dist/ui/client/state/appStore.d.ts +38 -0
  76. package/dist/ui/client/state/appStore.d.ts.map +1 -0
  77. package/dist/ui/client/state/appStore.js +121 -0
  78. package/dist/ui/client/state/appStore.js.map +1 -0
  79. package/dist/ui/client/state/dashboardStore.d.ts +31 -0
  80. package/dist/ui/client/state/dashboardStore.d.ts.map +1 -0
  81. package/dist/ui/client/state/dashboardStore.js +70 -0
  82. package/dist/ui/client/state/dashboardStore.js.map +1 -0
  83. package/dist/ui/client/state/uiStore.d.ts +43 -0
  84. package/dist/ui/client/state/uiStore.d.ts.map +1 -0
  85. package/dist/ui/client/state/uiStore.js +109 -0
  86. package/dist/ui/client/state/uiStore.js.map +1 -0
  87. package/dist/ui/client/styles/animations.css +327 -0
  88. package/dist/ui/client/styles/base.css +214 -0
  89. package/dist/ui/client/styles/components.css +400 -0
  90. package/dist/ui/client/styles/layout.css +224 -0
  91. package/dist/ui/client/styles/variables.css +140 -0
  92. package/dist/ui/client/utils/events.d.ts +19 -0
  93. package/dist/ui/client/utils/events.d.ts.map +1 -0
  94. package/dist/ui/client/utils/events.js +54 -0
  95. package/dist/ui/client/utils/events.js.map +1 -0
  96. package/dist/ui/client/utils/formatting.d.ts +11 -0
  97. package/dist/ui/client/utils/formatting.d.ts.map +1 -0
  98. package/dist/ui/client/utils/formatting.js +104 -0
  99. package/dist/ui/client/utils/formatting.js.map +1 -0
  100. package/dist/ui/client/utils/router.d.ts +25 -0
  101. package/dist/ui/client/utils/router.d.ts.map +1 -0
  102. package/dist/ui/client/utils/router.js +90 -0
  103. package/dist/ui/client/utils/router.js.map +1 -0
  104. package/dist/ui/guiServer.d.ts +8 -1
  105. package/dist/ui/guiServer.d.ts.map +1 -1
  106. package/dist/ui/guiServer.js +148 -110
  107. package/dist/ui/guiServer.js.map +1 -1
  108. package/dist/ui/menu.js +18 -27
  109. package/dist/ui/menu.js.map +1 -1
  110. package/dist/ui/prompts.js +34 -47
  111. package/dist/ui/prompts.js.map +1 -1
  112. package/dist/ui/server/middleware/errorHandler.d.ts +19 -0
  113. package/dist/ui/server/middleware/errorHandler.d.ts.map +1 -0
  114. package/dist/ui/server/middleware/errorHandler.js +100 -0
  115. package/dist/ui/server/middleware/errorHandler.js.map +1 -0
  116. package/dist/ui/server/routes/apps.d.ts +8 -0
  117. package/dist/ui/server/routes/apps.d.ts.map +1 -0
  118. package/dist/ui/server/routes/apps.js +74 -0
  119. package/dist/ui/server/routes/apps.js.map +1 -0
  120. package/dist/ui/server/routes/dashboard.d.ts +4 -0
  121. package/dist/ui/server/routes/dashboard.d.ts.map +1 -0
  122. package/dist/ui/server/routes/dashboard.js +57 -0
  123. package/dist/ui/server/routes/dashboard.js.map +1 -0
  124. package/dist/ui/server/routes/settings.d.ts +6 -0
  125. package/dist/ui/server/routes/settings.d.ts.map +1 -0
  126. package/dist/ui/server/routes/settings.js +31 -0
  127. package/dist/ui/server/routes/settings.js.map +1 -0
  128. package/dist/ui/server/services/appService.d.ts +45 -0
  129. package/dist/ui/server/services/appService.d.ts.map +1 -0
  130. package/dist/ui/server/services/appService.js +114 -0
  131. package/dist/ui/server/services/appService.js.map +1 -0
  132. package/dist/ui/server/services/removalService.d.ts +24 -0
  133. package/dist/ui/server/services/removalService.d.ts.map +1 -0
  134. package/dist/ui/server/services/removalService.js +83 -0
  135. package/dist/ui/server/services/removalService.js.map +1 -0
  136. package/dist/utils/filesystem.js +32 -49
  137. package/dist/utils/filesystem.js.map +1 -1
  138. package/dist/utils/logger.js +9 -18
  139. package/dist/utils/logger.js.map +1 -1
  140. package/dist/utils/platform.js +10 -22
  141. package/dist/utils/platform.js.map +1 -1
  142. package/dist/utils/upgrade.d.ts +22 -0
  143. package/dist/utils/upgrade.d.ts.map +1 -0
  144. package/dist/utils/upgrade.js +94 -0
  145. package/dist/utils/upgrade.js.map +1 -0
  146. package/package.json +4 -2
  147. package/src/core/appUpdateChecker.ts +1 -1
  148. package/src/core/detector.ts +6 -6
  149. package/src/core/duplicateFileFinder.ts +1 -1
  150. package/src/core/orphanedDependencyDetector.ts +2 -2
  151. package/src/core/performanceOptimizer.ts +1 -1
  152. package/src/core/permissionHandler.ts +2 -2
  153. package/src/core/pluginSystem.ts +1 -1
  154. package/src/core/removalRecorder.ts +2 -2
  155. package/src/core/remover.ts +11 -11
  156. package/src/core/reportGenerator.ts +2 -2
  157. package/src/core/scheduledCleanup.ts +2 -2
  158. package/src/core/serviceFileDetector.ts +2 -2
  159. package/src/core/verificationModule.ts +2 -2
  160. package/src/index.ts +133 -6
  161. package/src/managers/brewManager.ts +11 -9
  162. package/src/managers/customManager.ts +71 -30
  163. package/src/managers/linuxManager.ts +3 -3
  164. package/src/managers/npmManager.ts +3 -3
  165. package/src/ui/client/api/client.ts +163 -0
  166. package/src/ui/client/app.ts +121 -0
  167. package/src/ui/client/index.html +107 -0
  168. package/src/ui/client/pages/appDetails.ts +356 -0
  169. package/src/ui/client/pages/appSearch.ts +270 -0
  170. package/src/ui/client/pages/dashboard.ts +189 -0
  171. package/src/ui/client/pages/settings.ts +342 -0
  172. package/src/ui/client/state/appStore.ts +169 -0
  173. package/src/ui/client/state/dashboardStore.ts +113 -0
  174. package/src/ui/client/state/uiStore.ts +166 -0
  175. package/src/ui/client/styles/animations.css +327 -0
  176. package/src/ui/client/styles/base.css +214 -0
  177. package/src/ui/client/styles/components.css +400 -0
  178. package/src/ui/client/styles/layout.css +224 -0
  179. package/src/ui/client/styles/variables.css +140 -0
  180. package/src/ui/client/utils/events.ts +74 -0
  181. package/src/ui/client/utils/formatting.ts +157 -0
  182. package/src/ui/client/utils/router.ts +161 -0
  183. package/src/ui/guiServer.ts +206 -105
  184. package/src/ui/prompts.ts +1 -1
  185. package/src/ui/server/middleware/errorHandler.ts +174 -0
  186. package/src/ui/server/routes/apps.ts +132 -0
  187. package/src/ui/server/routes/dashboard.ts +93 -0
  188. package/src/ui/server/routes/settings.ts +63 -0
  189. package/src/ui/server/services/appService.ts +184 -0
  190. package/src/ui/server/services/removalService.ts +138 -0
  191. package/src/utils/upgrade.ts +143 -0
  192. package/tsconfig.json +3 -2
  193. package/INDEX.md +0 -165
  194. /package/{ACTION_CHECKLIST.md → MD_Files/ACTION_CHECKLIST.md} +0 -0
  195. /package/{APPCLEAN_SUMMARY.md → MD_Files/APPCLEAN_SUMMARY.md} +0 -0
  196. /package/{CHANGELOG.md → MD_Files/CHANGELOG.md} +0 -0
  197. /package/{CODE_OF_CONDUCT.md → MD_Files/CODE_OF_CONDUCT.md} +0 -0
  198. /package/{CODE_REVIEW_REPORT.md → MD_Files/CODE_REVIEW_REPORT.md} +0 -0
  199. /package/{COMMUNITY_POSTS.md → MD_Files/COMMUNITY_POSTS.md} +0 -0
  200. /package/{DEPLOYMENT_GUIDE.md → MD_Files/DEPLOYMENT_GUIDE.md} +0 -0
  201. /package/{DEPLOYMENT_STATUS.md → MD_Files/DEPLOYMENT_STATUS.md} +0 -0
  202. /package/{EXECUTIVE_REPORT.md → MD_Files/EXECUTIVE_REPORT.md} +0 -0
  203. /package/{GITHUB_OPTIMIZATION.md → MD_Files/GITHUB_OPTIMIZATION.md} +0 -0
  204. /package/{MARKETING_SUMMARY.md → MD_Files/MARKETING_SUMMARY.md} +0 -0
  205. /package/{NPM_PACKAGE_OPTIMIZATION.md → MD_Files/NPM_PACKAGE_OPTIMIZATION.md} +0 -0
  206. /package/{NPM_PUBLISH.md → MD_Files/NPM_PUBLISH.md} +0 -0
  207. /package/{PROJECT_SUMMARY.txt → MD_Files/PROJECT_SUMMARY.txt} +0 -0
  208. /package/{QUICKSTART.md → MD_Files/QUICKSTART.md} +0 -0
  209. /package/{SETUP_GITHUB.md → MD_Files/SETUP_GITHUB.md} +0 -0
  210. /package/{TESTING_SUMMARY.md → MD_Files/TESTING_SUMMARY.md} +0 -0
  211. /package/{setup-github.sh → MD_Files/setup-github.sh} +0 -0
@@ -0,0 +1,189 @@
1
+ /**
2
+ * Dashboard Page - Overview of system health and app statistics
3
+ */
4
+
5
+ import { dashboardStore, DashboardStats } from '../state/dashboardStore.js';
6
+ import { uiStore } from '../state/uiStore.js';
7
+ import { formatBytes, formatPercent, formatRelativeTime } from '../utils/formatting.js';
8
+
9
+ /**
10
+ * Render Dashboard Page
11
+ */
12
+ export function renderDashboard(): void {
13
+ const container = document.getElementById('page-container');
14
+ if (!container) return;
15
+
16
+ // Show loading state
17
+ container.innerHTML = `
18
+ <div class="loading-state" style="text-align: center; padding: 60px 20px;">
19
+ <div class="spinner-lg spinner"></div>
20
+ <p class="text-center text-muted mt-4">Loading dashboard...</p>
21
+ </div>
22
+ `;
23
+
24
+ // Load stats
25
+ dashboardStore.refreshStats(true).then(() => {
26
+ const state = dashboardStore.getState();
27
+ const stats = state.stats;
28
+
29
+ if (!stats) {
30
+ container.innerHTML = `
31
+ <div class="alert alert-danger">
32
+ <span>Failed to load dashboard stats</span>
33
+ </div>
34
+ `;
35
+ return;
36
+ }
37
+
38
+ renderDashboardContent(container, stats);
39
+
40
+ // Subscribe to updates
41
+ dashboardStore.subscribe((newState) => {
42
+ if (newState.stats) {
43
+ renderDashboardContent(container, newState.stats);
44
+ }
45
+ });
46
+ });
47
+ }
48
+
49
+ /**
50
+ * Render dashboard content
51
+ */
52
+ function renderDashboardContent(container: HTMLElement, stats: DashboardStats): void {
53
+ container.innerHTML = `
54
+ <div class="dashboard-page">
55
+ <!-- Header -->
56
+ <div class="page-header mb-8">
57
+ <h1 class="text-3xl font-bold">📊 Dashboard</h1>
58
+ <p class="text-secondary mt-2">System overview and app management</p>
59
+ </div>
60
+
61
+ <!-- Stats Grid -->
62
+ <div class="grid grid-cols-1 md:grid-cols-2 gap-6 mb-8">
63
+ ${renderStatCard('📦 Installed Apps', stats.totalApps.toString(), 'Total applications detected')}
64
+ ${renderStatCard('💾 Total Space Used', formatBytes(stats.totalSpaceUsed), 'Size of all app files')}
65
+ ${renderStatCard('🗑️ Removed (Session)', stats.sessionAppsRemoved.toString(), 'Apps removed this session')}
66
+ ${renderStatCard('📉 Space Freed', formatBytes(stats.sessionSpaceFreed), 'Total space recovered')}
67
+ </div>
68
+
69
+ <!-- System Health -->
70
+ <div class="card mb-8">
71
+ <div class="card-header">
72
+ <h2 class="text-xl font-bold">💻 System Health</h2>
73
+ </div>
74
+ <div class="card-body">
75
+ <div class="health-gauge mb-4">
76
+ <div class="flex-between mb-2">
77
+ <span class="text-sm font-medium">Disk Usage</span>
78
+ <span class="text-sm font-bold">${formatPercent(stats.diskUsagePercent)}</span>
79
+ </div>
80
+ <div class="progress-bar" style="height: 8px; background: #e5e7eb; border-radius: 4px; overflow: hidden;">
81
+ <div style="
82
+ height: 100%;
83
+ width: ${stats.diskUsagePercent}%;
84
+ background: ${getDiskUsageColor(stats.diskUsagePercent)};
85
+ transition: width 0.3s ease;
86
+ "></div>
87
+ </div>
88
+ </div>
89
+ <p class="text-sm text-muted">
90
+ ${stats.diskUsagePercent < 80
91
+ ? '✓ Disk usage is healthy'
92
+ : stats.diskUsagePercent < 90
93
+ ? '⚠️ Disk usage is getting high'
94
+ : '❌ Disk usage is critical'}
95
+ </p>
96
+ </div>
97
+ </div>
98
+
99
+ <!-- Recent Activity -->
100
+ <div class="card">
101
+ <div class="card-header">
102
+ <h2 class="text-xl font-bold">📜 Recent Activity</h2>
103
+ </div>
104
+ <div class="card-body">
105
+ ${
106
+ stats.recentlyRemoved.length > 0
107
+ ? `
108
+ <div class="activity-list">
109
+ ${stats.recentlyRemoved
110
+ .slice(0, 10)
111
+ .map(
112
+ (removal) => `
113
+ <div class="activity-item flex-between p-3 border-b border-color last:border-0" style="padding: 12px 0;">
114
+ <div class="activity-info">
115
+ <p class="text-sm font-medium">${escapeHtml(removal.appName)}</p>
116
+ <p class="text-xs text-muted">Removed ${formatRelativeTime(removal.timestamp)}</p>
117
+ </div>
118
+ <div class="activity-stats text-right">
119
+ <p class="text-sm font-bold text-success">${formatBytes(removal.freedSpace)}</p>
120
+ <p class="text-xs text-muted">${removal.filesRemoved} files</p>
121
+ </div>
122
+ </div>
123
+ `
124
+ )
125
+ .join('')}
126
+ </div>
127
+ `
128
+ : `
129
+ <p class="text-muted text-center py-8">
130
+ No apps removed yet. Start by searching for apps to remove.
131
+ </p>
132
+ `
133
+ }
134
+ </div>
135
+ </div>
136
+
137
+ <!-- Quick Actions -->
138
+ <div class="mt-8 grid grid-cols-1 md:grid-cols-2 gap-4">
139
+ <button class="btn btn-primary btn-lg w-full" onclick="window.location.hash = '#/apps'">
140
+ <span>🔍 Find & Remove Apps</span>
141
+ </button>
142
+ <button class="btn btn-secondary btn-lg w-full" onclick="window.location.hash = '#/settings'">
143
+ <span>⚙️ Settings</span>
144
+ </button>
145
+ </div>
146
+ </div>
147
+ `;
148
+
149
+ // Scroll to top
150
+ window.scrollTo(0, 0);
151
+ }
152
+
153
+ /**
154
+ * Render stat card
155
+ */
156
+ function renderStatCard(label: string, value: string, description: string): string {
157
+ return `
158
+ <div class="card">
159
+ <div class="card-body">
160
+ <p class="text-muted text-sm mb-2">${escapeHtml(label)}</p>
161
+ <p class="text-2xl font-bold text-primary mb-1">${escapeHtml(value)}</p>
162
+ <p class="text-xs text-muted">${escapeHtml(description)}</p>
163
+ </div>
164
+ </div>
165
+ `;
166
+ }
167
+
168
+ /**
169
+ * Get disk usage color
170
+ */
171
+ function getDiskUsageColor(percent: number): string {
172
+ if (percent < 50) return '#10b981'; // Green
173
+ if (percent < 80) return '#f59e0b'; // Amber
174
+ return '#ef4444'; // Red
175
+ }
176
+
177
+ /**
178
+ * Escape HTML to prevent XSS
179
+ */
180
+ function escapeHtml(text: string): string {
181
+ const map: Record<string, string> = {
182
+ '&': '&amp;',
183
+ '<': '&lt;',
184
+ '>': '&gt;',
185
+ '"': '&quot;',
186
+ "'": '&#039;',
187
+ };
188
+ return text.replace(/[&<>"']/g, (m) => map[m]);
189
+ }
@@ -0,0 +1,342 @@
1
+ /**
2
+ * Settings Page - Version management, theme, and uninstall
3
+ */
4
+
5
+ import { apiClient } from '../api/client.js';
6
+ import { uiStore } from '../state/uiStore.js';
7
+
8
+ export interface VersionInfo {
9
+ current: string;
10
+ latest: string;
11
+ isUpdateAvailable: boolean;
12
+ }
13
+
14
+ /**
15
+ * Render Settings Page
16
+ */
17
+ export function renderSettings(): void {
18
+ const container = document.getElementById('page-container');
19
+ if (!container) return;
20
+
21
+ // Show loading state
22
+ container.innerHTML = `
23
+ <div class="loading-state" style="text-align: center; padding: 60px 20px;">
24
+ <div class="spinner-lg spinner"></div>
25
+ <p class="text-center text-muted mt-4">Loading settings...</p>
26
+ </div>
27
+ `;
28
+
29
+ // Load version info
30
+ loadVersionInfo().then((versionInfo) => {
31
+ renderSettingsContent(container, versionInfo);
32
+ });
33
+ }
34
+
35
+ /**
36
+ * Load version information
37
+ */
38
+ async function loadVersionInfo(): Promise<VersionInfo> {
39
+ try {
40
+ return await apiClient.get<VersionInfo>('/api/version');
41
+ } catch (error) {
42
+ return {
43
+ current: 'unknown',
44
+ latest: 'unknown',
45
+ isUpdateAvailable: false,
46
+ };
47
+ }
48
+ }
49
+
50
+ /**
51
+ * Render settings content
52
+ */
53
+ function renderSettingsContent(container: HTMLElement, versionInfo: VersionInfo): void {
54
+ const currentTheme = uiStore.getState().theme;
55
+
56
+ container.innerHTML = `
57
+ <div class="settings-page">
58
+ <!-- Header -->
59
+ <div class="page-header mb-8">
60
+ <h1 class="text-3xl font-bold">⚙️ Settings</h1>
61
+ <p class="text-secondary mt-2">Configure AppClean</p>
62
+ </div>
63
+
64
+ <!-- Version & Updates Section -->
65
+ <div class="card mb-6">
66
+ <div class="card-header">
67
+ <h2 class="text-xl font-bold">📦 Version & Updates</h2>
68
+ </div>
69
+ <div class="card-body">
70
+ <div class="version-info p-4 bg-secondary rounded mb-4">
71
+ <div class="flex-between mb-3">
72
+ <span class="text-sm font-medium">Current Version</span>
73
+ <span class="text-lg font-bold text-primary">v${escapeHtml(versionInfo.current)}</span>
74
+ </div>
75
+ <div class="flex-between">
76
+ <span class="text-sm font-medium">Latest Version</span>
77
+ <span class="text-lg font-bold ${versionInfo.isUpdateAvailable ? 'text-warning' : 'text-success'}">
78
+ v${escapeHtml(versionInfo.latest)}
79
+ </span>
80
+ </div>
81
+ </div>
82
+
83
+ ${
84
+ versionInfo.isUpdateAvailable
85
+ ? `
86
+ <div class="alert alert-warning mb-4">
87
+ <span>A new version of AppClean is available!</span>
88
+ </div>
89
+ <div class="flex gap-2">
90
+ <button class="btn btn-primary flex-1" id="upgrade-btn">
91
+ 🚀 Upgrade Now
92
+ </button>
93
+ <button class="btn btn-secondary flex-1" id="check-update-btn">
94
+ 🔄 Check Again
95
+ </button>
96
+ </div>
97
+ `
98
+ : `
99
+ <div class="alert alert-success mb-4">
100
+ <span>✓ You're running the latest version!</span>
101
+ </div>
102
+ <button class="btn btn-secondary w-full" id="check-update-btn">
103
+ 🔄 Check for Updates
104
+ </button>
105
+ `
106
+ }
107
+ </div>
108
+ </div>
109
+
110
+ <!-- Theme Section -->
111
+ <div class="card mb-6">
112
+ <div class="card-header">
113
+ <h2 class="text-xl font-bold">🎨 Appearance</h2>
114
+ </div>
115
+ <div class="card-body">
116
+ <div class="flex-between items-center p-3 border border-color rounded mb-4">
117
+ <div>
118
+ <p class="font-medium">Dark Mode</p>
119
+ <p class="text-sm text-muted">Switch between light and dark themes</p>
120
+ </div>
121
+ <button
122
+ class="btn btn-sm ${currentTheme === 'dark' ? 'btn-primary' : 'btn-secondary'}"
123
+ id="theme-toggle-btn"
124
+ >
125
+ ${currentTheme === 'dark' ? '🌙 Dark' : '☀️ Light'}
126
+ </button>
127
+ </div>
128
+ </div>
129
+ </div>
130
+
131
+ <!-- About Section -->
132
+ <div class="card mb-6">
133
+ <div class="card-header">
134
+ <h2 class="text-xl font-bold">ℹ️ About</h2>
135
+ </div>
136
+ <div class="card-body">
137
+ <div class="space-y-4">
138
+ <div>
139
+ <p class="text-sm font-medium mb-2">AppClean</p>
140
+ <p class="text-sm text-muted">
141
+ Intelligently find and safely remove applications with all their artifacts.
142
+ </p>
143
+ </div>
144
+
145
+ <div>
146
+ <p class="text-sm font-medium mb-2">Links</p>
147
+ <div class="flex gap-2">
148
+ <a
149
+ href="https://github.com/praveenkay/AppClean"
150
+ target="_blank"
151
+ rel="noopener"
152
+ class="btn btn-sm btn-ghost"
153
+ >
154
+ 💻 GitHub
155
+ </a>
156
+ <a
157
+ href="https://github.com/praveenkay/AppClean/issues"
158
+ target="_blank"
159
+ rel="noopener"
160
+ class="btn btn-sm btn-ghost"
161
+ >
162
+ 🐛 Report Issue
163
+ </a>
164
+ </div>
165
+ </div>
166
+ </div>
167
+ </div>
168
+ </div>
169
+
170
+ <!-- Danger Zone -->
171
+ <div class="card" style="border-color: var(--color-danger);">
172
+ <div class="card-header" style="background: var(--color-danger-light);">
173
+ <h2 class="text-xl font-bold text-danger">⚠️ Danger Zone</h2>
174
+ </div>
175
+ <div class="card-body">
176
+ <p class="text-sm text-muted mb-4">
177
+ Uninstall AppClean from your system. This action cannot be undone.
178
+ </p>
179
+ <button class="btn btn-danger w-full" id="uninstall-btn">
180
+ 🗑️ Uninstall AppClean
181
+ </button>
182
+ </div>
183
+ </div>
184
+ </div>
185
+ `;
186
+
187
+ // Setup event listeners
188
+ setupSettingsListeners();
189
+
190
+ // Scroll to top
191
+ window.scrollTo(0, 0);
192
+ }
193
+
194
+ /**
195
+ * Setup event listeners
196
+ */
197
+ function setupSettingsListeners(): void {
198
+ const upgradeBtn = document.getElementById('upgrade-btn');
199
+ const checkUpdateBtn = document.getElementById('check-update-btn');
200
+ const themeToggleBtn = document.getElementById('theme-toggle-btn');
201
+ const uninstallBtn = document.getElementById('uninstall-btn');
202
+
203
+ // Upgrade button
204
+ if (upgradeBtn) {
205
+ upgradeBtn.addEventListener('click', async () => {
206
+ (upgradeBtn as HTMLButtonElement).disabled = true;
207
+ upgradeBtn.textContent = '⏳ Upgrading...';
208
+
209
+ try {
210
+ const result = await apiClient.post('/api/upgrade', {});
211
+
212
+ if (result.success) {
213
+ uiStore.showSuccess(result.message);
214
+ setTimeout(() => {
215
+ window.location.reload();
216
+ }, 2000);
217
+ } else {
218
+ uiStore.showError(result.message);
219
+ (upgradeBtn as HTMLButtonElement).disabled = false;
220
+ upgradeBtn.textContent = '🚀 Upgrade Now';
221
+ }
222
+ } catch (error) {
223
+ uiStore.showError(`Upgrade failed: ${(error as Error).message}`);
224
+ (upgradeBtn as HTMLButtonElement).disabled = false;
225
+ upgradeBtn.textContent = '🚀 Upgrade Now';
226
+ }
227
+ });
228
+ }
229
+
230
+ // Check update button
231
+ if (checkUpdateBtn) {
232
+ checkUpdateBtn.addEventListener('click', async () => {
233
+ (checkUpdateBtn as HTMLButtonElement).disabled = true;
234
+ checkUpdateBtn.textContent = '⏳ Checking...';
235
+
236
+ try {
237
+ const versionInfo = await apiClient.get<VersionInfo>('/api/version');
238
+
239
+ if (versionInfo.isUpdateAvailable) {
240
+ uiStore.showWarning(
241
+ `Update available: v${versionInfo.latest} (current: v${versionInfo.current})`
242
+ );
243
+ } else {
244
+ uiStore.showSuccess('You are running the latest version!');
245
+ }
246
+
247
+ // Refresh page to update UI
248
+ setTimeout(() => {
249
+ window.location.reload();
250
+ }, 1500);
251
+ } catch (error) {
252
+ uiStore.showError(`Failed to check updates: ${(error as Error).message}`);
253
+ (checkUpdateBtn as HTMLButtonElement).disabled = false;
254
+ checkUpdateBtn.textContent = '🔄 Check Again';
255
+ }
256
+ });
257
+ }
258
+
259
+ // Theme toggle button
260
+ if (themeToggleBtn) {
261
+ themeToggleBtn.addEventListener('click', () => {
262
+ uiStore.toggleTheme();
263
+
264
+ // Update button appearance
265
+ const newTheme = uiStore.getState().theme;
266
+ themeToggleBtn.className = `btn btn-sm ${newTheme === 'dark' ? 'btn-primary' : 'btn-secondary'}`;
267
+ themeToggleBtn.textContent = newTheme === 'dark' ? '🌙 Dark' : '☀️ Light';
268
+ });
269
+ }
270
+
271
+ // Uninstall button
272
+ if (uninstallBtn) {
273
+ uninstallBtn.addEventListener('click', () => {
274
+ const confirmed = confirm(
275
+ 'Are you sure you want to uninstall AppClean?\n\nThis action cannot be undone.'
276
+ );
277
+
278
+ if (!confirmed) return;
279
+
280
+ const doubleConfirmed = confirm(
281
+ 'This will permanently remove AppClean from your system.\n\nType "YES" in the next dialog to confirm.'
282
+ );
283
+
284
+ if (!doubleConfirmed) return;
285
+
286
+ uninstall();
287
+ });
288
+ }
289
+ }
290
+
291
+ /**
292
+ * Execute uninstall
293
+ */
294
+ async function uninstall(): Promise<void> {
295
+ const uninstallBtn = document.getElementById('uninstall-btn');
296
+ if (uninstallBtn) {
297
+ (uninstallBtn as HTMLButtonElement).disabled = true;
298
+ uninstallBtn.textContent = '⏳ Uninstalling...';
299
+ }
300
+
301
+ uiStore.setProcessing(true);
302
+
303
+ try {
304
+ const result = await apiClient.post('/api/uninstall', {});
305
+
306
+ if (result.success) {
307
+ uiStore.showSuccess(result.message);
308
+ setTimeout(() => {
309
+ // Redirect or close window
310
+ window.close();
311
+ }, 2000);
312
+ } else {
313
+ uiStore.showError(result.message);
314
+ if (uninstallBtn) {
315
+ (uninstallBtn as HTMLButtonElement).disabled = false;
316
+ uninstallBtn.textContent = '🗑️ Uninstall AppClean';
317
+ }
318
+ }
319
+ } catch (error) {
320
+ uiStore.showError(`Uninstall failed: ${(error as Error).message}`);
321
+ if (uninstallBtn) {
322
+ (uninstallBtn as HTMLButtonElement).disabled = false;
323
+ uninstallBtn.textContent = '🗑️ Uninstall AppClean';
324
+ }
325
+ } finally {
326
+ uiStore.setProcessing(false);
327
+ }
328
+ }
329
+
330
+ /**
331
+ * Escape HTML
332
+ */
333
+ function escapeHtml(text: string): string {
334
+ const map: Record<string, string> = {
335
+ '&': '&amp;',
336
+ '<': '&lt;',
337
+ '>': '&gt;',
338
+ '"': '&quot;',
339
+ "'": '&#039;',
340
+ };
341
+ return String(text).replace(/[&<>"']/g, (m) => map[m]);
342
+ }
@@ -0,0 +1,169 @@
1
+ /**
2
+ * AppStore - Manages installed applications list and search/filter state
3
+ */
4
+
5
+ import { Store } from '../utils/events.js';
6
+
7
+ export interface InstalledApp {
8
+ name: string;
9
+ version: string;
10
+ installMethod: string;
11
+ mainPath: string;
12
+ installedDate?: string;
13
+ size?: number;
14
+ }
15
+
16
+ export interface SearchOptions {
17
+ query: string;
18
+ installMethod?: string;
19
+ sortBy?: 'name' | 'size' | 'date';
20
+ }
21
+
22
+ export interface AppStoreState {
23
+ apps: InstalledApp[];
24
+ searchOptions: SearchOptions;
25
+ selectedApp: InstalledApp | null;
26
+ isLoading: boolean;
27
+ error: string | null;
28
+ total: number;
29
+ page: number;
30
+ pageSize: number;
31
+ }
32
+
33
+ const INITIAL_STATE: AppStoreState = {
34
+ apps: [],
35
+ searchOptions: {
36
+ query: '',
37
+ installMethod: undefined,
38
+ sortBy: 'name',
39
+ },
40
+ selectedApp: null,
41
+ isLoading: false,
42
+ error: null,
43
+ total: 0,
44
+ page: 1,
45
+ pageSize: 50,
46
+ };
47
+
48
+ export class AppStore extends Store<AppStoreState> {
49
+ constructor() {
50
+ super(INITIAL_STATE);
51
+ }
52
+
53
+ // Load all apps
54
+ async loadApps(): Promise<void> {
55
+ this.setState({ isLoading: true, error: null });
56
+ try {
57
+ const response = await fetch('/api/apps/list');
58
+ if (!response.ok) throw new Error('Failed to load apps');
59
+
60
+ const data = await response.json();
61
+ this.setState({
62
+ apps: data.apps,
63
+ total: data.total,
64
+ page: data.page,
65
+ isLoading: false,
66
+ });
67
+ } catch (error) {
68
+ this.setState({
69
+ error: (error as Error).message,
70
+ isLoading: false,
71
+ });
72
+ }
73
+ }
74
+
75
+ // Search apps
76
+ async searchApps(options: Partial<SearchOptions>): Promise<void> {
77
+ const searchOptions = { ...this.state.searchOptions, ...options };
78
+ this.setState({ searchOptions, isLoading: true, error: null, page: 1 });
79
+
80
+ try {
81
+ const params = new URLSearchParams();
82
+ if (searchOptions.query) params.append('q', searchOptions.query);
83
+ if (searchOptions.installMethod) params.append('method', searchOptions.installMethod);
84
+ if (searchOptions.sortBy) params.append('sort', searchOptions.sortBy);
85
+ params.append('limit', String(this.state.pageSize));
86
+ params.append('offset', '0');
87
+
88
+ const response = await fetch(`/api/apps/search?${params}`);
89
+ if (!response.ok) throw new Error('Failed to search apps');
90
+
91
+ const data = await response.json();
92
+ this.setState({
93
+ apps: data.apps,
94
+ total: data.count,
95
+ isLoading: false,
96
+ });
97
+ } catch (error) {
98
+ this.setState({
99
+ error: (error as Error).message,
100
+ isLoading: false,
101
+ });
102
+ }
103
+ }
104
+
105
+ // Set search query
106
+ setSearchQuery(query: string): void {
107
+ this.searchApps({ query });
108
+ }
109
+
110
+ // Set install method filter
111
+ setInstallMethodFilter(method?: string): void {
112
+ this.searchApps({ installMethod: method });
113
+ }
114
+
115
+ // Set sort option
116
+ setSortBy(sortBy: 'name' | 'size' | 'date'): void {
117
+ this.searchApps({ sortBy });
118
+ }
119
+
120
+ // Select an app
121
+ selectApp(app: InstalledApp): void {
122
+ this.setState({ selectedApp: app });
123
+ }
124
+
125
+ // Clear selection
126
+ clearSelection(): void {
127
+ this.setState({ selectedApp: null });
128
+ }
129
+
130
+ // Load next page
131
+ async loadNextPage(): Promise<void> {
132
+ const nextPage = this.state.page + 1;
133
+ const offset = (nextPage - 1) * this.state.pageSize;
134
+
135
+ this.setState({ isLoading: true });
136
+ try {
137
+ const params = new URLSearchParams();
138
+ if (this.state.searchOptions.query) params.append('q', this.state.searchOptions.query);
139
+ if (this.state.searchOptions.installMethod) {
140
+ params.append('method', this.state.searchOptions.installMethod);
141
+ }
142
+ params.append('limit', String(this.state.pageSize));
143
+ params.append('offset', String(offset));
144
+
145
+ const response = await fetch(`/api/apps/search?${params}`);
146
+ if (!response.ok) throw new Error('Failed to load more apps');
147
+
148
+ const data = await response.json();
149
+ this.setState({
150
+ apps: [...this.state.apps, ...data.apps],
151
+ page: nextPage,
152
+ isLoading: false,
153
+ });
154
+ } catch (error) {
155
+ this.setState({
156
+ error: (error as Error).message,
157
+ isLoading: false,
158
+ });
159
+ }
160
+ }
161
+
162
+ // Clear error
163
+ clearError(): void {
164
+ this.setState({ error: null });
165
+ }
166
+ }
167
+
168
+ // Export singleton instance
169
+ export const appStore = new AppStore();