This commit is contained in:
Arkaprabha Chakraborty
2025-06-23 19:45:55 +05:30
commit 521c3155eb
10 changed files with 8222 additions and 0 deletions

92
.gitignore vendored Normal file
View File

@@ -0,0 +1,92 @@
# Logs
logs
*.log
npm-debug.log*
yarn-debug.log*
yarn-error.log*
lerna-debug.log*
# Diagnostic reports (https://nodejs.org/api/report.html)
report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json
# Runtime data
pids
*.pid
*.seed
*.pid.lock
.DS_Store
# Directory for instrumented libs generated by jscoverage/JSCover
lib-cov
# Coverage directory used by tools like istanbul
coverage
*.lcov
# nyc test coverage
.nyc_output
# node-waf configuration
.lock-wscript
# Compiled binary addons (https://nodejs.org/api/addons.html)
build/Release
# Dependency directories
node_modules/
jspm_packages/
# TypeScript v1 declaration files
typings/
# TypeScript cache
*.tsbuildinfo
# Optional npm cache directory
.npm
# Optional eslint cache
.eslintcache
# Optional REPL history
.node_repl_history
# Output of 'npm pack'
*.tgz
# Yarn Integrity file
.yarn-integrity
# dotenv environment variables file
.env
.env.test
# parcel-bundler cache (https://parceljs.org/)
.cache
# next.js build output
.next
# nuxt.js build output
.nuxt
# vuepress build output
.vuepress/dist
# Serverless directories
.serverless/
# FuseBox cache
.fusebox/
# DynamoDB Local files
.dynamodb/
# Webpack
.webpack/
# Vite
.vite/
# Electron-Forge
out/

1031
bun.lock Normal file

File diff suppressed because it is too large Load Diff

44
forge.config.js Normal file
View File

@@ -0,0 +1,44 @@
const { FusesPlugin } = require('@electron-forge/plugin-fuses');
const { FuseV1Options, FuseVersion } = require('@electron/fuses');
module.exports = {
packagerConfig: {
asar: true,
},
rebuildConfig: {},
makers: [
{
name: '@electron-forge/maker-squirrel',
config: {},
},
{
name: '@electron-forge/maker-zip',
platforms: ['darwin'],
},
{
name: '@electron-forge/maker-deb',
config: {},
},
{
name: '@electron-forge/maker-rpm',
config: {},
},
],
plugins: [
{
name: '@electron-forge/plugin-auto-unpack-natives',
config: {},
},
// Fuses are used to enable/disable various Electron functionality
// at package time, before code signing the application
new FusesPlugin({
version: FuseVersion.V1,
[FuseV1Options.RunAsNode]: false,
[FuseV1Options.EnableCookieEncryption]: true,
[FuseV1Options.EnableNodeOptionsEnvironmentVariable]: false,
[FuseV1Options.EnableNodeCliInspectArguments]: false,
[FuseV1Options.EnableEmbeddedAsarIntegrityValidation]: true,
[FuseV1Options.OnlyLoadAppFromAsar]: true,
}),
],
};

6223
package-lock.json generated Normal file

File diff suppressed because it is too large Load Diff

36
package.json Normal file
View File

@@ -0,0 +1,36 @@
{
"name": "monitor",
"productName": "Monitor",
"version": "1.0.0",
"description": "Desktop monitoring application",
"main": "src/index.js",
"scripts": {
"start": "electron-forge start",
"package": "electron-forge package",
"make": "electron-forge make",
"publish": "electron-forge publish",
"lint": "echo \"No linting configured\""
},
"keywords": [],
"author": {
"name": "Arkaprabha Chakraborty",
"email": "arkorty@gmail.com"
},
"license": "MIT",
"dependencies": {
"electron-squirrel-startup": "^1.0.1",
"python-bridge": "^1.1.0",
"screenshot-desktop": "^1.15.0"
},
"devDependencies": {
"@electron-forge/cli": "^7.8.1",
"@electron-forge/maker-deb": "^7.8.1",
"@electron-forge/maker-rpm": "^7.8.1",
"@electron-forge/maker-squirrel": "^7.8.1",
"@electron-forge/maker-zip": "^7.8.1",
"@electron-forge/plugin-auto-unpack-natives": "^7.8.1",
"@electron-forge/plugin-fuses": "^7.8.1",
"@electron/fuses": "^1.8.0",
"electron": "36.4.0"
}
}

386
src/index.css Normal file
View File

