diff --git a/README.md b/submissions/Screen-Time-Tracker/README.md similarity index 100% rename from README.md rename to submissions/Screen-Time-Tracker/README.md diff --git a/submissions/Screen-Time-Tracker/background.js b/submissions/Screen-Time-Tracker/background.js new file mode 100644 index 00000000..09869ad6 --- /dev/null +++ b/submissions/Screen-Time-Tracker/background.js @@ -0,0 +1,102 @@ +let currentUrl = ''; +let startTime = null; +let todayData = {}; +let isTracking = false; + +chrome.runtime.onInstalled.addListener(() => { + const today = new Date().toLocaleDateString(); + chrome.storage.local.get([today], (result) => { + todayData = result[today] || {}; + }); + + chrome.alarms.create('saveData', { periodInMinutes: 1 }); +}); + +chrome.tabs.onUpdated.addListener((tabId, changeInfo, tab) => { + if (changeInfo.status === 'complete' && tab.active) { + updateCurrentTab(tab); + } +}); + +chrome.tabs.onActivated.addListener(activeInfo => { + chrome.tabs.get(activeInfo.tabId, updateCurrentTab); +}); + +chrome.windows.onFocusChanged.addListener(windowId => { + if (windowId === chrome.windows.WINDOW_ID_NONE) { + saveCurrentSession(); + isTracking = false; + } else { + chrome.tabs.query({ active: true, currentWindow: true }, tabs => { + if (tabs.length > 0) { + updateCurrentTab(tabs[0]); + } + }); + } +}); + +chrome.alarms.onAlarm.addListener((alarm) => { + if (alarm.name === 'saveData') { + saveCurrentSession(); + saveTodayData(); + } +}); + +function updateCurrentTab(tab) { + if (!tab || !tab.url || tab.url.startsWith('chrome://')) { + return; + } + + saveCurrentSession(); + + currentUrl = new URL(tab.url).hostname; + startTime = Date.now(); + isTracking = true; +} + +function saveCurrentSession() { + if (!isTracking || !currentUrl || !startTime) { + return; + } + + const now = Date.now(); + const duration = now - startTime; + + if (duration > 1000) { + if (!todayData[currentUrl]) { + todayData[currentUrl] = 0; + } + todayData[currentUrl] += duration; + } + + startTime = now; +} + +function saveTodayData() { + const today = new Date().toLocaleDateString(); + chrome.storage.local.set({ [today]: todayData }); +} + +chrome.runtime.onMessage.addListener((request, sender, sendResponse) => { + if (request.action === 'getStats') { + saveCurrentSession(); + + const today = new Date().toLocaleDateString(); + chrome.storage.local.get([today], (result) => { + sendResponse({ todayData: result[today] || {} }); + }); + return true; + } else if (request.action === 'getHistory') { + const dates = []; + for (let i = 0; i < 7; i++) { + const date = new Date(); + date.setDate(date.getDate() - i); + dates.push(date.toLocaleDateString()); + } + + chrome.storage.local.get(dates, (result) => { + sendResponse({ historyData: result }); + }); + return true; + } +}); diff --git a/submissions/Screen-Time-Tracker/icons/icon128.png b/submissions/Screen-Time-Tracker/icons/icon128.png new file mode 100644 index 00000000..6ba518d4 --- /dev/null +++ b/submissions/Screen-Time-Tracker/icons/icon128.png @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/submissions/Screen-Time-Tracker/icons/icon16.png b/submissions/Screen-Time-Tracker/icons/icon16.png new file mode 100644 index 00000000..d9010eca --- /dev/null +++ b/submissions/Screen-Time-Tracker/icons/icon16.png @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/submissions/Screen-Time-Tracker/icons/icon48.png b/submissions/Screen-Time-Tracker/icons/icon48.png new file mode 100644 index 00000000..d90e1bf9 --- /dev/null +++ b/submissions/Screen-Time-Tracker/icons/icon48.png @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/submissions/Screen-Time-Tracker/manifest.json b/submissions/Screen-Time-Tracker/manifest.json new file mode 100644 index 00000000..cc699b13 --- /dev/null +++ b/submissions/Screen-Time-Tracker/manifest.json @@ -0,0 +1,28 @@ +{ + "manifest_version": 3, + "name": "Screen Time Tracker", + "version": "1.0", + "description": "Tracks time spent on websites and provides detailed usage statistics", + "permissions": [ + "storage", + "tabs", + "activeTab", + "alarms" + ], + "background": { + "service_worker": "background.js" + }, + "action": { + "default_popup": "popup.html", + "default_icon": { + "16": "icons/icon16.png", + "48": "icons/icon48.png", + "128": "icons/icon128.png" + } + }, + "icons": { + "16": "icons/icon16.png", + "48": "icons/icon48.png", + "128": "icons/icon128.png" + } +} diff --git a/submissions/Screen-Time-Tracker/popup.css b/submissions/Screen-Time-Tracker/popup.css new file mode 100644 index 00000000..b3e8f587 --- /dev/null +++ b/submissions/Screen-Time-Tracker/popup.css @@ -0,0 +1,171 @@ +body { + font-family: 'Segoe UI', Arial, sans-serif; + width: 350px; + margin: 0; + background: #f8f9fa; + color: #2c3e50; +} + +.container { + padding: 20px; + background: white; + border-radius: 12px; + box-shadow: 0 2px 12px rgba(0, 0, 0, 0.08); + margin: 8px; +} + +h1 { + font-size: 20px; + margin-top: 0; + color: #1a73e8; + text-align: center; + font-weight: 600; +} + +h2 { + font-size: 16px; + margin-top: 16px; + color: #3c4043; +} + +.tabs { + display: flex; + margin-bottom: 20px; + border-bottom: 1px solid #e0e0e0; + padding: 0 8px; +} + +.tab-button { + flex: 1; + background: none; + border: none; + padding: 12px; + cursor: pointer; + font-size: 14px; + color: #5f6368; + transition: all 0.3s ease; + position: relative; + font-weight: 500; +} + +.tab-button:hover { + color: #1a73e8; +} + +.tab-button.active { + color: #1a73e8; +} + +.tab-button.active::after { + content: ''; + position: absolute; + bottom: -1px; + left: 0; + width: 100%; + height: 2px; + background: #1a73e8; + transition: all 0.3s ease; +} + +.tab-content { + display: none; + opacity: 0; + transition: opacity 0.3s ease; +} + +.tab-content.active { + display: block; + opacity: 1; +} + +.total-time { + margin-bottom: 20px; + font-weight: 500; + background: #f1f3f4; + padding: 12px; + border-radius: 8px; + text-align: center; +} + +.stats-container { + max-height: 300px; + overflow-y: auto; + padding: 4px; +} + +.site-stat { + display: flex; + flex-direction: column; + padding: 12px; + border-radius: 8px; + background: #f8f9fa; + margin-bottom: 8px; + transition: transform 0.2s ease, box-shadow 0.2s ease; +} + +.site-stat:hover { + transform: translateY(-1px); + box-shadow: 0 2px 8px rgba(0, 0, 0, 0.05); +} + +.site-info { + display: flex; + justify-content: space-between; + align-items: center; + margin-bottom: 8px; +} + +.site-name { + flex: 1; + white-space: nowrap; + overflow: hidden; + text-overflow: ellipsis; + color: #202124; + font-weight: 500; +} + +.site-time { + margin-left: 12px; + color: #5f6368; + font-weight: 500; +} + +.bar-container { + height: 6px; + width: 100%; + background-color: #e8eaed; + border-radius: 8px; + overflow: hidden; +} + +.bar { + height: 100%; + background-color: #1a73e8; + border-radius: 8px; + transition: width 0.3s ease; +} + +.stats-container::-webkit-scrollbar { + width: 8px; +} + +.stats-container::-webkit-scrollbar-track { + background: #f1f1f1; + border-radius: 4px; +} + +.stats-container::-webkit-scrollbar-thumb { + background: #c1c1c1; + border-radius: 4px; +} + +.stats-container::-webkit-scrollbar-thumb:hover { + background: #a8a8a8; +} + +#weeklyChart { + margin-top: 16px; + border-radius: 8px; + padding: 8px; + background: white; +} diff --git a/submissions/Screen-Time-Tracker/popup.html b/submissions/Screen-Time-Tracker/popup.html new file mode 100644 index 00000000..c59521c8 --- /dev/null +++ b/submissions/Screen-Time-Tracker/popup.html @@ -0,0 +1,35 @@ + + + + Screen Time Tracker + + + +
+

Screen Time Tracker

+ +
+ + +
+ +
+

Today's Activity

+
+ Total screen time: + 0m +
+
+
+ +
+

Last 7 Days

+
+ +
+
+ + + + + diff --git a/submissions/Screen-Time-Tracker/popup.js b/submissions/Screen-Time-Tracker/popup.js new file mode 100644 index 00000000..ec01ef0b --- /dev/null +++ b/submissions/Screen-Time-Tracker/popup.js @@ -0,0 +1,215 @@ +const todayTab = document.getElementById('todayTab'); +const historyTab = document.getElementById('historyTab'); +const todayView = document.getElementById('todayView'); +const historyView = document.getElementById('historyView'); +const totalTimeEl = document.getElementById('totalTime'); +const todayStatsEl = document.getElementById('todayStats'); +const historyStatsEl = document.getElementById('historyStats'); + +todayTab.addEventListener('click', () => { + todayTab.classList.add('active'); + historyTab.classList.remove('active'); + todayView.classList.add('active'); + historyView.classList.remove('active'); + + historyView.style.opacity = 0; + setTimeout(() => { + todayView.classList.add('active'); + historyView.classList.remove('active'); + requestAnimationFrame(() => { + todayView.style.opacity = 1; + }); + }, 300); + loadTodayStats(); +}); + +historyTab.addEventListener('click', () => { + historyTab.classList.add('active'); + todayTab.classList.remove('active'); + historyView.classList.add('active'); + todayView.classList.remove('active'); + + todayView.style.opacity = 0; + setTimeout(() => { + historyView.classList.add('active'); + todayView.classList.remove('active'); + requestAnimationFrame(() => { + historyView.style.opacity = 1; + }); + }, 300); + loadHistoryStats(); +}); + +function formatTime(ms) { + const seconds = Math.floor(ms / 1000); + const minutes = Math.floor(seconds / 60); + const hours = Math.floor(minutes / 60); + + if (hours > 0) { + return `${hours}h ${minutes % 60}m`; + } else if (minutes > 0) { + return `${minutes}m ${seconds % 60}s`; + } else { + return `${seconds}s`; + } +} + +function loadTodayStats() { + chrome.runtime.sendMessage({ action: 'getStats' }, (response) => { + const data = response.todayData; + todayStatsEl.innerHTML = ''; + + if (!data || Object.keys(data).length === 0) { + todayStatsEl.innerHTML = '

No data recorded today.

'; + totalTimeEl.textContent = '0m'; + return; + } + + let totalTime = 0; + let maxTime = 0; + + Object.values(data).forEach(time => { + totalTime += time; + if (time > maxTime) maxTime = time; + }); + + totalTimeEl.textContent = formatTime(totalTime); + + const sortedSites = Object.entries(data) + .sort((a, b) => b[1] - a[1]); + + sortedSites.forEach(([site, time], index) => { + const percentage = (time / maxTime) * 100; + + const siteEl = document.createElement('div'); + siteEl.className = 'site-stat'; + siteEl.style.opacity = '0'; + siteEl.style.transform = 'translateY(10px)'; + siteEl.innerHTML = ` +
+
${site}
+
${formatTime(time)}
+
+
+
+
+ `; + + todayStatsEl.appendChild(siteEl); + + setTimeout(() => { + siteEl.style.opacity = '1'; + siteEl.style.transform = 'translateY(0)'; + siteEl.querySelector('.bar').style.width = `${percentage}%`; + }, index * 100); + }); + }); +} + +function loadHistoryStats() { + chrome.runtime.sendMessage({ action: 'getHistory' }, (response) => { + const historyData = response.historyData; + historyStatsEl.innerHTML = ''; + + if (!historyData || Object.keys(historyData).length === 0) { + historyStatsEl.innerHTML = '

No historical data available.

'; + return; + } + + const dates = []; + const dailyTotals = []; + + const allSites = new Set(); + Object.values(historyData).forEach(dayData => { + Object.keys(dayData).forEach(site => allSites.add(site)); + }); + + const orderedDates = Object.keys(historyData).sort((a, b) => { + return new Date(b) - new Date(a); + }); + + const siteTotals = {}; + allSites.forEach(site => { + siteTotals[site] = 0; + orderedDates.forEach(date => { + if (historyData[date] && historyData[date][site]) { + siteTotals[site] += historyData[date][site]; + } + }); + }); + + const top5Sites = Object.entries(siteTotals) + .sort((a, b) => b[1] - a[1]) + .slice(0, 5); + + historyStatsEl.innerHTML = '

Top Sites (7 days)

'; + + top5Sites.forEach(([site, totalTime]) => { + const siteEl = document.createElement('div'); + siteEl.className = 'site-stat'; + siteEl.innerHTML = ` +
${site}
+
${formatTime(totalTime)}
+ `; + historyStatsEl.appendChild(siteEl); + }); + + orderedDates.reverse().forEach(date => { + const formattedDate = new Date(date).toLocaleDateString('en-US', { + month: 'short', + day: 'numeric' + }); + dates.push(formattedDate); + + let dailyTotal = 0; + if (historyData[date]) { + Object.values(historyData[date]).forEach(time => { + dailyTotal += time; + }); + } + + dailyTotals.push(dailyTotal / (1000 * 60 * 60)); + }); + + if (typeof Chart !== 'undefined') { + const ctx = document.getElementById('weeklyChart').getContext('2d'); + + if (window.screenTimeChart) { + window.screenTimeChart.destroy(); + } + + window.screenTimeChart = new Chart(ctx, { + type: 'bar', + data: { + labels: dates, + datasets: [{ + label: 'Screen Time (hours)', + data: dailyTotals, + backgroundColor: '#4285f4' + }] + }, + options: { + responsive: true, + scales: { + y: { + beginAtZero: true, + title: { + display: true, + text: 'Hours' + } + } + } + } + }); + } else { + const chartEl = document.getElementById('weeklyChart'); + chartEl.style.display = 'none'; + + const message = document.createElement('p'); + message.textContent = 'Chart.js is required for the weekly chart.'; + historyStatsEl.appendChild(message); + } + }); +} + +document.addEventListener('DOMContentLoaded', loadTodayStats);