projax 1.3.4 → 1.3.6
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/dist/api/database.d.ts +5 -0
- package/dist/api/database.d.ts.map +1 -1
- package/dist/api/database.js +48 -2
- package/dist/api/database.js.map +1 -1
- package/dist/api/routes/projects.d.ts.map +1 -1
- package/dist/api/routes/projects.js +14 -11
- package/dist/api/routes/projects.js.map +1 -1
- package/dist/api/types.d.ts +1 -0
- package/dist/api/types.d.ts.map +1 -1
- package/dist/core/database.d.ts +2 -0
- package/dist/core/database.js +6 -0
- package/dist/electron/core/database.d.ts +2 -0
- package/dist/electron/core/database.js +6 -0
- package/dist/electron/main.js +37 -18
- package/dist/electron/preload.d.ts +3 -0
- package/dist/electron/preload.js +1 -0
- package/dist/electron/renderer/assets/index-Cs0biDoD.css +1 -0
- package/dist/electron/renderer/assets/index-D8AivBJO.js +42 -0
- package/dist/electron/renderer/index.html +2 -2
- package/dist/electron/script-runner.js +25 -7
- package/dist/index.js +43 -56
- package/dist/prxi.d.ts +2 -0
- package/dist/prxi.js +297 -0
- package/dist/prxi.mjs +34059 -0
- package/dist/script-runner.js +25 -7
- package/package.json +8 -5
package/dist/index.js
CHANGED
|
@@ -202,6 +202,46 @@ program
|
|
|
202
202
|
.description('Project management dashboard CLI')
|
|
203
203
|
.version(packageJson.version)
|
|
204
204
|
.addHelpText('beforeAll', displayLogo());
|
|
205
|
+
// Interactive terminal UI command
|
|
206
|
+
program
|
|
207
|
+
.command('prxi')
|
|
208
|
+
.alias('i')
|
|
209
|
+
.description('Launch interactive terminal UI')
|
|
210
|
+
.action(async () => {
|
|
211
|
+
try {
|
|
212
|
+
await ensureAPIServerRunning(true);
|
|
213
|
+
// Find the prxi source file - check both locations
|
|
214
|
+
const prxiSourceNew = path.join(__dirname, '..', '..', 'prxi', 'src', 'index.tsx');
|
|
215
|
+
const prxiSourceOld = path.join(__dirname, '..', 'src', 'prxi.tsx');
|
|
216
|
+
const prxiSource = fs.existsSync(prxiSourceNew) ? prxiSourceNew : prxiSourceOld;
|
|
217
|
+
// Find tsx binary
|
|
218
|
+
const tsxBinRoot = path.join(__dirname, '..', '..', '..', 'node_modules', '.bin', 'tsx');
|
|
219
|
+
const tsxBinCli = path.join(__dirname, '..', 'node_modules', '.bin', 'tsx');
|
|
220
|
+
const tsxBin = fs.existsSync(tsxBinRoot) ? tsxBinRoot : tsxBinCli;
|
|
221
|
+
if (!fs.existsSync(prxiSource)) {
|
|
222
|
+
console.error('Error: prxi UI source not found.');
|
|
223
|
+
console.error(`Looked for: ${prxiSourceNew} and ${prxiSourceOld}`);
|
|
224
|
+
process.exit(1);
|
|
225
|
+
}
|
|
226
|
+
if (!fs.existsSync(tsxBin)) {
|
|
227
|
+
console.error('Error: tsx not found. Please install dependencies: npm install');
|
|
228
|
+
process.exit(1);
|
|
229
|
+
}
|
|
230
|
+
// Run prxi using tsx
|
|
231
|
+
const { spawn } = require('child_process');
|
|
232
|
+
const child = spawn(tsxBin, [prxiSource], {
|
|
233
|
+
stdio: 'inherit',
|
|
234
|
+
cwd: path.dirname(prxiSource),
|
|
235
|
+
});
|
|
236
|
+
child.on('exit', (code) => {
|
|
237
|
+
process.exit(code || 0);
|
|
238
|
+
});
|
|
239
|
+
}
|
|
240
|
+
catch (error) {
|
|
241
|
+
console.error('Error launching prxi:', error instanceof Error ? error.message : error);
|
|
242
|
+
process.exit(1);
|
|
243
|
+
}
|
|
244
|
+
});
|
|
205
245
|
// Add project command
|
|
206
246
|
program
|
|
207
247
|
.command('add')
|
|
@@ -736,6 +776,8 @@ program
|
|
|
736
776
|
// Start Desktop UI command
|
|
737
777
|
program
|
|
738
778
|
.command('web')
|
|
779
|
+
.alias('desktop')
|
|
780
|
+
.alias('ui')
|
|
739
781
|
.description('Start the Desktop web interface')
|
|
740
782
|
.option('--dev', 'Start in development mode (with hot reload)')
|
|
741
783
|
.action(async (options) => {
|
|
@@ -876,61 +918,6 @@ program
|
|
|
876
918
|
// Ensure NODE_ENV is not set to development when using bundled files
|
|
877
919
|
process.env.NODE_ENV = 'production';
|
|
878
920
|
}
|
|
879
|
-
// Automatically rebuild better-sqlite3 for Desktop/Electron if needed
|
|
880
|
-
// This ensures it's compiled for Electron's Node.js version
|
|
881
|
-
if (hasBundledDesktop) {
|
|
882
|
-
try {
|
|
883
|
-
const electronPkg = require('electron/package.json');
|
|
884
|
-
let rebuild;
|
|
885
|
-
try {
|
|
886
|
-
rebuild = require('@electron/rebuild').rebuild;
|
|
887
|
-
}
|
|
888
|
-
catch {
|
|
889
|
-
// @electron/rebuild not available, skip auto-rebuild
|
|
890
|
-
}
|
|
891
|
-
if (rebuild) {
|
|
892
|
-
const sqlitePath = require.resolve('better-sqlite3');
|
|
893
|
-
// Find the package root - better-sqlite3 is in projax/node_modules/better-sqlite3
|
|
894
|
-
// Path structure: projax/node_modules/better-sqlite3/lib/index.js
|
|
895
|
-
// So we need to go: sqlitePath -> lib -> better-sqlite3 -> node_modules -> projax (package root)
|
|
896
|
-
const sqliteDir = path.dirname(sqlitePath); // .../better-sqlite3/lib
|
|
897
|
-
const betterSqlite3Dir = path.dirname(sqliteDir); // .../better-sqlite3
|
|
898
|
-
const nodeModulesDir = path.dirname(betterSqlite3Dir); // .../node_modules
|
|
899
|
-
// The package root is the directory containing node_modules (i.e., projax package root)
|
|
900
|
-
const packageRoot = path.dirname(nodeModulesDir); // .../projax
|
|
901
|
-
console.log('Ensuring better-sqlite3 is built for Desktop/Electron...');
|
|
902
|
-
try {
|
|
903
|
-
await new Promise((resolve, reject) => {
|
|
904
|
-
rebuild({
|
|
905
|
-
buildPath: packageRoot,
|
|
906
|
-
electronVersion: electronPkg.version,
|
|
907
|
-
onlyModules: ['better-sqlite3'],
|
|
908
|
-
force: true,
|
|
909
|
-
})
|
|
910
|
-
.then(() => {
|
|
911
|
-
console.log('✓ better-sqlite3 ready for Desktop/Electron');
|
|
912
|
-
resolve();
|
|
913
|
-
})
|
|
914
|
-
.catch((err) => {
|
|
915
|
-
// Don't fail if rebuild has issues, just warn
|
|
916
|
-
console.warn('⚠️ Could not rebuild better-sqlite3 automatically:', err.message);
|
|
917
|
-
console.warn(' The app may still work, but if you see errors, run:');
|
|
918
|
-
console.warn(` cd ${packageRoot}`);
|
|
919
|
-
console.warn(' npm rebuild better-sqlite3 --build-from-source');
|
|
920
|
-
resolve(); // Continue anyway
|
|
921
|
-
});
|
|
922
|
-
});
|
|
923
|
-
}
|
|
924
|
-
catch (rebuildError) {
|
|
925
|
-
// Continue even if rebuild fails
|
|
926
|
-
console.warn('⚠️ Rebuild check failed, continuing anyway...');
|
|
927
|
-
}
|
|
928
|
-
}
|
|
929
|
-
}
|
|
930
|
-
catch (checkError) {
|
|
931
|
-
// If we can't check, just continue - the error will show when Desktop app tries to use it
|
|
932
|
-
}
|
|
933
|
-
}
|
|
934
921
|
console.log('Starting Desktop app...');
|
|
935
922
|
const { spawn } = require('child_process');
|
|
936
923
|
const electron = require('electron');
|
|
@@ -1044,7 +1031,7 @@ program
|
|
|
1044
1031
|
// Check if first argument is not a known command
|
|
1045
1032
|
(async () => {
|
|
1046
1033
|
const args = process.argv.slice(2);
|
|
1047
|
-
const knownCommands = ['add', 'list', 'scan', 'remove', 'rn', 'rename', 'cd', 'pwd', 'web', 'scripts', 'scan-ports', 'api', '--help', '-h', '--version', '-V'];
|
|
1034
|
+
const knownCommands = ['prxi', 'i', 'add', 'list', 'scan', 'remove', 'rn', 'rename', 'cd', 'pwd', 'web', 'desktop', 'ui', 'scripts', 'scan-ports', 'api', '--help', '-h', '--version', '-V'];
|
|
1048
1035
|
// If we have at least 1 argument and first is not a known command, treat as project identifier
|
|
1049
1036
|
if (args.length >= 1 && !knownCommands.includes(args[0])) {
|
|
1050
1037
|
const projectIdentifier = args[0];
|
package/dist/prxi.d.ts
ADDED
package/dist/prxi.js
ADDED
|
@@ -0,0 +1,297 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
"use strict";
|
|
3
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
4
|
+
if (k2 === undefined) k2 = k;
|
|
5
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
6
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
7
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
8
|
+
}
|
|
9
|
+
Object.defineProperty(o, k2, desc);
|
|
10
|
+
}) : (function(o, m, k, k2) {
|
|
11
|
+
if (k2 === undefined) k2 = k;
|
|
12
|
+
o[k2] = m[k];
|
|
13
|
+
}));
|
|
14
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
15
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
16
|
+
}) : function(o, v) {
|
|
17
|
+
o["default"] = v;
|
|
18
|
+
});
|
|
19
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
20
|
+
var ownKeys = function(o) {
|
|
21
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
22
|
+
var ar = [];
|
|
23
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
24
|
+
return ar;
|
|
25
|
+
};
|
|
26
|
+
return ownKeys(o);
|
|
27
|
+
};
|
|
28
|
+
return function (mod) {
|
|
29
|
+
if (mod && mod.__esModule) return mod;
|
|
30
|
+
var result = {};
|
|
31
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
32
|
+
__setModuleDefault(result, mod);
|
|
33
|
+
return result;
|
|
34
|
+
};
|
|
35
|
+
})();
|
|
36
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
37
|
+
const react_1 = __importStar(require("react"));
|
|
38
|
+
const ink_1 = require("ink");
|
|
39
|
+
const core_bridge_1 = require("./core-bridge");
|
|
40
|
+
const script_runner_1 = require("./script-runner");
|
|
41
|
+
// Color scheme matching desktop app
|
|
42
|
+
const colors = {
|
|
43
|
+
bgPrimary: '#0d1117',
|
|
44
|
+
bgSecondary: '#161b22',
|
|
45
|
+
bgTertiary: '#1c2128',
|
|
46
|
+
bgHover: '#21262d',
|
|
47
|
+
borderColor: '#30363d',
|
|
48
|
+
textPrimary: '#c9d1d9',
|
|
49
|
+
textSecondary: '#8b949e',
|
|
50
|
+
textTertiary: '#6e7681',
|
|
51
|
+
accentCyan: '#39c5cf',
|
|
52
|
+
accentBlue: '#58a6ff',
|
|
53
|
+
accentGreen: '#3fb950',
|
|
54
|
+
accentPurple: '#bc8cff',
|
|
55
|
+
accentOrange: '#ffa657',
|
|
56
|
+
};
|
|
57
|
+
const HelpModal = ({ onClose }) => {
|
|
58
|
+
(0, ink_1.useInput)((input, key) => {
|
|
59
|
+
if (input === 'q' || key.escape || key.return) {
|
|
60
|
+
onClose();
|
|
61
|
+
}
|
|
62
|
+
});
|
|
63
|
+
return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", borderStyle: "round", borderColor: colors.accentCyan, padding: 1, width: 60 },
|
|
64
|
+
react_1.default.createElement(ink_1.Text, { bold: true, color: colors.accentCyan }, "PROJAX Terminal UI - Help"),
|
|
65
|
+
react_1.default.createElement(ink_1.Text, null, " "),
|
|
66
|
+
react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, "Navigation:"),
|
|
67
|
+
react_1.default.createElement(ink_1.Text, null, " \u2191/k Move up in project list"),
|
|
68
|
+
react_1.default.createElement(ink_1.Text, null, " \u2193/j Move down in project list"),
|
|
69
|
+
react_1.default.createElement(ink_1.Text, null, " Enter Select project"),
|
|
70
|
+
react_1.default.createElement(ink_1.Text, null, " "),
|
|
71
|
+
react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, "Actions:"),
|
|
72
|
+
react_1.default.createElement(ink_1.Text, null, " s Scan selected project"),
|
|
73
|
+
react_1.default.createElement(ink_1.Text, null, " p Scan ports"),
|
|
74
|
+
react_1.default.createElement(ink_1.Text, null, " "),
|
|
75
|
+
react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, "General:"),
|
|
76
|
+
react_1.default.createElement(ink_1.Text, null, " q/Esc Quit"),
|
|
77
|
+
react_1.default.createElement(ink_1.Text, null, " ? Show this help"),
|
|
78
|
+
react_1.default.createElement(ink_1.Text, null, " "),
|
|
79
|
+
react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, "Press any key to close...")));
|
|
80
|
+
};
|
|
81
|
+
const LoadingModal = ({ message }) => {
|
|
82
|
+
return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", borderStyle: "round", borderColor: colors.accentCyan, padding: 1, width: 40 },
|
|
83
|
+
react_1.default.createElement(ink_1.Text, null, message),
|
|
84
|
+
react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, "Please wait...")));
|
|
85
|
+
};
|
|
86
|
+
const ErrorModal = ({ message, onClose }) => {
|
|
87
|
+
(0, ink_1.useInput)((input, key) => {
|
|
88
|
+
if (key.escape || key.return) {
|
|
89
|
+
onClose();
|
|
90
|
+
}
|
|
91
|
+
});
|
|
92
|
+
return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", borderStyle: "round", borderColor: "#f85149", padding: 1, width: 50 },
|
|
93
|
+
react_1.default.createElement(ink_1.Text, { color: "#f85149", bold: true }, "Error"),
|
|
94
|
+
react_1.default.createElement(ink_1.Text, null, " "),
|
|
95
|
+
react_1.default.createElement(ink_1.Text, null, message),
|
|
96
|
+
react_1.default.createElement(ink_1.Text, null, " "),
|
|
97
|
+
react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, "Press any key to close...")));
|
|
98
|
+
};
|
|
99
|
+
const ProjectListComponent = ({ projects, selectedIndex }) => {
|
|
100
|
+
return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", width: "30%", borderStyle: "round", borderColor: colors.borderColor, padding: 1 },
|
|
101
|
+
react_1.default.createElement(ink_1.Text, { bold: true, color: colors.textPrimary }, "Projects"),
|
|
102
|
+
react_1.default.createElement(ink_1.Text, null, " "),
|
|
103
|
+
projects.length === 0 ? (react_1.default.createElement(ink_1.Text, { color: colors.textTertiary }, "No projects found")) : (projects.map((project, index) => {
|
|
104
|
+
const isSelected = index === selectedIndex;
|
|
105
|
+
const desc = project.description || project.path;
|
|
106
|
+
const shortDesc = desc.length > 30 ? desc.substring(0, 27) + '...' : desc;
|
|
107
|
+
return (react_1.default.createElement(ink_1.Text, { key: project.id, color: isSelected ? colors.accentCyan : colors.textPrimary, bold: isSelected },
|
|
108
|
+
isSelected ? '▶ ' : ' ',
|
|
109
|
+
project.name,
|
|
110
|
+
" - ",
|
|
111
|
+
shortDesc));
|
|
112
|
+
}))));
|
|
113
|
+
};
|
|
114
|
+
const ProjectDetailsComponent = ({ project }) => {
|
|
115
|
+
const [scripts, setScripts] = (0, react_1.useState)(null);
|
|
116
|
+
const [ports, setPorts] = (0, react_1.useState)([]);
|
|
117
|
+
(0, react_1.useEffect)(() => {
|
|
118
|
+
if (!project) {
|
|
119
|
+
setScripts(null);
|
|
120
|
+
setPorts([]);
|
|
121
|
+
return;
|
|
122
|
+
}
|
|
123
|
+
// Load scripts
|
|
124
|
+
try {
|
|
125
|
+
const projectScripts = (0, script_runner_1.getProjectScripts)(project.path);
|
|
126
|
+
setScripts(projectScripts);
|
|
127
|
+
}
|
|
128
|
+
catch (error) {
|
|
129
|
+
setScripts(null);
|
|
130
|
+
}
|
|
131
|
+
// Load ports
|
|
132
|
+
try {
|
|
133
|
+
const db = (0, core_bridge_1.getDatabaseManager)();
|
|
134
|
+
const projectPorts = db.getProjectPorts(project.id);
|
|
135
|
+
setPorts(projectPorts);
|
|
136
|
+
}
|
|
137
|
+
catch (error) {
|
|
138
|
+
setPorts([]);
|
|
139
|
+
}
|
|
140
|
+
}, [project]);
|
|
141
|
+
if (!project) {
|
|
142
|
+
return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", flexGrow: 1, borderStyle: "round", borderColor: colors.borderColor, padding: 1 },
|
|
143
|
+
react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, "Select a project to view details")));
|
|
144
|
+
}
|
|
145
|
+
const lastScanned = project.last_scanned
|
|
146
|
+
? new Date(project.last_scanned * 1000).toLocaleString()
|
|
147
|
+
: 'Never';
|
|
148
|
+
return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", flexGrow: 1, borderStyle: "round", borderColor: colors.borderColor, padding: 1 },
|
|
149
|
+
react_1.default.createElement(ink_1.Text, { bold: true, color: colors.textPrimary }, project.name),
|
|
150
|
+
react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, project.description || project.path),
|
|
151
|
+
react_1.default.createElement(ink_1.Text, { color: colors.textTertiary }, project.path),
|
|
152
|
+
react_1.default.createElement(ink_1.Text, null, " "),
|
|
153
|
+
project.framework && (react_1.default.createElement(react_1.default.Fragment, null,
|
|
154
|
+
react_1.default.createElement(ink_1.Text, null,
|
|
155
|
+
"Framework: ",
|
|
156
|
+
react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, project.framework)))),
|
|
157
|
+
react_1.default.createElement(ink_1.Text, null,
|
|
158
|
+
"Last Scanned: ",
|
|
159
|
+
lastScanned),
|
|
160
|
+
react_1.default.createElement(ink_1.Text, null, " "),
|
|
161
|
+
scripts && scripts.scripts && scripts.scripts.size > 0 && (react_1.default.createElement(react_1.default.Fragment, null,
|
|
162
|
+
react_1.default.createElement(ink_1.Text, { bold: true },
|
|
163
|
+
"Available Scripts (",
|
|
164
|
+
react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, scripts.scripts.size),
|
|
165
|
+
"):"),
|
|
166
|
+
Array.from(scripts.scripts.entries()).map(([name, script]) => (react_1.default.createElement(ink_1.Text, { key: name },
|
|
167
|
+
' ',
|
|
168
|
+
react_1.default.createElement(ink_1.Text, { color: colors.accentGreen }, name),
|
|
169
|
+
' - ',
|
|
170
|
+
script.command))),
|
|
171
|
+
react_1.default.createElement(ink_1.Text, null, " "))),
|
|
172
|
+
ports.length > 0 && (react_1.default.createElement(react_1.default.Fragment, null,
|
|
173
|
+
react_1.default.createElement(ink_1.Text, { bold: true },
|
|
174
|
+
"Detected Ports (",
|
|
175
|
+
react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, ports.length),
|
|
176
|
+
"):"),
|
|
177
|
+
ports.map((port) => (react_1.default.createElement(ink_1.Text, { key: port.id },
|
|
178
|
+
' ',
|
|
179
|
+
"Port ",
|
|
180
|
+
react_1.default.createElement(ink_1.Text, { color: colors.accentCyan }, port.port),
|
|
181
|
+
" - ",
|
|
182
|
+
port.config_source))),
|
|
183
|
+
react_1.default.createElement(ink_1.Text, null, " ")))));
|
|
184
|
+
};
|
|
185
|
+
const StatusBar = () => {
|
|
186
|
+
return (react_1.default.createElement(ink_1.Box, null,
|
|
187
|
+
react_1.default.createElement(ink_1.Text, { color: colors.accentGreen }, "\u25CF API CONNECTED"),
|
|
188
|
+
react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, " | Press "),
|
|
189
|
+
react_1.default.createElement(ink_1.Text, { bold: true }, "q"),
|
|
190
|
+
react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, " to quit, "),
|
|
191
|
+
react_1.default.createElement(ink_1.Text, { bold: true }, "?"),
|
|
192
|
+
react_1.default.createElement(ink_1.Text, { color: colors.textSecondary }, " for help")));
|
|
193
|
+
};
|
|
194
|
+
const App = () => {
|
|
195
|
+
const { exit } = (0, ink_1.useApp)();
|
|
196
|
+
const [projects, setProjects] = (0, react_1.useState)([]);
|
|
197
|
+
const [selectedIndex, setSelectedIndex] = (0, react_1.useState)(0);
|
|
198
|
+
const [showHelp, setShowHelp] = (0, react_1.useState)(false);
|
|
199
|
+
const [isLoading, setIsLoading] = (0, react_1.useState)(false);
|
|
200
|
+
const [loadingMessage, setLoadingMessage] = (0, react_1.useState)('');
|
|
201
|
+
const [error, setError] = (0, react_1.useState)(null);
|
|
202
|
+
(0, react_1.useEffect)(() => {
|
|
203
|
+
loadProjects();
|
|
204
|
+
}, []);
|
|
205
|
+
const loadProjects = () => {
|
|
206
|
+
const allProjects = (0, core_bridge_1.getAllProjects)();
|
|
207
|
+
setProjects(allProjects);
|
|
208
|
+
if (selectedIndex >= allProjects.length) {
|
|
209
|
+
setSelectedIndex(Math.max(0, allProjects.length - 1));
|
|
210
|
+
}
|
|
211
|
+
};
|
|
212
|
+
const selectedProject = projects.length > 0 ? projects[selectedIndex] : null;
|
|
213
|
+
(0, ink_1.useInput)((input, key) => {
|
|
214
|
+
// Don't process input if modal is showing
|
|
215
|
+
if (showHelp || isLoading || error) {
|
|
216
|
+
return;
|
|
217
|
+
}
|
|
218
|
+
// Quit
|
|
219
|
+
if (input === 'q' || key.escape) {
|
|
220
|
+
exit();
|
|
221
|
+
return;
|
|
222
|
+
}
|
|
223
|
+
// Help
|
|
224
|
+
if (input === '?') {
|
|
225
|
+
setShowHelp(true);
|
|
226
|
+
return;
|
|
227
|
+
}
|
|
228
|
+
// Navigation
|
|
229
|
+
if (key.upArrow || input === 'k') {
|
|
230
|
+
setSelectedIndex((prev) => Math.max(0, prev - 1));
|
|
231
|
+
return;
|
|
232
|
+
}
|
|
233
|
+
if (key.downArrow || input === 'j') {
|
|
234
|
+
setSelectedIndex((prev) => Math.min(projects.length - 1, prev + 1));
|
|
235
|
+
return;
|
|
236
|
+
}
|
|
237
|
+
// Scan project
|
|
238
|
+
if (input === 's' && selectedProject) {
|
|
239
|
+
setIsLoading(true);
|
|
240
|
+
setLoadingMessage(`Scanning ${selectedProject.name}...`);
|
|
241
|
+
setTimeout(async () => {
|
|
242
|
+
try {
|
|
243
|
+
await (0, core_bridge_1.scanProject)(selectedProject.id);
|
|
244
|
+
loadProjects();
|
|
245
|
+
setIsLoading(false);
|
|
246
|
+
}
|
|
247
|
+
catch (err) {
|
|
248
|
+
setIsLoading(false);
|
|
249
|
+
setError(err instanceof Error ? err.message : String(err));
|
|
250
|
+
}
|
|
251
|
+
}, 100);
|
|
252
|
+
return;
|
|
253
|
+
}
|
|
254
|
+
// Scan ports
|
|
255
|
+
if (input === 'p' && selectedProject) {
|
|
256
|
+
setIsLoading(true);
|
|
257
|
+
setLoadingMessage(`Scanning ports for ${selectedProject.name}...`);
|
|
258
|
+
setTimeout(async () => {
|
|
259
|
+
try {
|
|
260
|
+
// Import port scanner dynamically
|
|
261
|
+
const { scanProjectPorts } = await Promise.resolve().then(() => __importStar(require('./port-scanner')));
|
|
262
|
+
await scanProjectPorts(selectedProject.id);
|
|
263
|
+
loadProjects();
|
|
264
|
+
setIsLoading(false);
|
|
265
|
+
}
|
|
266
|
+
catch (err) {
|
|
267
|
+
setIsLoading(false);
|
|
268
|
+
setError(err instanceof Error ? err.message : String(err));
|
|
269
|
+
}
|
|
270
|
+
}, 100);
|
|
271
|
+
return;
|
|
272
|
+
}
|
|
273
|
+
});
|
|
274
|
+
if (showHelp) {
|
|
275
|
+
return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", padding: 1 },
|
|
276
|
+
react_1.default.createElement(HelpModal, { onClose: () => setShowHelp(false) })));
|
|
277
|
+
}
|
|
278
|
+
if (isLoading) {
|
|
279
|
+
return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", padding: 1 },
|
|
280
|
+
react_1.default.createElement(LoadingModal, { message: loadingMessage })));
|
|
281
|
+
}
|
|
282
|
+
if (error) {
|
|
283
|
+
return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", padding: 1 },
|
|
284
|
+
react_1.default.createElement(ErrorModal, { message: error, onClose: () => setError(null) })));
|
|
285
|
+
}
|
|
286
|
+
return (react_1.default.createElement(ink_1.Box, { flexDirection: "column", padding: 1 },
|
|
287
|
+
react_1.default.createElement(ink_1.Box, { borderStyle: "round", borderColor: colors.borderColor, padding: 1 },
|
|
288
|
+
react_1.default.createElement(ink_1.Text, { bold: true, color: colors.accentCyan }, "PROJAX")),
|
|
289
|
+
react_1.default.createElement(ink_1.Box, { flexDirection: "row", marginTop: 1 },
|
|
290
|
+
react_1.default.createElement(ProjectListComponent, { projects: projects, selectedIndex: selectedIndex }),
|
|
291
|
+
react_1.default.createElement(ink_1.Box, { width: 1 }),
|
|
292
|
+
react_1.default.createElement(ProjectDetailsComponent, { project: selectedProject })),
|
|
293
|
+
react_1.default.createElement(ink_1.Box, { marginTop: 1 },
|
|
294
|
+
react_1.default.createElement(StatusBar, null))));
|
|
295
|
+
};
|
|
296
|
+
// Render the app
|
|
297
|
+
(0, ink_1.render)(react_1.default.createElement(App, null));
|