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.
- package/GUI_IMPLEMENTATION_STATUS.md +143 -0
- package/MD_Files/INDEX.md +51 -0
- package/MD_Files/PUBLICATION_SUCCESS_REPORT.md +227 -0
- package/PHASE2_COMPLETION.md +281 -0
- package/PHASE3_COMPLETION.md +364 -0
- package/README.md +446 -376
- package/assets/logo.svg +34 -0
- package/dist/core/appUpdateChecker.js +12 -16
- package/dist/core/appUpdateChecker.js.map +1 -1
- package/dist/core/detector.js +14 -18
- package/dist/core/detector.js.map +1 -1
- package/dist/core/duplicateFileFinder.js +12 -19
- package/dist/core/duplicateFileFinder.js.map +1 -1
- package/dist/core/orphanedDependencyDetector.js +19 -26
- package/dist/core/orphanedDependencyDetector.js.map +1 -1
- package/dist/core/performanceOptimizer.js +6 -10
- package/dist/core/performanceOptimizer.js.map +1 -1
- package/dist/core/permissionHandler.js +21 -25
- package/dist/core/permissionHandler.js.map +1 -1
- package/dist/core/pluginSystem.js +9 -13
- package/dist/core/pluginSystem.js.map +1 -1
- package/dist/core/removalRecorder.js +12 -19
- package/dist/core/removalRecorder.js.map +1 -1
- package/dist/core/remover.js +59 -66
- package/dist/core/remover.js.map +1 -1
- package/dist/core/reportGenerator.d.ts +1 -1
- package/dist/core/reportGenerator.d.ts.map +1 -1
- package/dist/core/reportGenerator.js +27 -34
- package/dist/core/reportGenerator.js.map +1 -1
- package/dist/core/scheduledCleanup.js +23 -30
- package/dist/core/scheduledCleanup.js.map +1 -1
- package/dist/core/serviceFileDetector.js +24 -31
- package/dist/core/serviceFileDetector.js.map +1 -1
- package/dist/core/verificationModule.js +10 -14
- package/dist/core/verificationModule.js.map +1 -1
- package/dist/index.js +190 -90
- package/dist/index.js.map +1 -1
- package/dist/managers/brewManager.d.ts.map +1 -1
- package/dist/managers/brewManager.js +35 -41
- package/dist/managers/brewManager.js.map +1 -1
- package/dist/managers/customManager.d.ts +2 -1
- package/dist/managers/customManager.d.ts.map +1 -1
- package/dist/managers/customManager.js +79 -53
- package/dist/managers/customManager.js.map +1 -1
- package/dist/managers/linuxManager.js +29 -36
- package/dist/managers/linuxManager.js.map +1 -1
- package/dist/managers/npmManager.js +27 -34
- package/dist/managers/npmManager.js.map +1 -1
- package/dist/types/index.js +1 -2
- package/dist/ui/client/api/client.d.ts +24 -0
- package/dist/ui/client/api/client.d.ts.map +1 -0
- package/dist/ui/client/api/client.js +96 -0
- package/dist/ui/client/api/client.js.map +1 -0
- package/dist/ui/client/app.d.ts +7 -0
- package/dist/ui/client/app.d.ts.map +1 -0
- package/dist/ui/client/app.js +71 -0
- package/dist/ui/client/app.js.map +1 -0
- package/dist/ui/client/index.html +107 -0
- package/dist/ui/client/pages/appDetails.d.ts +8 -0
- package/dist/ui/client/pages/appDetails.d.ts.map +1 -0
- package/dist/ui/client/pages/appDetails.js +287 -0
- package/dist/ui/client/pages/appDetails.js.map +1 -0
- package/dist/ui/client/pages/appSearch.d.ts +2 -0
- package/dist/ui/client/pages/appSearch.d.ts.map +1 -0
- package/dist/ui/client/pages/appSearch.js +210 -0
- package/dist/ui/client/pages/appSearch.js.map +1 -0
- package/dist/ui/client/pages/dashboard.d.ts +2 -0
- package/dist/ui/client/pages/dashboard.d.ts.map +1 -0
- package/dist/ui/client/pages/dashboard.js +154 -0
- package/dist/ui/client/pages/dashboard.js.map +1 -0
- package/dist/ui/client/pages/settings.d.ts +7 -0
- package/dist/ui/client/pages/settings.d.ts.map +1 -0
- package/dist/ui/client/pages/settings.js +279 -0
- package/dist/ui/client/pages/settings.js.map +1 -0
- package/dist/ui/client/state/appStore.d.ts +38 -0
- package/dist/ui/client/state/appStore.d.ts.map +1 -0
- package/dist/ui/client/state/appStore.js +121 -0
- package/dist/ui/client/state/appStore.js.map +1 -0
- package/dist/ui/client/state/dashboardStore.d.ts +31 -0
- package/dist/ui/client/state/dashboardStore.d.ts.map +1 -0
- package/dist/ui/client/state/dashboardStore.js +70 -0
- package/dist/ui/client/state/dashboardStore.js.map +1 -0
- package/dist/ui/client/state/uiStore.d.ts +43 -0
- package/dist/ui/client/state/uiStore.d.ts.map +1 -0
- package/dist/ui/client/state/uiStore.js +109 -0
- package/dist/ui/client/state/uiStore.js.map +1 -0
- package/dist/ui/client/styles/animations.css +327 -0
- package/dist/ui/client/styles/base.css +214 -0
- package/dist/ui/client/styles/components.css +400 -0
- package/dist/ui/client/styles/layout.css +224 -0
- package/dist/ui/client/styles/variables.css +140 -0
- package/dist/ui/client/utils/events.d.ts +19 -0
- package/dist/ui/client/utils/events.d.ts.map +1 -0
- package/dist/ui/client/utils/events.js +54 -0
- package/dist/ui/client/utils/events.js.map +1 -0
- package/dist/ui/client/utils/formatting.d.ts +11 -0
- package/dist/ui/client/utils/formatting.d.ts.map +1 -0
- package/dist/ui/client/utils/formatting.js +104 -0
- package/dist/ui/client/utils/formatting.js.map +1 -0
- package/dist/ui/client/utils/router.d.ts +25 -0
- package/dist/ui/client/utils/router.d.ts.map +1 -0
- package/dist/ui/client/utils/router.js +90 -0
- package/dist/ui/client/utils/router.js.map +1 -0
- package/dist/ui/guiServer.d.ts +8 -1
- package/dist/ui/guiServer.d.ts.map +1 -1
- package/dist/ui/guiServer.js +148 -110
- package/dist/ui/guiServer.js.map +1 -1
- package/dist/ui/menu.js +18 -27
- package/dist/ui/menu.js.map +1 -1
- package/dist/ui/prompts.js +34 -47
- package/dist/ui/prompts.js.map +1 -1
- package/dist/ui/server/middleware/errorHandler.d.ts +19 -0
- package/dist/ui/server/middleware/errorHandler.d.ts.map +1 -0
- package/dist/ui/server/middleware/errorHandler.js +100 -0
- package/dist/ui/server/middleware/errorHandler.js.map +1 -0
- package/dist/ui/server/routes/apps.d.ts +8 -0
- package/dist/ui/server/routes/apps.d.ts.map +1 -0
- package/dist/ui/server/routes/apps.js +74 -0
- package/dist/ui/server/routes/apps.js.map +1 -0
- package/dist/ui/server/routes/dashboard.d.ts +4 -0
- package/dist/ui/server/routes/dashboard.d.ts.map +1 -0
- package/dist/ui/server/routes/dashboard.js +57 -0
- package/dist/ui/server/routes/dashboard.js.map +1 -0
- package/dist/ui/server/routes/settings.d.ts +6 -0
- package/dist/ui/server/routes/settings.d.ts.map +1 -0
- package/dist/ui/server/routes/settings.js +31 -0
- package/dist/ui/server/routes/settings.js.map +1 -0
- package/dist/ui/server/services/appService.d.ts +45 -0
- package/dist/ui/server/services/appService.d.ts.map +1 -0
- package/dist/ui/server/services/appService.js +114 -0
- package/dist/ui/server/services/appService.js.map +1 -0
- package/dist/ui/server/services/removalService.d.ts +24 -0
- package/dist/ui/server/services/removalService.d.ts.map +1 -0
- package/dist/ui/server/services/removalService.js +83 -0
- package/dist/ui/server/services/removalService.js.map +1 -0
- package/dist/utils/filesystem.js +32 -49
- package/dist/utils/filesystem.js.map +1 -1
- package/dist/utils/logger.js +9 -18
- package/dist/utils/logger.js.map +1 -1
- package/dist/utils/platform.js +10 -22
- package/dist/utils/platform.js.map +1 -1
- package/dist/utils/upgrade.d.ts +22 -0
- package/dist/utils/upgrade.d.ts.map +1 -0
- package/dist/utils/upgrade.js +94 -0
- package/dist/utils/upgrade.js.map +1 -0
- package/package.json +4 -2
- package/src/core/appUpdateChecker.ts +1 -1
- package/src/core/detector.ts +6 -6
- package/src/core/duplicateFileFinder.ts +1 -1
- package/src/core/orphanedDependencyDetector.ts +2 -2
- package/src/core/performanceOptimizer.ts +1 -1
- package/src/core/permissionHandler.ts +2 -2
- package/src/core/pluginSystem.ts +1 -1
- package/src/core/removalRecorder.ts +2 -2
- package/src/core/remover.ts +11 -11
- package/src/core/reportGenerator.ts +2 -2
- package/src/core/scheduledCleanup.ts +2 -2
- package/src/core/serviceFileDetector.ts +2 -2
- package/src/core/verificationModule.ts +2 -2
- package/src/index.ts +133 -6
- package/src/managers/brewManager.ts +11 -9
- package/src/managers/customManager.ts +71 -30
- package/src/managers/linuxManager.ts +3 -3
- package/src/managers/npmManager.ts +3 -3
- package/src/ui/client/api/client.ts +163 -0
- package/src/ui/client/app.ts +121 -0
- package/src/ui/client/index.html +107 -0
- package/src/ui/client/pages/appDetails.ts +356 -0
- package/src/ui/client/pages/appSearch.ts +270 -0
- package/src/ui/client/pages/dashboard.ts +189 -0
- package/src/ui/client/pages/settings.ts +342 -0
- package/src/ui/client/state/appStore.ts +169 -0
- package/src/ui/client/state/dashboardStore.ts +113 -0
- package/src/ui/client/state/uiStore.ts +166 -0
- package/src/ui/client/styles/animations.css +327 -0
- package/src/ui/client/styles/base.css +214 -0
- package/src/ui/client/styles/components.css +400 -0
- package/src/ui/client/styles/layout.css +224 -0
- package/src/ui/client/styles/variables.css +140 -0
- package/src/ui/client/utils/events.ts +74 -0
- package/src/ui/client/utils/formatting.ts +157 -0
- package/src/ui/client/utils/router.ts +161 -0
- package/src/ui/guiServer.ts +206 -105
- package/src/ui/prompts.ts +1 -1
- package/src/ui/server/middleware/errorHandler.ts +174 -0
- package/src/ui/server/routes/apps.ts +132 -0
- package/src/ui/server/routes/dashboard.ts +93 -0
- package/src/ui/server/routes/settings.ts +63 -0
- package/src/ui/server/services/appService.ts +184 -0
- package/src/ui/server/services/removalService.ts +138 -0
- package/src/utils/upgrade.ts +143 -0
- package/tsconfig.json +3 -2
- package/INDEX.md +0 -165
- /package/{ACTION_CHECKLIST.md → MD_Files/ACTION_CHECKLIST.md} +0 -0
- /package/{APPCLEAN_SUMMARY.md → MD_Files/APPCLEAN_SUMMARY.md} +0 -0
- /package/{CHANGELOG.md → MD_Files/CHANGELOG.md} +0 -0
- /package/{CODE_OF_CONDUCT.md → MD_Files/CODE_OF_CONDUCT.md} +0 -0
- /package/{CODE_REVIEW_REPORT.md → MD_Files/CODE_REVIEW_REPORT.md} +0 -0
- /package/{COMMUNITY_POSTS.md → MD_Files/COMMUNITY_POSTS.md} +0 -0
- /package/{DEPLOYMENT_GUIDE.md → MD_Files/DEPLOYMENT_GUIDE.md} +0 -0
- /package/{DEPLOYMENT_STATUS.md → MD_Files/DEPLOYMENT_STATUS.md} +0 -0
- /package/{EXECUTIVE_REPORT.md → MD_Files/EXECUTIVE_REPORT.md} +0 -0
- /package/{GITHUB_OPTIMIZATION.md → MD_Files/GITHUB_OPTIMIZATION.md} +0 -0
- /package/{MARKETING_SUMMARY.md → MD_Files/MARKETING_SUMMARY.md} +0 -0
- /package/{NPM_PACKAGE_OPTIMIZATION.md → MD_Files/NPM_PACKAGE_OPTIMIZATION.md} +0 -0
- /package/{NPM_PUBLISH.md → MD_Files/NPM_PUBLISH.md} +0 -0
- /package/{PROJECT_SUMMARY.txt → MD_Files/PROJECT_SUMMARY.txt} +0 -0
- /package/{QUICKSTART.md → MD_Files/QUICKSTART.md} +0 -0
- /package/{SETUP_GITHUB.md → MD_Files/SETUP_GITHUB.md} +0 -0
- /package/{TESTING_SUMMARY.md → MD_Files/TESTING_SUMMARY.md} +0 -0
- /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
|
+
'&': '&',
|
|
183
|
+
'<': '<',
|
|
184
|
+
'>': '>',
|
|
185
|
+
'"': '"',
|
|
186
|
+
"'": ''',
|
|
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
|
+
'&': '&',
|
|
336
|
+
'<': '<',
|
|
337
|
+
'>': '>',
|
|
338
|
+
'"': '"',
|
|
339
|
+
"'": ''',
|
|
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();
|