@@ -0,0 +1,386 @@
/* Base styles */
:root {
--primary-color: #2563eb;
--success-color: #10b981;
--warning-color: #f59e0b;
--error-color: #ef4444;
--text-primary: #1f2937;
--text-secondary: #6b7280;
--bg-primary: #ffffff;
--bg-secondary: #f3f4f6;
--border-color: #e5e7eb;
--sidebar-width: 300px;
}
* {
margin: 0;
padding: 0;
box-sizing: border-box;
}
body {
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Helvetica,
Arial, sans-serif;
background-color: var(--bg-secondary);
color: var(--text-primary);
line-height: 1.5;
}
/* Layout */
.app-container {
display: flex;
min-height: 100vh;
}
.sidebar {
width: var(--sidebar-width);
background-color: var(--bg-primary);
border-right: 1px solid var(--border-color);
padding: 1.5rem;
position: fixed;
height: 100vh;
overflow-y: auto;
}
.main-content {
flex: 1;
margin-left: var(--sidebar-width);
padding: 2rem;
}
/* Header styles */
.app-header {
margin-bottom: 2rem;
}
.app-header h1 {
font-size: 1.5rem;
font-weight: 600;
margin-bottom: 1rem;
}
.status-indicator {
display: flex;
align-items: center;
gap: 0.5rem;
font-size: 0.875rem;
color: var(--text-secondary);
}
.status-indicator.active .status-dot {
background-color: var(--success-color);
}
.status-dot {
width: 8px;
height: 8px;
border-radius: 50%;
background-color: var(--text-secondary);
}
/* Stats and monitoring */
.monitoring-stats {
display: flex;
flex-direction: column;
gap: 1.5rem;
}
.stat-card {
background-color: var(--bg-secondary);
border-radius: 0.75rem;
padding: 1rem;
}
.stat-title {
font-size: 0.875rem;
font-weight: 500;
color: var(--text-secondary);
margin-bottom: 0.75rem;
}
.progress-container {
background-color: var(--bg-primary);
border-radius: 0.5rem;
height: 0.5rem;
overflow: hidden;
position: relative;
}
.progress-bar {
background-color: var(--primary-color);
height: 100%;
transition: width 1s linear;
width: 100%;
}
.progress-text {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
font-size: 0.75rem;
font-weight: 500;
color: var(--text-primary);
margin-top: 1rem;
}
.stat-grid {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 1rem;
}
.stat-item {
text-align: center;
}
.stat-value {
display: block;
font-size: 1.5rem;
font-weight: 600;
color: var(--primary-color);
}
.stat-label {
font-size: 0.75rem;
color: var(--text-secondary);
}
/* Content area */
.content-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 2rem;
}
.content-header h2 {
font-size: 1.25rem;
font-weight: 600;
}
.view-controls {
display: flex;
gap: 0.5rem;
}
.view-btn {
padding: 0.5rem 1rem;
border: 1px solid var(--border-color);
background-color: var(--bg-primary);
border-radius: 0.375rem;
font-size: 0.875rem;
cursor: pointer;
transition: all 0.2s;
}
.view-btn:hover {
background-color: var(--bg-secondary);
}
.view-btn.active {
background-color: var(--primary-color);
color: white;
border-color: var(--primary-color);
}
/* Screenshot grid */
.screenshot-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(300px, 1fr));
gap: 1.5rem;
}
.screenshot-list {
display: flex;
flex-direction: column;
gap: 1rem;
}
.screenshot-card {
background-color: var(--bg-primary);
border-radius: 0.75rem;
overflow: hidden;
box-shadow: 0 1px 3px rgba(0, 0, 0, 0.1);
transition: transform 0.2s;
}
.screenshot-card:hover {
transform: translateY(-2px);
}
.screenshot-preview {
position: relative;
aspect-ratio: 16/9;
}
.screenshot-preview img {
width: 100%;
height: 100%;
object-fit: cover;
}
.screenshot-overlay {
position: absolute;
bottom: 0;
left: 0;
right: 0;
padding: 1rem;
background: linear-gradient(transparent, rgba(0, 0, 0, 0.7));
color: white;
}
.screenshot-time {
font-size: 0.875rem;
}
.screenshot-details {
padding: 1rem;
}
/* Analysis styles */
.analysis-status {
display: flex;
align-items: center;
gap: 0.5rem;
margin-bottom: 0.75rem;
}
.analysis-indicator {
width: 8px;
height: 8px;
border-radius: 50%;
background-color: var(--text-secondary);
}
.analysis-indicator.success {
background-color: var(--success-color);
}
.analysis-indicator.warning {
background-color: var(--warning-color);
}
.analysis-spinner {
width: 16px;
height: 16px;
border: 2px solid var(--border-color);
border-top-color: var(--primary-color);
border-radius: 50%;
animation: spin 1s linear infinite;
}
@keyframes spin {
to {
transform: rotate(360deg);
}
}
.analysis-summary {
background-color: var(--bg-secondary);
border-radius: 0.5rem;
padding: 1rem;
}
.analysis-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 0.75rem;
}
.role-badge {
background-color: var(--primary-color);
color: white;
padding: 0.25rem 0.5rem;
border-radius: 0.25rem;
font-size: 0.75rem;
font-weight: 500;
}
.confidence-score {
font-size: 0.75rem;
color: var(--text-secondary);
}
.anomalies-list {
display: flex;
flex-direction: column;
gap: 0.75rem;
}
.anomaly-item {
padding-top: 0.75rem;
border-top: 1px solid var(--border-color);
}
.anomaly-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 0.25rem;
}
.anomaly-type {
font-weight: 500;
font-size: 0.875rem;
}
.anomaly-confidence {
font-size: 0.75rem;
color: var(--warning-color);
}
.anomaly-explanation {
font-size: 0.875rem;
color: var(--text-secondary);
margin: 0;
}
/* Notifications */
.notification {
position: fixed;
top: 1rem;
right: 1rem;
padding: 1rem;
border-radius: 0.5rem;
background-color: white;
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
display: none;
z-index: 1000;
animation: slideIn 0.3s ease-out;
}
.notification.success {
border-left: 4px solid var(--success-color);
}
.notification.error {
border-left: 4px solid var(--error-color);
}
@keyframes slideIn {
from {
transform: translateX(100%);
opacity: 0;
}
to {
transform: translateX(0);
opacity: 1;
}
}
/* Error states */
.analysis-error {
display: flex;
align-items: center;
gap: 0.5rem;
color: var(--text-secondary);
font-size: 0.875rem;
}
.error-icon {
width: 16px;
height: 16px;
fill: currentColor;
}

222
src/index.html Normal file
View File

@@ -0,0 +1,222 @@
<!DOCTYPE html>
<html>
<head>
<meta charset="UTF-8" />
<title>Screen Monitor Pro</title>
<link rel="stylesheet" href="index.css" />
</head>
<body>
<div class="app-container">
<div class="sidebar">
<div class="app-header">
<h1>Screen Monitor Pro</h1>
<div class="status-indicator active">
<span class="status-dot"></span>
Monitoring Active
</div>
</div>
<div class="monitoring-stats">
<div class="stat-card">
<div class="stat-title">Next Screenshot</div>
<div class="progress-container">
<div class="progress-bar" id="screenshotProgress"></div>
<div class="progress-text" id="countdown">30s</div>
</div>
</div>
<div class="stat-card">
<div class="stat-title">Today's Activity</div>
<div class="stat-grid">
<div class="stat-item">
<span class="stat-value" id="totalScreenshots">0</span>
<span class="stat-label">Screenshots</span>
</div>
<div class="stat-item">
<span class="stat-value" id="totalAnomalies">0</span>
<span class="stat-label">Anomalies</span>
</div>
</div>
</div>
</div>
</div>
<div class="main-content">
<div class="content-header">
<h2>Recent Activity</h2>
<div class="view-controls">
<button class="view-btn active" data-view="grid">Grid View</button>
<button class="view-btn" data-view="list">List View</button>
</div>
</div>
<div id="notification" class="notification"></div>
<div class="screenshot-grid" id="screenshotList">
<!-- Screenshots will be added here dynamically -->
</div>
</div>
</div>
<script>
const { ipcRenderer } = require("electron");
let totalScreenshots = 0;
let totalAnomalies = 0;
function updateCountdown(seconds) {
const progressBar = document.getElementById("screenshotProgress");
const countdownElement = document.getElementById("countdown");
// Update progress bar
const progressPercent = (seconds / 30) * 100;
progressBar.style.width = `${progressPercent}%`;
// Update countdown text
const minutes = Math.floor(seconds / 60);
const remainingSeconds = seconds % 60;
countdownElement.textContent = `${minutes}:${remainingSeconds.toString().padStart(2, "0")}`;
}
function showNotification(message, type = "success") {
const notification = document.getElementById("notification");
notification.textContent = message;
notification.className = `notification ${type}`;
notification.style.display = "block";
setTimeout(() => {
notification.style.display = "none";
}, 3000);
}
function addScreenshotToList(screenshotPath) {
const list = document.getElementById("screenshotList");
const item = document.createElement("div");
item.className = "screenshot-card";
item.id = `screenshot-${screenshotPath.split("/").pop()}`;
const timestamp = new Date().toLocaleString();
const relativePath = screenshotPath.split("/").pop();
item.innerHTML = `
<div class="screenshot-preview">
<img src="file://${screenshotPath}" alt="Screenshot">
<div class="screenshot-overlay">
<div class="screenshot-time">${timestamp}</div>
</div>
</div>
<div class="screenshot-details">
<div class="analysis-status">
<div class="analysis-indicator"></div>
<span>Analyzing...</span>
</div>
<div class="analysis-result loading">
<div class="analysis-spinner"></div>
Analyzing screenshot...
</div>
</div>
`;
list.insertBefore(item, list.firstChild);
// Update stats
totalScreenshots++;
document.getElementById("totalScreenshots").textContent = totalScreenshots;
return item;
}
function updateScreenshotAnalysis(screenshotPath, analysis) {
const item = document.getElementById(`screenshot-${screenshotPath.split("/").pop()}`);
if (item) {
const analysisDiv = item.querySelector(".analysis-result");
const statusDiv = item.querySelector(".analysis-status");
analysisDiv.className = "analysis-result";
if (!analysis || analysis.status !== "success" || !analysis.analysis) {
analysisDiv.innerHTML = `
<div class="analysis-error">
<svg viewBox="0 0 24 24" class="error-icon">
<path d="M12 2C6.48 2 2 6.48 2 12s4.48 10 10 10 10-4.48 10-10S17.52 2 12 2zm1 15h-2v-2h2v2zm0-4h-2V7h2v6z"/>
</svg>
No analysis available
</div>
`;
return;
}
const a = analysis.analysis;
const hasAnomalies = a.anomalies && a.anomalies.length > 0;
if (hasAnomalies) {
totalAnomalies += a.anomalies.length;
document.getElementById("totalAnomalies").textContent = totalAnomalies;
}
statusDiv.innerHTML = `
<div class="analysis-indicator ${hasAnomalies ? 'warning' : 'success'}"></div>
<span>${hasAnomalies ? 'Anomaly Detected' : 'No Anomalies'}</span>
`;
let html = `
<div class="analysis-summary ${hasAnomalies ? 'has-anomalies' : ''}">
<div class="analysis-header">
<span class="role-badge">${a.baseline_role}</span>
<span class="confidence-score">
${hasAnomalies ? `Confidence: ${Math.round(a.anomalies[0].confidence * 100)}%` : ''}
</span>
</div>
`;
if (hasAnomalies) {
html += '<div class="anomalies-list">';
a.anomalies.forEach(anomaly => {
html += `
<div class="anomaly-item">
<div class="anomaly-header">
<span class="anomaly-type">${anomaly.type.replace(/([A-Z])/g, ' $1').trim()}</span>
<span class="anomaly-confidence">${Math.round(anomaly.confidence * 100)}%</span>
</div>
<p class="anomaly-explanation">${anomaly.explanation}</p>
</div>
`;
});
html += '</div>';
}
html += '</div>';
analysisDiv.innerHTML = html;
}
}
// View switching functionality
document.querySelectorAll('.view-btn').forEach(btn => {
btn.addEventListener('click', () => {
document.querySelectorAll('.view-btn').forEach(b => b.classList.remove('active'));
btn.classList.add('active');
const view = btn.dataset.view;
const list = document.getElementById('screenshotList');
list.className = view === 'grid' ? 'screenshot-grid' : 'screenshot-list';
});
});
// Restore ipcRenderer event listeners for real-time updates
ipcRenderer.on("countdown-update", (event, seconds) => {
updateCountdown(seconds);
});
ipcRenderer.on("screenshot-taken", (event, screenshotPath) => {
showNotification("Screenshot taken successfully!");
addScreenshotToList(screenshotPath);
});
ipcRenderer.on("screenshot-analysis", (event, data) => {
updateScreenshotAnalysis(data.screenshotPath, data.analysis);
});
ipcRenderer.on("screenshot-error", (event, error) => {
showNotification(`Error: ${error}`, "error");
});
</script>
</body>
</html>

185
src/index.js Normal file
View File

@@ -0,0 +1,185 @@
const { app, BrowserWindow, ipcMain } = require("electron");
const path = require("path");
const screenshot = require("screenshot-desktop");
const fs = require("fs");
const os = require("os");
const { spawn, exec } = require("child_process");
const appName = require("../package.json").name;
const screenshotsDir = path.join(
os.homedir(),
".cache",
appName,
"screenshots"
);
if (!fs.existsSync(screenshotsDir)) {
fs.mkdirSync(screenshotsDir, { recursive: true });
}
const logicDir = path.join(__dirname, "logic");
const venvDir = path.join(logicDir, "venv");
const pythonPath = path.join(venvDir, "bin", "python");
const pipPath = path.join(venvDir, "bin", "pip");
async function setupPythonEnv() {
return new Promise((resolve, reject) => {
if (fs.existsSync(venvDir)) {
console.log("Python virtual environment already exists");
resolve();
return;
}
console.log("Creating Python virtual environment...");
const python = process.platform === "win32" ? "python" : "python3";
exec(`${python} -m venv ${venvDir}`, (error) => {
if (error) {
console.error("Error creating virtual environment:", error);
reject(error);
return;
}
console.log("Installing Python dependencies...");
exec(
`${pipPath} install -r ${path.join(logicDir, "requirements.txt")}`,
(error) => {
if (error) {
console.error("Error installing dependencies:", error);
reject(error);
return;
}
console.log("Python environment setup complete");
resolve();
}
);
});
});
}
let mainWindow;
let countdownInterval;
const SCREENSHOT_INTERVAL = 30 * 1000;
function createWindow() {
mainWindow = new BrowserWindow({
width: 800,
height: 600,
webPreferences: {
nodeIntegration: true,
contextIsolation: false,
},
});
mainWindow.loadFile("src/index.html");
}
async function takeScreenshot() {
try {
const timestamp = new Date().toISOString().replace(/[:.]/g, "-");
const screenshotPath = path.join(
screenshotsDir,
`screenshot-${timestamp}.png`
);
await screenshot({ filename: screenshotPath });
mainWindow.webContents.send("screenshot-taken", screenshotPath);
const pythonScript = path.join(logicDir, "src/main.py");
const prohibitedPath = path.join(logicDir, "src/data", "prohibited.csv");
const pythonProcess = spawn(pythonPath, [
pythonScript,
"--single",
screenshotPath,
"--prohibited",
prohibitedPath,
"--role",
"developer",
"--model",
"gemini",
]);
let outputData = "";
pythonProcess.stdout.on("data", (data) => {
outputData += data.toString();
});
pythonProcess.stderr.on("data", (data) => {
console.error(`Python script error: ${data}`);
mainWindow.webContents.send("screenshot-error", data.toString());
});
pythonProcess.on("close", (code) => {
if (code === 0) {
try {
const jsonResponse = JSON.parse(outputData);
mainWindow.webContents.send("screenshot-analysis", {
screenshotPath,
analysis: jsonResponse,
});
} catch (error) {
console.error("Error parsing Python script output:", error);
mainWindow.webContents.send(
"screenshot-error",
"Invalid JSON response from analysis"
);
}
} else {
mainWindow.webContents.send(
"screenshot-error",
`Python script exited with code ${code}`
);
}
});
} catch (error) {
console.error("Screenshot error:", error);
mainWindow.webContents.send("screenshot-error", error.message);
}
}
function startCountdown() {
let timeLeft = SCREENSHOT_INTERVAL / 1000;
countdownInterval = setInterval(() => {
timeLeft--;
if (mainWindow && !mainWindow.isDestroyed()) {
mainWindow.webContents.send("countdown-update", timeLeft);
}
if (timeLeft <= 0) {
takeScreenshot();
timeLeft = SCREENSHOT_INTERVAL / 1000;
}
}, 1000);
}
app.whenReady().then(async () => {
try {
await setupPythonEnv();
createWindow();
startCountdown();
takeScreenshot();
app.on("activate", () => {
if (BrowserWindow.getAllWindows().length === 0) {
createWindow();
}
});
} catch (error) {
console.error("Failed to setup Python environment:", error);
app.quit();
}
});
app.on("window-all-closed", () => {
if (process.platform !== "darwin") {
clearInterval(countdownInterval);
app.quit();
}
});

1
src/logic Submodule

Submodule src/logic added at 8f656db8d9

2
src/preload.js Normal file
View File

@@ -0,0 +1,2 @@
// See the Electron documentation for details on how to use preload scripts:
// https://www.electronjs.org/docs/latest/tutorial/process-model#preload-scripts