42 Commits

Author SHA1 Message Date
cfc9ac73b2 0.4.4
All checks were successful
Deploy to Production / deploy (push) Successful in 12s
2026-02-27 07:43:26 +00:00
e095c0190b style: restore danger styling for stop sniffing button 2026-02-27 07:43:12 +00:00
45342d456a 0.4.3
All checks were successful
Deploy to Production / deploy (push) Successful in 12s
2026-02-27 07:09:38 +00:00
3ea7f63b83 style: remove panel borders and backgrounds on mobile for cleaner look 2026-02-27 07:09:27 +00:00
8b5705c12f 0.4.2
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-27 07:03:14 +00:00
a3bc069029 fix: responsive layout for url cleaner and sniffer icon alignment 2026-02-27 07:02:58 +00:00
a5fc242a97 0.4.1
All checks were successful
Deploy to Production / deploy (push) Successful in 12s
2026-02-27 06:49:01 +00:00
a0346a64f0 feat: url cleaner exceptions keep-all and defaults reset 2026-02-27 06:48:39 +00:00
4c2d423715 0.4.0
All checks were successful
Deploy to Production / deploy (push) Successful in 12s
2026-02-27 06:14:51 +00:00
204aeda00c feat: implement url cleaner tool, local storage persistence and extension integration 2026-02-27 06:14:43 +00:00
7d989be27f feat: url cleaner tool and extension update 2026-02-27 06:11:12 +00:00
efe23a99ac 0.3.5
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-27 04:50:57 +00:00
bebb63c1de feat: improve PWA update mechanism with visibility check 2026-02-27 04:50:39 +00:00
98d76e3a35 0.3.4
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-27 04:44:34 +00:00
cc7e80a807 feat: improve mobile layout for Password Generator button 2026-02-27 04:43:58 +00:00
1f5500f7d7 0.3.3
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-27 04:37:29 +00:00
d404370027 fix: adjust layout for Password Generator and Clipboard Sniffer on smaller screens 2026-02-27 04:37:21 +00:00
8fb3ee1069 refactor: align privacy policy layout with app design
All checks were successful
Deploy to Production / deploy (push) Successful in 7s
2026-02-27 04:20:29 +00:00
348c78612d 0.3.2
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-27 04:16:17 +00:00
dc99dce485 fix: restore dark mode styles and scope privacy policy styles 2026-02-27 04:16:01 +00:00
20dc18dd28 0.3.1
All checks were successful
Deploy to Production / deploy (push) Successful in 12s
2026-02-27 04:03:50 +00:00
30d67472cb docs: add privacy policy and extension documentation 2026-02-27 04:03:35 +00:00
ac425d3df2 chore: remove dev-dist from version control
All checks were successful
Deploy to Production / deploy (push) Successful in 6s
2026-02-27 03:39:16 +00:00
e817ff6169 0.3.0
All checks were successful
Deploy to Production / deploy (push) Successful in 14s
2026-02-27 03:37:50 +00:00
8e8bf47297 feat: improve Clipboard Sniffer extension integration and UI fixes 2026-02-27 03:37:33 +00:00
1d7fd9a5bc 0.2.0
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-27 01:32:12 +00:00
8bf44cde6a feat: unify tool styles and add dynamic height for textareas 2026-02-27 01:32:08 +00:00
782786ec7e 0.1.2
All checks were successful
Deploy to Production / deploy (push) Successful in 6s
2026-02-27 00:31:40 +00:00
c8a009dda1 feat: update passwords tool UI, fix layout issues, and set PWA theme color 2026-02-27 00:31:26 +00:00
7c4f1b20b3 fix(pwa): add workbox cleanup options to fix undefined payload error
All checks were successful
Deploy to Production / deploy (push) Successful in 7s
2026-02-26 23:52:47 +00:00
c3f96bacc7 fix(pwa): improve install prompt visibility and styling
All checks were successful
Deploy to Production / deploy (push) Successful in 6s
2026-02-26 23:49:55 +00:00
8655533a2d feat(pwa): add PWA support with auto-update and install prompt
All checks were successful
Deploy to Production / deploy (push) Successful in 12s
2026-02-26 23:46:17 +00:00
bec57f9e49 feat(ui): add new app icon svg
All checks were successful
Deploy to Production / deploy (push) Successful in 5s
2026-02-26 23:41:28 +00:00
9ec71db87c fix(ui): prevent label wrapping in passwords tool
All checks were successful
Deploy to Production / deploy (push) Successful in 6s
2026-02-26 23:38:07 +00:00
3b229b4719 fix(ui): adjust mobile bottom padding to match top margin
All checks were successful
Deploy to Production / deploy (push) Successful in 5s
2026-02-26 23:35:00 +00:00
c6baace721 fix(ui): prevent text selection on buttons
All checks were successful
Deploy to Production / deploy (push) Successful in 6s
2026-02-26 23:32:36 +00:00
ab2da36aa1 0.1.1
All checks were successful
Deploy to Production / deploy (push) Successful in 8s
2026-02-26 23:21:39 +00:00
8691106d2f fix(passwords): adjust textarea width with box-sizing 2026-02-26 23:21:30 +00:00
5480b57615 0.1.0
All checks were successful
Deploy to Production / deploy (push) Successful in 9s
2026-02-26 23:19:31 +00:00
f0dffa6cad feat: implement bulk passwords generator 2026-02-26 23:19:13 +00:00
ace98a0fbc 0.0.2
All checks were successful
Deploy to Production / deploy (push) Successful in 9s
2026-02-26 23:08:04 +00:00
b2ddd95ff5 refactor: improve mobile layout, sticky footer, and UI spacing 2026-02-26 23:07:51 +00:00
38 changed files with 8425 additions and 30 deletions

4
.gitignore vendored
View File

@@ -22,3 +22,7 @@ dist-ssr
*.njsproj *.njsproj
*.sln *.sln
*.sw? *.sw?
dev-dist
extension-release.zip
*.zip
tools-app-extension-*.zip

View File

@@ -1 +1,67 @@
# Tools App # Tools App 🛠️
A collection of useful developer tools in one place. Built with Vue 3 and Vite.
**Live App:** [https://tools.7u.pl/](https://tools.7u.pl/)
## Available Tools
### 🔐 Bulk Passwords Generator
Generate strong, secure passwords in bulk.
- Customizable character sets (lowercase, uppercase, digits, special characters).
- Option to skip similar characters (e.g., `l`, `1`, `I`, `O`, `0`).
- Adjustable length and quantity.
- Generates thousands of passwords instantly.
### 📋 Clipboard Sniffer
Monitor and capture your clipboard history in real-time.
- **Web Mode:** Works when the tab is active and focused.
- **Background Mode (with Extension):** Captures clipboard changes even when you are working in other applications or tabs.
- Clears history on demand.
- Privacy-focused: Data is processed locally and never sent to any server.
---
## Chrome Extension 🧩
To unlock the full potential of the **Clipboard Sniffer**, you can install the companion Chrome Extension.
### Why install the extension?
By default, web browsers restrict clipboard access to when the tab is active and focused. The **Tools App Extension** runs in the background, allowing the application to detect clipboard changes even when you are using other apps or browsing different websites.
### Features
- **Background Monitoring:** seamlessly captures copied text while you work.
- **Smart Integration:** automatically connects to the Tools App when open.
- **Privacy First:** The extension only communicates with the Tools App (`tools.7u.pl` or `localhost`). No data is sent to third-party servers. [Privacy Policy](https://tools.7u.pl/extension-privacy-policy)
### Installation
1. Download the latest release or build from source.
2. Open Chrome and navigate to `chrome://extensions/`.
3. Enable "Developer mode" in the top right.
4. Click "Load unpacked" and select the `extension` folder (or drag and drop the `.zip` file).
---
## Development
### Project Setup
```bash
npm install
```
### Run for Development
```bash
npm run dev
```
### Build for Production
```bash
npm run build
```
### Build Extension
To create a production-ready zip file for the Chrome Extension:
```bash
python3 scripts/build_extension.py
```
This will generate `extension-release.zip` in the project root.

28
extension/README.md Normal file
View File

@@ -0,0 +1,28 @@
# Tools App Extension 🚀
This is the companion Chrome Extension for **Tools App** ([tools.7u.pl](https://tools.7u.pl/)).
## Overview
**Tools App Extension** enhances your experience with the Tools App, specifically designed to power the **Clipboard Sniffer** tool. By default, web applications can only read your clipboard when the browser tab is active and focused. This extension runs in the background, allowing you to capture clipboard history seamlessly while you work in other applications or browse different websites.
## Key Features
- **Background Clipboard Monitoring:** Automatically captures copied text even when the Tools App tab is in the background.
- **Privacy-First Design:** All clipboard data is processed locally within your browser and sent directly to the open Tools App tab. **No data is ever sent to external servers.**
- **Smart Activation:** The extension only activates when you explicitly start the "Clipboard Sniffer" tool in the web app. It stops monitoring immediately when you stop the tool or close the tab.
- **Visual Feedback:** The extension icon changes to indicate when it is actively monitoring your clipboard.
## How it Works
1. Open [tools.7u.pl](https://tools.7u.pl/) and navigate to the **Clipboard Sniffer** tool.
2. The web app will detect the extension and show a "Connected" status.
3. Click "Start Sniffing". The extension will begin monitoring clipboard changes in the background.
4. Any text you copy (from any application) will appear in the Tools App list.
5. Click "Stop Sniffing" to disable monitoring.
## Permissions Explained
- **Read data from the clipboard (`clipboardRead`):** Essential for detecting when you copy text.
- **Run scripts on tools.7u.pl (`scripting`):** Allows the extension to communicate with the Tools App web page to send clipboard updates.
- **Storage:** Used to save user preferences (e.g., sound notifications).
- **Alarms:** Used to keep the background process alive while sniffing is active (to prevent timeout).
## Privacy Policy
We value your privacy. This extension does not collect, store, or transmit any personal data. Clipboard content is only temporarily read and passed to the local instance of the Tools App running in your browser tab.

167
extension/background.js Normal file
View File

@@ -0,0 +1,167 @@
// background.js
// Listen for messages from content scripts or offscreen document
let isSniffing = false;
let lastClipboardContent = '';
let creatingOffscreenDocument;
// Hot-reconnect: Inject content script into existing tabs upon installation/update/restart
const injectContentScriptIfNeeded = async () => {
const tabs = await chrome.tabs.query({ url: ['http://localhost/*', 'http://localhost:*/*', 'https://tools.7u.pl/*'] });
for (const tab of tabs) {
try {
// Try to ping the tab first
try {
await chrome.tabs.sendMessage(tab.id, { action: 'ping' });
// console.log('Content script already active in tab:', tab.id);
} catch (e) {
// If ping fails (no listener), inject script
await chrome.scripting.executeScript({
target: { tabId: tab.id },
files: ['content.js']
});
// console.log('Injected content script into existing tab:', tab.id);
}
} catch (err) {
// console.error('Failed to handle tab:', tab.id, err);
}
}
};
chrome.runtime.onInstalled.addListener(injectContentScriptIfNeeded);
// Also run on startup (when extension is enabled/reloaded)
injectContentScriptIfNeeded();
// Listen for alarms
try {
if (chrome.alarms) {
chrome.alarms.onAlarm.addListener((alarm) => {
if (alarm.name === 'keepAlive') {
refreshOffscreenDocument();
}
});
} else {
// console.warn('chrome.alarms API is not available.');
}
} catch (e) {
// console.error('Error initializing alarms:', e);
}
// Setup offscreen document
async function setupOffscreenDocument(path) {
// Check if an offscreen document already exists
const existingContexts = await chrome.runtime.getContexts({
contextTypes: ['OFFSCREEN_DOCUMENT'],
});
if (existingContexts.length > 0) {
return;
}
// Create an offscreen document
if (creatingOffscreenDocument) {
await creatingOffscreenDocument;
} else {
creatingOffscreenDocument = chrome.offscreen.createDocument({
url: path,
reasons: ['CLIPBOARD', 'AUDIO_PLAYBACK'],
justification: 'To read clipboard content in the background and play notification sounds',
});
await creatingOffscreenDocument;
creatingOffscreenDocument = null;
}
}
// Lifecycle management: Refresh offscreen document every 25s to avoid 30s timeout
async function refreshOffscreenDocument() {
if (isSniffing) {
await chrome.offscreen.closeDocument();
await setupOffscreenDocument('offscreen.html');
}
}
// Start sniffing when requested
chrome.runtime.onMessage.addListener(async (request, sender, sendResponse) => {
if (request.action === 'startSniffing') {
if (isSniffing) {
sendResponse({ status: 'already_started' });
return true;
}
isSniffing = true;
// console.log('Starting sniffing process...');
await setupOffscreenDocument('offscreen.html');
// Setup interval to keep offscreen alive - more aggressive
chrome.alarms.create('keepAlive', { periodInMinutes: 0.1 }); // every 6 seconds
sendResponse({ status: 'started' });
return true;
}
if (request.action === 'stopSniffing') {
if (!isSniffing) {
sendResponse({ status: 'not_running' });
return true;
}
isSniffing = false;
// console.log('Stopping sniffing process...');
// Stop alarm
chrome.alarms.clear('keepAlive');
// Close offscreen document
if (creatingOffscreenDocument) {
await creatingOffscreenDocument;
}
await chrome.offscreen.closeDocument().catch(() => {});
creatingOffscreenDocument = null;
sendResponse({ status: 'stopped' });
return true;
}
if (request.action === 'writeClipboard') {
if (isSniffing) {
chrome.runtime.sendMessage({
target: 'offscreen',
type: 'write-clipboard',
data: request.content
}).catch(() => {});
}
return true;
}
if (request.type === 'clipboard-data' && request.target === 'background') {
// Received data from offscreen document
if (isSniffing && request.data && request.data !== lastClipboardContent) {
lastClipboardContent = request.data;
// console.log('Clipboard changed:', request.data.substring(0, 20) + '...');
// Check if sound should be played
chrome.storage.local.get(['playSound'], (result) => {
if (result.playSound !== false) {
// Send message to offscreen document to play sound
chrome.runtime.sendMessage({
target: 'offscreen',
type: 'play-sound'
});
}
});
// Broadcast to all active tabs (content scripts)
// We could filter by sender.tab.id if we knew which tab started sniffing,
// but broadcasting is simpler for now and covers multiple open tabs of the app.
const tabs = await chrome.tabs.query({ url: ['http://localhost/*', 'http://localhost:*/*', 'https://tools.7u.pl/*'] });
for (const tab of tabs) {
chrome.tabs.sendMessage(tab.id, {
action: 'clipboardUpdate',
content: request.data
}).catch(() => {
// Tab might be closed or content script not injected yet
});
}
}
}
});

77
extension/content.js Normal file
View File

@@ -0,0 +1,77 @@
// content.js
// This script runs on the web app page (e.g. localhost:5173)
console.log('Tools App Extension: Content script injected');
// Listen for messages from the Web App (Vue)
window.addEventListener('message', (event) => {
// We should verify the origin, but since we are running on the page itself, we trust window messages
// from our own app.
if (event.source !== window) return;
if (event.data.type && event.data.type === 'TOOLS_APP_INIT') {
// console.log('Tools App Extension: Received init from Web App');
window.postMessage({ type: 'TOOLS_APP_EXTENSION_READY', version: '1.0' }, '*');
}
// Heartbeat check
if (event.data.type === 'TOOLS_APP_PING') {
try {
// Only respond if the extension context is still valid
if (chrome.runtime && chrome.runtime.id) {
window.postMessage({ type: 'TOOLS_APP_PONG' }, '*');
}
} catch (e) {
// Extension context invalidated
// console.warn('Extension context invalidated during ping');
}
}
// Example: Receive request to sniff clipboard
if (event.data.type === 'TOOLS_APP_START_SNIFFING') {
// console.log('Tools App Extension: Start sniffing request');
// Relay to background script
try {
chrome.runtime.sendMessage({ action: 'startSniffing' });
} catch (e) {
console.warn('Tools App Extension: Connection lost, please reload the page', e);
}
}
if (event.data.type === 'TOOLS_APP_STOP_SNIFFING') {
// console.log('Tools App Extension: Stop sniffing request');
try {
chrome.runtime.sendMessage({ action: 'stopSniffing' });
} catch (e) {
// ignore
}
}
if (event.data.type === 'TOOLS_APP_CLIPBOARD_WRITE') {
try {
chrome.runtime.sendMessage({
action: 'writeClipboard',
content: event.data.content
});
} catch (e) {
console.warn('Tools App Extension: Write clipboard failed', e);
}
}
});
// Listen for messages from the Extension Background
try {
chrome.runtime.onMessage.addListener((request, sender, sendResponse) => {
if (request.action === 'clipboardUpdate') {
// Send to Web App
window.postMessage({ type: 'TOOLS_APP_CLIPBOARD_UPDATE', content: request.content }, '*');
}
// Respond to background ping to confirm we are alive
if (request.action === 'ping') {
sendResponse('pong');
}
});
} catch (e) {
console.warn('Tools App Extension: Could not add listener', e);
}

BIN
extension/icon-128.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 11 KiB

BIN
extension/icon-16.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 862 B

BIN
extension/icon-48.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.6 KiB

48
extension/manifest.json Normal file
View File

@@ -0,0 +1,48 @@
{
"manifest_version": 3,
"name": "Tools App Extension",
"version": "1.1",
"description": "Browser extension for Tools App",
"permissions": [
"clipboardRead",
"clipboardWrite",
"offscreen",
"storage",
"alarms",
"scripting"
],
"host_permissions": [
"http://localhost/*",
"http://localhost:*/*",
"https://tools.7u.pl/*"
],
"background": {
"service_worker": "background.js"
},
"content_scripts": [
{
"matches": [
"http://localhost/*",
"http://localhost:*/*",
"https://tools.7u.pl/*"
],
"js": [
"content.js"
],
"run_at": "document_start"
}
],
"action": {
"default_popup": "popup.html",
"default_icon": {
"16": "icon-16.png",
"48": "icon-48.png",
"128": "icon-128.png"
}
},
"icons": {
"16": "icon-16.png",
"48": "icon-48.png",
"128": "icon-128.png"
}
}

10
extension/offscreen.html Normal file
View File

@@ -0,0 +1,10 @@
<!DOCTYPE html>
<html>
<head>
<title>Offscreen Clipboard Access</title>
</head>
<body>
<textarea id="text"></textarea>
<script src="offscreen.js"></script>
</body>
</html>

84
extension/offscreen.js Normal file
View File

@@ -0,0 +1,84 @@
// offscreen.js
// This script runs in the offscreen document to access DOM APIs like navigator.clipboard
const textEl = document.querySelector('#text');
let lastText = '';
setInterval(async () => {
try {
textEl.focus();
textEl.value = '';
textEl.select();
// Method 1: execCommand
try {
document.execCommand('paste');
} catch (e) {
// Ignore
}
let text = textEl.value;
// Method 2: navigator.clipboard (Fallback)
if (!text) {
try {
text = await navigator.clipboard.readText();
} catch (e) {
// Silent fail for navigator
}
}
if (text && text.trim().length > 0 && text !== lastText) {
lastText = text;
chrome.runtime.sendMessage({
type: 'clipboard-data',
target: 'background',
data: text
}).catch(() => {});
}
} catch (error) {
// Ignore critical errors to keep running
}
}, 50);
// Listen for messages from background
chrome.runtime.onMessage.addListener((message) => {
if (message.target === 'offscreen') {
// Handle commands
if (message.type === 'play-sound') {
playNotificationSound();
} else if (message.type === 'write-clipboard') {
try {
const text = message.data;
if (text) {
textEl.value = text;
textEl.select();
document.execCommand('copy');
lastText = text; // Update internal state to avoid re-triggering update
}
} catch (e) {
console.error('Failed to write clipboard:', e);
}
}
}
});
function playNotificationSound() {
const audioContext = new (window.AudioContext || window.webkitAudioContext)();
const oscillator = audioContext.createOscillator();
const gainNode = audioContext.createGain();
oscillator.connect(gainNode);
gainNode.connect(audioContext.destination);
oscillator.type = 'sine';
oscillator.frequency.setValueAtTime(500, audioContext.currentTime);
oscillator.frequency.exponentialRampToValueAtTime(1000, audioContext.currentTime + 0.1);
gainNode.gain.setValueAtTime(0.1, audioContext.currentTime);
gainNode.gain.exponentialRampToValueAtTime(0.01, audioContext.currentTime + 0.1);
oscillator.start();
oscillator.stop(audioContext.currentTime + 0.1);
}

55
extension/popup.html Normal file
View File

@@ -0,0 +1,55 @@
<!DOCTYPE html>
<html>
<head>
<title>Tools App Extension</title>
<style>
body {
width: 250px;
padding: 15px;
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;
background-color: #f5f5f5;
user-select: none;
}
h3 {
margin-top: 0;
color: #333;
}
.status {
padding: 10px;
border-radius: 6px;
background: #e0e0e0;
margin-top: 10px;
font-size: 14px;
}
.status.active {
background: #e3f2fd;
color: #1565c0;
border: 1px solid #90caf9;
}
.footer {
margin-top: 15px;
font-size: 12px;
color: #666;
text-align: center;
}
</style>
</head>
<body>
<h3>Tools App Extension</h3>
<div class="status active">
Extension is active and ready to communicate with Tools App.
</div>
<div style="margin-top: 15px;">
<label style="display: flex; align-items: center; cursor: pointer;">
<input type="checkbox" id="soundToggle" style="margin-right: 8px;">
<span>Play sound on capture</span>
</label>
</div>
<div class="footer">
Visit <a href="https://tools.7u.pl" target="_blank">tools.7u.pl</a>
</div>
<script src="popup.js"></script>
</body>
</html>

34
extension/popup.js Normal file
View File

@@ -0,0 +1,34 @@
// popup.js
document.addEventListener('DOMContentLoaded', () => {
const soundToggle = document.getElementById('soundToggle');
// Load saved setting
chrome.storage.local.get(['playSound'], (result) => {
soundToggle.checked = result.playSound !== false; // Default to true
});
// Save setting on change
soundToggle.addEventListener('change', () => {
chrome.storage.local.set({ playSound: soundToggle.checked });
// Play test sound if enabled
if (soundToggle.checked) {
const audioContext = new (window.AudioContext || window.webkitAudioContext)();
const oscillator = audioContext.createOscillator();
const gainNode = audioContext.createGain();
oscillator.connect(gainNode);
gainNode.connect(audioContext.destination);
oscillator.type = 'sine';
oscillator.frequency.setValueAtTime(500, audioContext.currentTime);
oscillator.frequency.exponentialRampToValueAtTime(1000, audioContext.currentTime + 0.1);
gainNode.gain.setValueAtTime(0.1, audioContext.currentTime);
gainNode.gain.exponentialRampToValueAtTime(0.01, audioContext.currentTime + 0.1);
oscillator.start();
oscillator.stop(audioContext.currentTime + 0.1);
}
});
});

View File

@@ -2,6 +2,9 @@
<html lang="en"> <html lang="en">
<head> <head>
<meta charset="UTF-8" /> <meta charset="UTF-8" />
<link rel="icon" type="image/svg+xml" href="/favicon.svg" />
<link rel="apple-touch-icon" href="/favicon.svg" />
<meta name="theme-color" content="#4facfe" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" /> <meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Tools App</title> <title>Tools App</title>
</head> </head>

4776
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -1,7 +1,7 @@
{ {
"name": "tools-app", "name": "tools-app",
"private": true, "private": true,
"version": "0.0.1", "version": "0.4.4",
"type": "module", "type": "module",
"scripts": { "scripts": {
"dev": "vite", "dev": "vite",
@@ -15,6 +15,7 @@
}, },
"devDependencies": { "devDependencies": {
"@vitejs/plugin-vue": "^6.0.2", "@vitejs/plugin-vue": "^6.0.2",
"vite": "^7.3.1" "vite": "^7.3.1",
"vite-plugin-pwa": "^1.2.0"
} }
} }

20
public/favicon.svg Normal file
View File

@@ -0,0 +1,20 @@
<svg width="512" height="512" viewBox="0 0 512 512" fill="none" xmlns="http://www.w3.org/2000/svg">
<defs>
<linearGradient id="grad1" x1="0%" y1="0%" x2="100%" y2="100%">
<stop offset="0%" style="stop-color:#6366f1;stop-opacity:1" />
<stop offset="100%" style="stop-color:#0ea5e9;stop-opacity:1" />
</linearGradient>
<filter id="shadow" x="-20%" y="-20%" width="140%" height="140%">
<feDropShadow dx="0" dy="4" stdDeviation="4" flood-opacity="0.25"/>
</filter>
</defs>
<!-- Background Circle with Gradient -->
<circle cx="256" cy="256" r="240" fill="url(#grad1)" />
<!-- Gear Icon -->
<g transform="translate(106, 106) scale(0.6)" filter="url(#shadow)">
<path fill="white" d="M487.4 315.7l-42.6-24.6c4.3-23.2 4.3-47 0-70.2l42.6-24.6c4.9-2.8 7.1-8.6 5.5-14-11.1-35.6-30-67.8-54.7-94.6-3.8-4.1-10-5.1-14.8-2.3L380.8 110c-17.9-15.4-38.5-27.3-60.8-35.1V25.8c0-5.6-3.9-10.5-9.4-11.7-36.7-8.2-74.3-7.8-109.2 0-5.5 1.2-9.4 6.1-9.4 11.7V75c-22.2 7.9-42.8 19.8-60.8 35.1L88.7 85.5c-4.9-2.8-11-1.9-14.8 2.3-24.7 26.7-43.6 58.9-54.7 94.6-1.7 5.4.6 11.2 5.5 14L67.3 221c-4.3 23.2-4.3 47 0 70.2l-42.6 24.6c-4.9 2.8-7.1 8.6-5.5 14 11.1 35.6 30 67.8 54.7 94.6 3.8 4.1 10 5.1 14.8 2.3l42.6-24.6c17.9 15.4 38.5 27.3 60.8 35.1v49.2c0 5.6 3.9 10.5 9.4 11.7 36.7 8.2 74.3 7.8 109.2 0 5.5-1.2 9.4-6.1 9.4-11.7v-49.2c22.2-7.9 42.8-19.8 60.8-35.1l42.6 24.6c4.9 2.8 11 1.9 14.8-2.3 24.7-26.7 43.6-58.9 54.7-94.6 1.5-5.5-.7-11.3-5.6-14.1zM256 336c-44.1 0-80-35.9-80-80s35.9-80 80-80 80 35.9 80 80-35.9 80-80 80z"/>
</g>
</svg>

After

Width:  |  Height:  |  Size: 1.5 KiB

View File

@@ -0,0 +1,58 @@
import json
import os
import shutil
import zipfile
# Configuration
SOURCE_DIR = "extension"
BUILD_DIR = "dist-extension"
OUTPUT_ZIP = "extension-release.zip"
MANIFEST_FILE = "manifest.json"
# Remove build directory if exists
if os.path.exists(BUILD_DIR):
shutil.rmtree(BUILD_DIR)
# Copy source directory to build directory
shutil.copytree(SOURCE_DIR, BUILD_DIR)
# Modify manifest.json for production
manifest_path = os.path.join(BUILD_DIR, MANIFEST_FILE)
with open(manifest_path, "r") as f:
manifest = json.load(f)
# Filter permissions and content scripts (remove localhost)
print("Removing localhost from manifest...")
# Filter host_permissions
if "host_permissions" in manifest:
manifest["host_permissions"] = [
perm for perm in manifest["host_permissions"]
if "localhost" not in perm
]
# Filter content_scripts matches
if "content_scripts" in manifest:
for script in manifest["content_scripts"]:
if "matches" in script:
script["matches"] = [
match for match in script["matches"]
if "localhost" not in match
]
# Save modified manifest
with open(manifest_path, "w") as f:
json.dump(manifest, f, indent=2)
# Create ZIP file
print(f"Creating {OUTPUT_ZIP}...")
with zipfile.ZipFile(OUTPUT_ZIP, "w", zipfile.ZIP_DEFLATED) as zipf:
for root, dirs, files in os.walk(BUILD_DIR):
for file in files:
file_path = os.path.join(root, file)
arcname = os.path.relpath(file_path, BUILD_DIR)
zipf.write(file_path, arcname)
# Cleanup
shutil.rmtree(BUILD_DIR)
print(f"Done! {OUTPUT_ZIP} is ready for upload to Chrome Web Store.")

View File

@@ -0,0 +1,61 @@
from PIL import Image, ImageOps
def resize_image_canvas(input_path, output_path, canvas_width, canvas_height, bg_color=(255, 255, 255)):
"""
Resizes an image to fit within a specific canvas size, centering it and adding padding.
"""
try:
original_image = Image.open(input_path)
# Calculate aspect ratios
img_ratio = original_image.width / original_image.height
canvas_ratio = canvas_width / canvas_height
# Determine new dimensions
if img_ratio > canvas_ratio:
# Image is wider than canvas
new_width = canvas_width
new_height = int(canvas_width / img_ratio)
else:
# Image is taller than canvas
new_height = canvas_height
new_width = int(canvas_height * img_ratio)
# Resize the original image
resized_image = original_image.resize((new_width, new_height), Image.Resampling.LANCZOS)
# Create a new canvas with the specified background color
new_image = Image.new("RGB", (canvas_width, canvas_height), bg_color)
# Calculate position to center the image
x_offset = (canvas_width - new_width) // 2
y_offset = (canvas_height - new_height) // 2
# Paste the resized image onto the canvas
new_image.paste(resized_image, (x_offset, y_offset))
# Save the result
new_image.save(output_path, quality=95)
print(f"Success! Image saved to {output_path}")
except Exception as e:
print(f"Error: {e}")
if __name__ == "__main__":
# Example usage:
# Replace 'screenshot.png' with your actual file name
# We will look for png or jpg files in current dir if not specified
import glob
import sys
files = glob.glob("*.png") + glob.glob("*.jpg") + glob.glob("*.jpeg")
if not files:
print("No image files found in the current directory.")
sys.exit(1)
print("Found images:", files)
input_file = files[0] # Take the first one found
print(f"Processing {input_file}...")
resize_image_canvas(input_file, "cws_screenshot_1280x800.png", 1280, 800, bg_color=(245, 247, 250)) # Light gray-ish background matching the app

View File

@@ -1,21 +1,60 @@
<script setup> <script setup>
import { ref } from 'vue' import { ref, onMounted, onUnmounted } from 'vue'
import { useRouter } from 'vue-router'
import Header from './components/Header.vue' import Header from './components/Header.vue'
import Footer from './components/Footer.vue' import Footer from './components/Footer.vue'
import Sidebar from './components/Sidebar.vue' import Sidebar from './components/Sidebar.vue'
import InstallPrompt from './components/InstallPrompt.vue'
import ReloadPrompt from './components/ReloadPrompt.vue'
const isSidebarOpen = ref(true) const isSidebarOpen = ref(window.innerWidth >= 768)
const router = useRouter()
// Close sidebar on route change for mobile
router.afterEach(() => {
if (window.innerWidth < 768) {
isSidebarOpen.value = false
}
})
let lastWidth = window.innerWidth
const handleResize = () => {
const width = window.innerWidth
// Only change state when crossing the breakpoint
if (width >= 768 && lastWidth < 768) {
isSidebarOpen.value = true
} else if (width < 768 && lastWidth >= 768) {
isSidebarOpen.value = false
}
lastWidth = width
}
onMounted(() => {
window.addEventListener('resize', handleResize)
})
onUnmounted(() => {
window.removeEventListener('resize', handleResize)
})
</script> </script>
<template> <template>
<Header @toggle-sidebar="isSidebarOpen = !isSidebarOpen" /> <Header @toggle-sidebar="isSidebarOpen = !isSidebarOpen" />
<div class="app-body"> <div class="app-body">
<div
class="sidebar-overlay"
:class="{ 'is-visible': isSidebarOpen }"
@click="isSidebarOpen = false"
></div>
<Sidebar :is-open="isSidebarOpen" /> <Sidebar :is-open="isSidebarOpen" />
<main class="main-content"> <main class="main-content">
<router-view /> <router-view />
</main> </main>
</div> </div>
<Footer /> <Footer />
<InstallPrompt />
<ReloadPrompt />
</template> </template>
<style scoped> <style scoped>
@@ -30,6 +69,52 @@ const isSidebarOpen = ref(true)
flex: 1; flex: 1;
padding: 2rem; padding: 2rem;
width: 100%; width: 100%;
max-width: 100%; /* Ensure it doesn't overflow */ max-width: 100%;
/* Space for fixed footer on mobile + extra margin (match top padding 2rem + footer height ~40px) */
padding-bottom: calc(2rem + 40px + env(safe-area-inset-bottom));
}
@media (max-width: 640px) {
.main-content {
padding: 1rem;
padding-bottom: calc(1rem + 40px + env(safe-area-inset-bottom));
}
}
@media (min-width: 768px) {
.app-body {
overflow: hidden;
}
.main-content {
overflow-y: auto;
height: 100%;
padding-bottom: 2rem;
}
}
.sidebar-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background-color: rgba(0, 0, 0, 0.5);
z-index: 900; /* Below sidebar (1000), above content */
opacity: 0;
pointer-events: none;
transition: opacity 0.3s ease;
backdrop-filter: blur(2px);
}
.sidebar-overlay.is-visible {
opacity: 1;
pointer-events: auto;
}
@media (min-width: 768px) {
.sidebar-overlay {
display: none;
}
} }
</style> </style>

View File

@@ -4,7 +4,7 @@ const version = __APP_VERSION__;
</script> </script>
<template> <template>
<footer class="app-footer unselectable"> <footer class="app-footer glass-panel unselectable">
<div class="footer-content"> <div class="footer-content">
<p>&copy; {{ currentYear }} Tools App. v{{ version }}</p> <p>&copy; {{ currentYear }} Tools App. v{{ version }}</p>
</div> </div>
@@ -14,13 +14,29 @@ const version = __APP_VERSION__;
<style scoped> <style scoped>
.app-footer { .app-footer {
width: 100%; width: 100%;
padding: 1rem; padding: 0.5rem;
background-color: #242424; /* Background handled by glass-panel */
border-top: 1px solid rgba(255, 255, 255, 0.1); border-left: none;
border-right: none;
border-bottom: none;
border-radius: 0;
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
margin-top: auto; /* Push to bottom in flex container */ z-index: 10;
/* Remove fixed height to allow content to dictate size */
/* height: 30px; */
position: fixed;
bottom: 0;
left: 0;
padding-bottom: max(0.5rem, env(safe-area-inset-bottom));
}
@media (min-width: 768px) {
.app-footer {
position: static;
margin-top: auto;
}
} }
.footer-content { .footer-content {
@@ -31,7 +47,7 @@ const version = __APP_VERSION__;
p { p {
font-size: 0.9rem; font-size: 0.9rem;
color: rgba(255, 255, 255, 0.6); color: var(--text-muted);
margin: 0; margin: 0;
font-family: monospace; font-family: monospace;
} }

View File

@@ -31,7 +31,13 @@ onMounted(() => {
<header class="app-header glass-panel unselectable"> <header class="app-header glass-panel unselectable">
<div class="header-content"> <div class="header-content">
<div class="header-left"> <div class="header-left">
<button class="menu-btn" @click="$emit('toggleSidebar')" aria-label="Toggle Menu" v-ripple> <button
class="menu-btn"
@click="$emit('toggleSidebar')"
aria-label="Toggle Menu"
title="Toggle Menu"
v-ripple
>
<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"> <svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round">
<line x1="3" y1="12" x2="21" y2="12"></line> <line x1="3" y1="12" x2="21" y2="12"></line>
<line x1="3" y1="6" x2="21" y2="6"></line> <line x1="3" y1="6" x2="21" y2="6"></line>
@@ -61,7 +67,8 @@ onMounted(() => {
color: var(--text-color); color: var(--text-color);
padding: 1rem; padding: 1rem;
/* box-shadow handled by glass-panel class */ /* box-shadow handled by glass-panel class */
position: relative; position: sticky;
top: 0;
z-index: 100; z-index: 100;
border-left: none; border-left: none;
border-right: none; border-right: none;

View File

@@ -0,0 +1,153 @@
<template>
<div v-if="showInstallPrompt" class="install-prompt glass-panel unselectable">
<div class="prompt-content">
<span class="prompt-text">Install app for faster access</span>
<div class="prompt-actions">
<button @click="installPWA" class="install-btn">Install</button>
<button @click="dismissPrompt" class="dismiss-btn"></button>
</div>
</div>
</div>
</template>
<script setup>
import { ref, onMounted, onBeforeUnmount } from 'vue'
const showInstallPrompt = ref(false)
let deferredPrompt = null
const handleBeforeInstallPrompt = (e) => {
// Prevent Chrome 67 and earlier from automatically showing the prompt
e.preventDefault()
// Only show install prompt if we are on the specific domain
if (window.location.hostname !== 'tools.7u.pl') {
return
}
// Stash the event so it can be triggered later.
deferredPrompt = e
// Update UI to notify the user they can add to home screen
showInstallPrompt.value = true
}
const installPWA = async () => {
if (!deferredPrompt) return
// Show the install prompt
deferredPrompt.prompt()
// Wait for the user to respond to the prompt
const { outcome } = await deferredPrompt.userChoice
console.log(`User response to the install prompt: ${outcome}`)
// We've used the prompt, and can't use it again, throw it away
deferredPrompt = null
showInstallPrompt.value = false
}
const dismissPrompt = () => {
showInstallPrompt.value = false
deferredPrompt = null
}
onMounted(() => {
window.addEventListener('beforeinstallprompt', handleBeforeInstallPrompt)
})
onBeforeUnmount(() => {
window.removeEventListener('beforeinstallprompt', handleBeforeInstallPrompt)
})
</script>
<style scoped>
.install-prompt {
position: fixed;
bottom: calc(5rem + env(safe-area-inset-bottom)); /* Above footer with extra space */
left: 50%;
transform: translateX(-50%);
width: 90%;
max-width: 400px;
padding: 1rem;
z-index: 9999;
animation: slideUp 0.3s ease-out;
border-radius: 12px;
}
.prompt-content {
display: flex;
align-items: center;
justify-content: space-between;
gap: 1rem;
}
.prompt-text {
font-size: 0.9rem;
font-weight: 500;
color: var(--text-strong);
}
.prompt-actions {
display: flex;
align-items: center;
gap: 0.5rem;
}
.install-btn {
background-color: var(--primary-accent);
color: #000; /* Dark text on bright cyan accent for dark mode */
border: none;
padding: 0.5rem 1rem;
border-radius: 6px;
font-size: 0.9rem;
font-weight: 600;
cursor: pointer;
white-space: nowrap;
}
.dismiss-btn {
background: transparent;
border: none;
color: var(--text-secondary);
font-size: 1.2rem;
cursor: pointer;
padding: 0 0.5rem;
}
@keyframes slideUp {
from {
opacity: 0;
transform: translate(-50%, 20px);
}
to {
opacity: 1;
transform: translate(-50%, 0);
}
}
@media (min-width: 768px) {
.install-prompt {
bottom: 2rem;
right: 2rem;
left: auto;
transform: none;
}
@keyframes slideUp {
from {
opacity: 0;
transform: translateY(20px);
}
to {
opacity: 1;
transform: translateY(0);
}
}
}
/* Light theme overrides for better contrast */
:global(:root[data-theme="light"]) .install-btn {
background-color: var(--accent-purple);
color: #fff;
}
</style>

View File

@@ -0,0 +1,94 @@
<script setup>
import { useRegisterSW } from 'virtual:pwa-register/vue'
import { watch, onMounted } from 'vue'
// Zmieniamy na autoUpdate w configu, więc tutaj tylko nasłuchujemy i ewentualnie wymuszamy
const {
needRefresh,
updateServiceWorker,
} = useRegisterSW({
immediate: true,
onRegistered(r) {
// Sprawdzaj aktualizacje co godzinę
r && setInterval(() => {
r.update()
}, 60 * 60 * 1000)
}
})
const updateSW = async () => {
if ('serviceWorker' in navigator) {
try {
const registration = await navigator.serviceWorker.ready
console.log('Checking for SW update...')
await registration.update()
} catch (e) {
console.error('Failed to update SW:', e)
}
}
}
onMounted(() => {
// Check on load
updateSW()
// Check when app becomes visible again (e.g. switching tabs/apps)
document.addEventListener('visibilitychange', () => {
if (document.visibilityState === 'visible') {
updateSW()
}
})
})
</script>
<template>
<div
v-if="needRefresh"
class="pwa-toast"
role="alert"
>
<div class="message">
New content available, click on reload button to update.
</div>
<button @click="updateServiceWorker()">
Reload
</button>
</div>
</template>
<style scoped>
.pwa-toast {
position: fixed;
right: 0;
bottom: 0;
margin: 16px;
padding: 12px;
border: 1px solid var(--glass-border);
border-radius: 4px;
z-index: 10000;
text-align: left;
box-shadow: var(--glass-shadow);
background-color: var(--glass-bg);
backdrop-filter: blur(10px);
color: var(--text-color);
display: flex;
flex-direction: column;
gap: 8px;
}
.pwa-toast button {
border: 1px solid var(--glass-border);
outline: none;
margin-right: 5px;
border-radius: 2px;
padding: 3px 10px;
cursor: pointer;
background: var(--button-bg);
color: var(--text-color);
transition: all 0.2s;
}
.pwa-toast button:hover {
background: var(--button-hover-bg);
}
</style>

View File

@@ -11,26 +11,50 @@ defineProps({
<aside class="sidebar unselectable" :class="{ 'is-open': isOpen }"> <aside class="sidebar unselectable" :class="{ 'is-open': isOpen }">
<nav class="sidebar-nav"> <nav class="sidebar-nav">
<router-link to="/passwords" class="nav-item" v-ripple>Passwords</router-link> <router-link to="/passwords" class="nav-item" v-ripple>Passwords</router-link>
<router-link to="/clipboard-sniffer" class="nav-item" v-ripple>Clipboard Sniffer</router-link>
<router-link to="/url-cleaner" class="nav-item" v-ripple>URL Cleaner</router-link>
</nav> </nav>
</aside> </aside>
</template> </template>
<style scoped> <style scoped>
.sidebar { .sidebar {
width: 0; /* Mobile First Styles */
position: fixed;
top: 0;
left: 0;
bottom: 0;
width: 250px;
background-color: var(--panel-bg); background-color: var(--panel-bg);
backdrop-filter: blur(12px); backdrop-filter: blur(12px);
-webkit-backdrop-filter: blur(12px); -webkit-backdrop-filter: blur(12px);
overflow-x: hidden; overflow-x: hidden;
transition: width 0.3s ease; transition: transform 0.3s ease;
border-right: 1px solid var(--panel-border); border-right: 1px solid var(--panel-border);
flex-shrink: 0; z-index: 1000;
height: 100%;
box-shadow: var(--panel-shadow); box-shadow: var(--panel-shadow);
transform: translateX(-100%);
} }
.sidebar.is-open { .sidebar.is-open {
width: 250px; transform: translateX(0);
}
@media (min-width: 768px) {
.sidebar {
position: static;
transform: none;
width: 0;
height: auto;
transition: width 0.3s ease;
z-index: auto;
box-shadow: none; /* Shadow handled by panel logic or not needed inline */
}
.sidebar.is-open {
width: 250px;
transform: none;
}
} }
.sidebar-nav { .sidebar-nav {
@@ -38,11 +62,19 @@ defineProps({
display: flex; display: flex;
flex-direction: column; flex-direction: column;
width: 250px; /* Fixed width for content */ width: 250px; /* Fixed width for content */
min-width: 250px;
padding-top: 1rem; /* Add some top padding for mobile aesthetic */
}
@media (min-width: 768px) {
.sidebar-nav {
padding-top: 0;
}
} }
.nav-item { .nav-item {
display: block; display: block;
padding: 0.5rem; padding: 0.75rem 1rem;
color: var(--text-secondary); color: var(--text-secondary);
text-decoration: none; text-decoration: none;
transition: background-color 0.2s, color 0.2s; transition: background-color 0.2s, color 0.2s;

View File

@@ -0,0 +1,157 @@
<script setup>
import { ref, watch, nextTick } from 'vue'
import { useFillHeight } from '../../composables/useFillHeight'
import { useExtension } from '../../composables/useExtension'
import { useLocalStorage } from '../../composables/useLocalStorage'
import ExtensionStatus from './common/ExtensionStatus.vue'
const clipboardContent = useLocalStorage('clipboard-sniffer-content', '')
const textareaRef = ref(null)
const {
isExtensionReady,
isListening,
lastClipboardText,
startListening,
stopListening
} = useExtension()
const { height: textareaHeight } = useFillHeight(textareaRef, 40)
// Watch for clipboard updates from extension
watch(lastClipboardText, (newText) => {
if (newText) {
clipboardContent.value += (clipboardContent.value ? '\n' : '') + newText
scrollToBottom()
}
})
const scrollToBottom = () => {
nextTick(() => {
const textarea = document.querySelector('.tool-textarea')
if (textarea) {
textarea.scrollTop = textarea.scrollHeight
}
})
}
const copyToClipboard = () => {
if (clipboardContent.value) {
navigator.clipboard.writeText(clipboardContent.value)
}
}
const clearText = () => {
clipboardContent.value = ''
}
</script>
<template>
<div class="tool-container full-width">
<div class="tool-panel">
<div class="tool-header">
<h2 class="tool-title">Clipboard Sniffer</h2>
<div class="extension-indicator-wrapper">
<ExtensionStatus :isReady="isExtensionReady" />
</div>
</div>
<div class="controls">
<button
v-if="!isListening"
class="btn-neon"
@click="startListening"
:disabled="!isExtensionReady"
:title="!isExtensionReady ? 'Extension required' : 'Start capturing clipboard'"
v-ripple
>
Start Sniffing
</button>
<button
v-else
class="btn-neon danger"
@click="stopListening"
v-ripple
>
Stop Sniffing
</button>
<button class="btn-neon" @click="copyToClipboard" v-ripple>
Copy
</button>
<button class="btn-neon" @click="clearText" v-ripple>
Clear
</button>
</div>
<div class="result-area" style="margin-top: 1rem;">
<div ref="textareaRef" :style="{ height: textareaHeight, width: '100%' }">
<textarea
v-model="clipboardContent"
class="tool-textarea"
placeholder="Clipboard content will appear here line by line..."
></textarea>
</div>
</div>
</div>
</div>
</template>
<style scoped>
.tool-header {
display: flex;
justify-content: center;
align-items: center;
position: relative;
width: 100%;
}
.controls {
display: flex;
justify-content: center;
gap: 1rem;
flex-wrap: wrap;
}
.tool-container.full-width {
max-width: 100%;
height: 100%;
display: flex;
flex-direction: column;
}
.tool-panel {
display: flex;
flex-direction: column;
height: 100%;
gap: 1.5rem;
}
.tool-textarea {
width: 100%;
height: 100%;
padding: 1rem;
background-color: var(--toggle-bg);
border: 1px solid var(--toggle-border);
border-radius: 8px;
color: var(--text-color);
font-family: monospace;
resize: none;
font-size: 0.9rem;
line-height: 1.5;
}
.tool-textarea:focus {
outline: none;
border-color: var(--primary-accent);
}
.extension-indicator-wrapper {
position: absolute;
right: 0;
top: 50%;
transform: translateY(-50%);
display: flex;
}
</style>

View File

@@ -1,15 +1,398 @@
<script setup> <script setup>
import { ref } from 'vue';
import { useFillHeight } from '../../composables/useFillHeight';
// Options
const useLower = ref(true);
const useUpper = ref(true);
const useDigits = ref(true);
const useSpecial = ref(false);
const skipSimilar = ref(true);
const length = ref(16);
const count = ref(20);
// Result
const result = ref('');
const textareaRef = ref(null);
const { height: textareaHeight } = useFillHeight(textareaRef, 40);
// Character Sets
const CHARS_LOWER = 'abcdefghijklmnopqrstuvwxyz';
const CHARS_UPPER = 'ABCDEFGHIJKLMNOPQRSTUVWXYZ';
const CHARS_DIGITS = '0123456789';
const CHARS_SPECIAL = '!@#$%^&*()_+~`|}{[]:;?><,./-=';
const CHARS_SIMILAR = /[Il1O0o]/g;
const generatePasswords = () => {
let charset = '';
if (useLower.value) charset += CHARS_LOWER;
if (useUpper.value) charset += CHARS_UPPER;
if (useDigits.value) charset += CHARS_DIGITS;
if (useSpecial.value) charset += CHARS_SPECIAL;
if (skipSimilar.value) {
charset = charset.replace(CHARS_SIMILAR, '');
}
if (!charset) {
result.value = 'Please select at least one character set.';
return;
}
const passwords = [];
const charsetLength = charset.length;
// We need (count * length) random values.
// However, crypto.getRandomValues has a limit on array size (65536 bytes).
// So we'll generate one random value per character needed in a loop or batch.
// For simplicity and safety, let's generate per password or per character if needed.
// The most robust way for "bulk" is to just loop.
for (let i = 0; i < count.value; i++) {
let password = '';
// Create a typed array for the random values for this password
const randomValues = new Uint32Array(length.value);
crypto.getRandomValues(randomValues);
for (let j = 0; j < length.value; j++) {
// Use modulo to map the random 32-bit integer to a charset index
// Note: This introduces a tiny bias if charsetLength is not a power of 2,
// but for typical password generation it is negligible compared to Math.random().
// For cryptographic perfection, rejection sampling would be needed,
// but this is standard practice for password managers.
const randomIndex = randomValues[j] % charsetLength;
password += charset[randomIndex];
}
passwords.push(password);
}
result.value = passwords.join('\n');
};
</script> </script>
<template> <template>
<div class="passwords-tool"> <div class="tool-container full-width">
<h2>Passwords Generator</h2> <div class="tool-panel">
<p>This is the passwords generator tool (work in progress).</p> <div class="panel-header">
<h2 class="tool-title">Bulk Passwords Generator</h2>
<div class="action-area desktop-only">
<button class="btn-neon generate-btn" @click="generatePasswords" v-ripple>
Generate
</button>
</div>
</div>
<div class="options-grid">
<div class="checkbox-group">
<label class="checkbox-label">
<input type="checkbox" v-model="useLower">
<span class="checkmark"></span>
Lower Case (a-z)
</label>
<label class="checkbox-label">
<input type="checkbox" v-model="useUpper">
<span class="checkmark"></span>
Upper Case (A-Z)
</label>
<label class="checkbox-label">
<input type="checkbox" v-model="useDigits">
<span class="checkmark"></span>
Digits (0-9)
</label>
<label class="checkbox-label">
<input type="checkbox" v-model="useSpecial">
<span class="checkmark"></span>
Special Chars
</label>
<label class="checkbox-label">
<input type="checkbox" v-model="skipSimilar">
<span class="checkmark"></span>
Skip Similar (I, l, 1, O, 0, o)
</label>
</div>
<div class="inputs-group">
<div class="input-wrapper">
<label>Length</label>
<div class="number-control">
<button class="control-btn" @click="length > 4 ? length-- : null">-</button>
<input type="number" v-model="length" min="4" max="128" class="number-input">
<button class="control-btn" @click="length < 128 ? length++ : null">+</button>
</div>
</div>
<div class="input-wrapper">
<label>Count</label>
<div class="number-control">
<button class="control-btn" @click="count > 1 ? count-- : null">-</button>
<input type="number" v-model="count" min="1" max="1000" class="number-input">
<button class="control-btn" @click="count < 1000 ? count++ : null">+</button>
</div>
</div>
</div>
</div>
<div class="mobile-only" style="margin-top: 1rem; width: 100%;">
<button class="btn-neon generate-btn" @click="generatePasswords" v-ripple style="width: 100%;">
Generate
</button>
</div>
<div class="result-area" :style="{ height: textareaHeight }">
<textarea
class="tool-textarea"
v-model="result"
placeholder="Generated passwords will appear here..."
readonly
></textarea>
</div>
</div>
</div> </div>
</template> </template>
<style scoped> <style scoped>
.passwords-tool { .tool-container.full-width {
max-width: 100%;
height: 100%;
display: flex;
flex-direction: column;
}
.tool-panel {
display: flex;
flex-direction: column;
height: 100%;
gap: 1.5rem;
}
.panel-header {
display: flex;
justify-content: space-between;
align-items: center;
flex-wrap: wrap;
gap: 1rem;
}
.tool-title {
margin: 0;
}
.options-grid {
display: flex;
flex-wrap: wrap;
gap: 2rem;
padding: 1rem; padding: 1rem;
background: var(--toggle-bg);
border: 1px solid var(--toggle-border);
border-radius: 12px;
}
.checkbox-group {
display: flex;
flex-wrap: wrap;
gap: 1.5rem;
flex: 2;
}
.inputs-group {
display: flex;
gap: 1rem;
flex: 1;
min-width: 200px;
flex-wrap: wrap;
}
.input-wrapper {
display: flex;
flex-direction: column;
gap: 0.5rem;
flex: 1;
min-width: 140px;
}
.checkbox-label {
display: flex;
align-items: center;
gap: 10px;
cursor: pointer;
user-select: none;
color: var(--text-secondary);
font-weight: 500;
white-space: nowrap;
}
/* Custom Checkbox */
.checkbox-label input {
position: absolute;
opacity: 0;
cursor: pointer;
height: 0;
width: 0;
}
.checkmark {
position: relative;
height: 20px;
width: 20px;
background-color: var(--toggle-bg);
border: 1px solid var(--toggle-border);
border-radius: 4px;
transition: all 0.3s ease;
}
.checkbox-label:hover .checkmark {
border-color: var(--toggle-hover-border);
}
.checkbox-label input:checked ~ .checkmark {
background-color: var(--primary-accent);
border-color: var(--primary-accent);
box-shadow: 0 0 10px var(--primary-accent);
}
.checkmark:after {
content: "";
position: absolute;
display: none;
}
.checkbox-label input:checked ~ .checkmark:after {
display: block;
}
.checkbox-label .checkmark:after {
left: 6px;
top: 2px;
width: 5px;
height: 10px;
border: solid #000;
border-width: 0 2px 2px 0;
transform: rotate(45deg);
}
/* Number Control */
.number-control {
display: flex;
align-items: stretch;
background: var(--toggle-bg);
border: 1px solid var(--toggle-border);
border-radius: 8px;
overflow: hidden;
gap: 0;
}
.control-btn {
background: none;
border: none;
color: var(--text-color);
font-size: 1.2rem;
width: 40px;
height: auto;
min-height: 40px;
cursor: pointer;
transition: background 0.2s;
padding: 0;
margin: 0;
display: flex;
align-items: center;
justify-content: center;
}
.control-btn:hover {
background: var(--button-hover-bg);
}
.number-input {
background: none;
border: none;
color: var(--text-color);
width: 100%;
flex: 1;
text-align: center;
font-size: 1rem;
font-weight: bold;
appearance: textfield;
-moz-appearance: textfield;
height: 100%;
border-radius: 0;
min-width: 60px;
}
.number-input:focus {
outline: none;
box-shadow: none;
background: rgba(0, 0, 0, 0.05);
}
.number-input::-webkit-outer-spin-button,
.number-input::-webkit-inner-spin-button {
-webkit-appearance: none;
margin: 0;
}
.result-area {
flex: 1;
display: flex;
flex-direction: column;
min-height: 200px;
}
.tool-textarea {
width: 100%;
height: 100%;
padding: 1rem;
border-radius: 12px;
border: 1px solid var(--glass-border);
background: var(--glass-bg);
color: var(--text-color);
font-family: monospace;
font-size: 0.9rem;
resize: none;
outline: none;
box-shadow: inset 0 2px 4px rgba(0, 0, 0, 0.1);
}
.generate-btn {
padding: 0.75rem 2rem;
font-size: 1rem;
font-weight: bold;
text-transform: uppercase;
letter-spacing: 1px;
}
.desktop-only {
display: block;
}
.mobile-only {
display: none;
}
@media (max-width: 768px) {
.options-grid {
flex-direction: column;
gap: 1rem;
}
.inputs-group {
flex-direction: column;
min-width: 100%;
}
.panel-header {
flex-direction: column;
align-items: stretch;
}
.generate-btn {
width: 100%;
}
.desktop-only {
display: none;
}
.mobile-only {
display: block !important;
}
} }
</style> </style>

View File

@@ -0,0 +1,542 @@
<script setup>
import { ref, watch, onUnmounted } from 'vue'
import { Copy, Trash2, ExternalLink, Power, Zap, X, Settings } from 'lucide-vue-next'
import { useExtension } from '../../composables/useExtension'
import { useLocalStorage } from '../../composables/useLocalStorage'
import ExtensionStatus from './common/ExtensionStatus.vue'
import UrlCleanerExceptionsModal from './UrlCleanerExceptionsModal.vue'
// Extension integration
const { isExtensionReady, isListening, lastClipboardText, startListening, stopListening, writeClipboard } = useExtension()
const inputUrl = ref('')
// Use local storage for history persistence
const cleanedHistory = useLocalStorage('url-cleaner-history', [])
const isWatchEnabled = useLocalStorage('url-cleaner-watch-enabled', false)
// Exceptions management
const showExceptionsModal = ref(false)
const defaultExceptions = [
{ id: 'yt', domainPattern: '*.youtube.com', keepParams: ['v', 't'], keepHash: false, keepAllParams: false, isEnabled: true, isDefault: true },
{ id: 'yt-short', domainPattern: 'youtu.be', keepParams: ['t'], keepHash: false, keepAllParams: false, isEnabled: true, isDefault: true }
]
const exceptions = useLocalStorage('url-cleaner-exceptions', defaultExceptions)
// Helper to match domain with glob pattern
const matchDomain = (pattern, domain) => {
// Escape regex chars except *
const regexString = '^' + pattern.replace(/[.+^${}()|[\]\\]/g, '\\$&').replace(/\*/g, '.*') + '$'
return new RegExp(regexString, 'i').test(domain)
}
// Watch for clipboard changes from extension
watch(lastClipboardText, (newText) => {
if (isWatchEnabled.value && newText) {
processUrl(newText, true)
}
})
// Sync watch state with extension listener
watch(isWatchEnabled, (enabled) => {
if (enabled) {
startListening()
} else {
stopListening()
}
}, { immediate: true })
// Re-enable listening when extension becomes ready
watch(isExtensionReady, (ready) => {
if (ready && isWatchEnabled.value) {
startListening()
}
})
// Toggle watch mode
const toggleWatch = () => {
isWatchEnabled.value = !isWatchEnabled.value
}
// Manual clean
const handleClean = () => {
if (inputUrl.value) {
processUrl(inputUrl.value, false)
inputUrl.value = ''
}
}
const processUrl = (text, autoClipboard = false) => {
try {
// Basic URL validation
if (!text.match(/^https?:\/\//i)) {
// Not a URL, ignore in watch mode
if (autoClipboard) return
}
const originalLength = text.length
let cleanedUrl = text
try {
const urlObj = new URL(text)
const hostname = urlObj.hostname
// Check for exceptions
const matchedRule = exceptions.value.find(rule =>
rule.isEnabled && matchDomain(rule.domainPattern, hostname)
)
if (matchedRule) {
if (!matchedRule.keepAllParams) {
// Exception logic: keep specific params
const params = new URLSearchParams(urlObj.search)
const keys = Array.from(params.keys())
for (const key of keys) {
if (!matchedRule.keepParams.includes(key)) {
params.delete(key)
}
}
urlObj.search = params.toString()
}
if (!matchedRule.keepHash) {
urlObj.hash = ''
}
} else {
// Default behavior: remove all query params and hash
if (urlObj.search || urlObj.hash) {
urlObj.search = ''
urlObj.hash = ''
}
}
cleanedUrl = urlObj.toString()
// Remove trailing slash if it wasn't there before? usually keep it standard
} catch (e) {
// Invalid URL format
if (!autoClipboard) {
// Show error or just return original
}
return
}
// If no change, ignore in watch mode to avoid loops
if (cleanedUrl === text && autoClipboard) {
return
}
const newLength = cleanedUrl.length
const savedChars = originalLength - newLength
const savedPercent = originalLength > 0 ? Math.round((savedChars / originalLength) * 100) : 0
// Add to history
const entry = {
id: Date.now(),
original: text,
cleaned: cleanedUrl,
savedPercent,
timestamp: new Date().toLocaleTimeString()
}
cleanedHistory.value.unshift(entry)
// Limit history
if (cleanedHistory.value.length > 50) {
cleanedHistory.value.pop()
}
// Auto-copy back to clipboard if in watch mode
if (autoClipboard && savedChars > 0) {
writeClipboard(cleanedUrl)
}
} catch (e) {
console.error('Error processing URL:', e)
}
}
const copyToClipboard = (text) => {
navigator.clipboard.writeText(text)
}
const removeEntry = (id) => {
cleanedHistory.value = cleanedHistory.value.filter(item => item.id !== id)
}
const clearHistory = () => {
cleanedHistory.value = []
}
onUnmounted(() => {
if (isListening.value) {
stopListening()
}
})
</script>
<template>
<div class="tool-container full-width">
<div class="tool-panel">
<div class="panel-header">
<h2 class="tool-title">URL Cleaner</h2>
<div class="header-actions">
<button class="icon-btn settings-btn" @click="showExceptionsModal = true" title="Cleaning Exceptions">
<Settings size="20" />
</button>
<ExtensionStatus :isReady="isExtensionReady" />
</div>
</div>
<div class="input-section">
<div class="input-wrapper">
<input
v-model="inputUrl"
type="text"
placeholder="Paste URL here to clean..."
class="url-input"
@keyup.enter="handleClean"
>
<button class="btn-neon" @click="handleClean">
Clean
</button>
</div>
<div class="watch-toggle">
<button
class="btn-neon toggle-btn"
:class="{ 'active': isWatchEnabled && isExtensionReady }"
@click="toggleWatch"
:disabled="!isExtensionReady"
:title="!isExtensionReady ? 'Extension required for auto-watch' : 'Automatically clean URLs from clipboard'"
>
<Power size="18" />
<span>Watch Clipboard</span>
<span v-if="isWatchEnabled && isExtensionReady" class="status-dot"></span>
</button>
</div>
</div>
<div class="history-section" v-if="cleanedHistory.length > 0">
<div class="history-header">
<h3>Cleaned URLs</h3>
<button class="icon-btn" @click="clearHistory" title="Clear History">
<Trash2 size="18" />
</button>
</div>
<div class="history-list">
<div v-for="item in cleanedHistory" :key="item.id" class="history-item">
<div class="item-info">
<div class="cleaned-url">{{ item.cleaned }}</div>
<div class="meta-info">
<span class="timestamp">{{ item.timestamp }}</span>
<span class="savings" v-if="item.savedPercent > 0">
<Zap size="12" /> -{{ item.savedPercent }}% junk removed
</span>
<span class="no-change" v-else>No junk found</span>
</div>
</div>
<div class="item-actions">
<button class="icon-btn" @click="copyToClipboard(item.cleaned)" title="Copy">
<Copy size="18" />
</button>
<a :href="item.cleaned" target="_blank" class="icon-btn" title="Open">
<ExternalLink size="18" />
</a>
<button class="icon-btn delete-btn" @click="removeEntry(item.id)" title="Remove">
<X size="18" />
</button>
</div>
</div>
</div>
</div>
<div class="empty-state" v-else>
<p>Paste a URL above or enable "Watch Clipboard" to automatically clean links.</p>
</div>
<UrlCleanerExceptionsModal
:isOpen="showExceptionsModal"
:exceptions="exceptions"
:defaultRules="defaultExceptions"
@update:exceptions="exceptions = $event"
@close="showExceptionsModal = false"
/>
</div>
</div>
</template>
<style scoped>
.tool-container.full-width {
max-width: 100%;
height: 100%;
display: flex;
flex-direction: column;
}
.tool-panel {
display: flex;
flex-direction: column;
height: 100%;
gap: 1.5rem;
position: relative;
}
.panel-header {
display: flex;
justify-content: center;
align-items: center;
position: relative;
width: 100%;
}
.input-section {
display: flex;
flex-direction: column;
gap: 1rem;
padding: 1.5rem;
background: var(--glass-bg);
border: 1px solid var(--glass-border);
border-radius: 12px;
}
@media (max-width: 640px) {
.input-section {
background: transparent;
border: none;
border-radius: 0;
padding: 0;
}
}
.input-wrapper {
display: flex;
gap: 1rem;
}
@media (max-width: 640px) {
.input-wrapper {
flex-direction: column;
}
.watch-toggle {
justify-content: center;
}
.toggle-btn {
width: 100%;
justify-content: center;
}
}
.url-input {
flex: 1;
padding: 0.8rem 1rem;
border-radius: 8px;
border: 1px solid var(--toggle-border);
background: var(--toggle-bg);
color: var(--text-color);
font-size: 1rem;
outline: none;
transition: all 0.2s;
}
.url-input:focus {
border-color: var(--primary-accent);
box-shadow: 0 0 0 2px rgba(var(--primary-accent-rgb), 0.2);
}
.watch-toggle {
display: flex;
justify-content: flex-end;
}
.toggle-btn {
display: flex;
align-items: center;
gap: 0.5rem;
position: relative;
}
.toggle-btn.active {
background: rgba(var(--primary-accent-rgb), 0.2);
border-color: var(--primary-accent);
color: var(--primary-accent);
}
.status-dot {
width: 8px;
height: 8px;
border-radius: 50%;
background-color: #4ade80; /* Green */
box-shadow: 0 0 8px #4ade80;
margin-left: 0.2rem;
animation: pulse 2s infinite;
}
@keyframes pulse {
0% { opacity: 1; }
50% { opacity: 0.5; }
100% { opacity: 1; }
}
.history-section {
flex: 1;
display: flex;
flex-direction: column;
overflow: hidden;
background: var(--glass-bg);
border: 1px solid var(--glass-border);
border-radius: 12px;
}
.history-header {
padding: 1rem;
border-bottom: 1px solid var(--glass-border);
display: flex;
justify-content: space-between;
align-items: center;
}
.history-header h3 {
margin: 0;
font-size: 1.1rem;
color: var(--text-strong);
}
.history-list {
flex: 1;
overflow-y: auto;
padding: 0.5rem;
}
.history-item {
display: flex;
justify-content: space-between;
align-items: center;
padding: 0.8rem;
border-bottom: 1px solid var(--glass-border);
transition: background 0.2s;
}
.history-item:last-child {
border-bottom: none;
}
.history-item:hover {
background: rgba(255, 255, 255, 0.05);
}
.item-info {
flex: 1;
overflow: hidden;
padding-right: 1rem;
}
.cleaned-url {
color: var(--primary-accent);
font-family: monospace;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
margin-bottom: 0.4rem;
}
.meta-info {
display: flex;
gap: 1rem;
font-size: 0.85rem;
color: var(--text-secondary);
}
.savings {
color: #4ade80;
display: flex;
align-items: center;
gap: 0.2rem;
}
:global(:root[data-theme="light"]) .savings {
color: #16a34a;
font-weight: 500;
}
.item-actions {
display: flex;
gap: 0.5rem;
}
.icon-btn {
background: none;
border: none;
color: var(--text-secondary);
cursor: pointer;
padding: 0.4rem;
border-radius: 4px;
transition: all 0.2s;
display: flex;
align-items: center;
justify-content: center;
}
.icon-btn:hover {
color: var(--text-color);
background: rgba(255, 255, 255, 0.1);
}
.delete-btn:hover {
background: none;
color: #ef4444;
}
:global(:root[data-theme="light"]) .delete-btn:hover {
background: none;
color: #dc2626;
}
.empty-state {
flex: 1;
display: flex;
justify-content: center;
align-items: center;
color: var(--text-secondary);
text-align: center;
padding: 2rem;
}
.header-actions {
display: flex;
align-items: center;
gap: 0.8rem;
position: absolute;
right: 0;
top: 50%;
transform: translateY(-50%);
}
.settings-btn {
background: rgba(255, 255, 255, 0.1);
width: 32px;
height: 32px;
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
padding: 0;
color: var(--text-secondary);
transition: all 0.3s ease;
}
.settings-btn:hover {
background: rgba(255, 255, 255, 0.2);
color: var(--primary-accent);
}
:global(:root[data-theme="light"]) .settings-btn {
background: rgba(0, 0, 0, 0.05);
}
:global(:root[data-theme="light"]) .settings-btn:hover {
background: rgba(0, 0, 0, 0.1);
}
</style>

View File

@@ -0,0 +1,566 @@
<script setup>
import { ref, computed } from 'vue'
import { X, Plus, Trash2, RotateCcw } from 'lucide-vue-next'
const props = defineProps({
isOpen: Boolean,
exceptions: Array,
defaultRules: Array
})
const emit = defineEmits(['close', 'update:exceptions'])
const newRule = ref({
domainPattern: '',
keepParams: '',
keepHash: false,
keepAllParams: false
})
const localExceptions = computed({
get: () => props.exceptions,
set: (val) => emit('update:exceptions', val)
})
const addRule = () => {
if (!newRule.value.domainPattern) return
const params = newRule.value.keepParams
.split(',')
.map(p => p.trim())
.filter(p => p)
const existingRuleIndex = localExceptions.value.findIndex(
r => r.domainPattern === newRule.value.domainPattern
)
if (existingRuleIndex >= 0) {
// Merge with existing rule
const existingRule = localExceptions.value[existingRuleIndex]
// Merge params (unique)
const mergedParams = [...new Set([...existingRule.keepParams, ...params])]
// Merge keepHash
const mergedKeepHash = existingRule.keepHash || newRule.value.keepHash
const mergedKeepAllParams = !!existingRule.keepAllParams || !!newRule.value.keepAllParams
const updatedRule = {
...existingRule,
keepParams: mergedParams,
keepHash: mergedKeepHash,
keepAllParams: mergedKeepAllParams,
isEnabled: true // Re-enable if disabled
}
const newExceptions = [...localExceptions.value]
newExceptions[existingRuleIndex] = updatedRule
localExceptions.value = newExceptions
} else {
// Create new rule
const rule = {
id: Date.now().toString(),
domainPattern: newRule.value.domainPattern,
keepParams: params,
keepHash: newRule.value.keepHash,
keepAllParams: newRule.value.keepAllParams,
isEnabled: true,
isDefault: false
}
localExceptions.value = [...localExceptions.value, rule]
}
// Reset form
newRule.value = {
domainPattern: '',
keepParams: '',
keepHash: false,
keepAllParams: false
}
}
const removeRule = (id) => {
localExceptions.value = localExceptions.value.filter(r => r.id !== id)
}
const toggleRule = (id) => {
localExceptions.value = localExceptions.value.map(r => {
if (r.id === id) {
return { ...r, isEnabled: !r.isEnabled }
}
return r
})
}
const removeParam = (ruleId, param) => {
localExceptions.value = localExceptions.value.map(r => {
if (r.id === ruleId) {
return {
...r,
keepParams: r.keepParams.filter(p => p !== param)
}
}
return r
})
}
const toggleKeepHash = (ruleId) => {
localExceptions.value = localExceptions.value.map(r => {
if (r.id === ruleId) {
return { ...r, keepHash: !r.keepHash }
}
return r
})
}
const toggleKeepAllParams = (ruleId) => {
localExceptions.value = localExceptions.value.map(r => {
if (r.id === ruleId) {
return { ...r, keepAllParams: !r.keepAllParams }
}
return r
})
}
const resetToDefault = (ruleId) => {
const defaultRule = (props.defaultRules || []).find(r => r.id === ruleId)
if (!defaultRule) return
localExceptions.value = localExceptions.value.map(r => {
if (r.id !== ruleId) return r
return {
...r,
domainPattern: defaultRule.domainPattern,
keepParams: Array.isArray(defaultRule.keepParams) ? [...defaultRule.keepParams] : [],
keepHash: !!defaultRule.keepHash,
keepAllParams: !!defaultRule.keepAllParams,
isEnabled: true,
isDefault: true
}
})
}
</script>
<template>
<Teleport to="body">
<div v-if="isOpen" class="modal-overlay" @click.self="$emit('close')">
<div class="modal-content glass-panel">
<div class="modal-header">
<h3>URL Cleaning Exceptions</h3>
<button class="close-btn" @click="$emit('close')">
<X size="24" />
</button>
</div>
<div class="modal-body">
<p class="description">
Define rules to keep specific parameters or anchors for certain domains.
Standard cleaning removes all parameters and anchors.
</p>
<div class="add-rule-form">
<h4>Add New Exception</h4>
<div class="form-row">
<input
v-model="newRule.domainPattern"
placeholder="Domain (e.g. *.youtube.com)"
class="input-field"
@keyup.enter="addRule"
>
<input
v-model="newRule.keepParams"
placeholder="Keep params (comma separated, e.g. v, id)"
class="input-field"
@keyup.enter="addRule"
>
</div>
<div class="form-row checkbox-row">
<div class="checkbox-group">
<label class="checkbox-label">
<input type="checkbox" v-model="newRule.keepHash">
Keep Anchor (#)
</label>
<label class="checkbox-label">
<input type="checkbox" v-model="newRule.keepAllParams">
Keep all params
</label>
</div>
<button class="btn-neon small" @click="addRule" :disabled="!newRule.domainPattern">
<Plus size="16" /> Add Rule
</button>
</div>
</div>
<div class="rules-list">
<h4>Active Rules</h4>
<div v-if="localExceptions.length === 0" class="empty-rules">
No exceptions defined.
</div>
<div v-for="rule in localExceptions" :key="rule.id" class="rule-item" :class="{ disabled: !rule.isEnabled }">
<div class="rule-info">
<div class="rule-domain">{{ rule.domainPattern }}</div>
<div class="rule-details">
<div class="params-list">
<template v-if="rule.keepAllParams">
<span class="detail-tag">
Keep all params
<button class="remove-param-btn" @click.stop="toggleKeepAllParams(rule.id)" title="Disable keep all params">
<X size="12" />
</button>
</span>
</template>
<template v-else>
<span v-for="param in rule.keepParams" :key="param" class="detail-tag">
{{ param }}
<button class="remove-param-btn" @click.stop="removeParam(rule.id, param)" title="Remove parameter">
<X size="12" />
</button>
</span>
</template>
<span v-if="rule.keepHash" class="detail-tag hash-tag">
Keep #
<button class="remove-param-btn" @click.stop="toggleKeepHash(rule.id)" title="Remove hash exception">
<X size="12" />
</button>
</span>
<span v-if="!rule.keepAllParams && rule.keepParams.length === 0 && !rule.keepHash" class="no-params">No params kept</span>
</div>
</div>
</div>
<div class="rule-actions">
<button
class="icon-btn"
@click="toggleRule(rule.id)"
:title="rule.isEnabled ? 'Disable rule' : 'Enable rule'"
>
<div class="toggle-switch" :class="{ active: rule.isEnabled }"></div>
</button>
<button
v-if="!rule.isDefault"
class="icon-btn delete-btn"
@click="removeRule(rule.id)"
title="Remove rule"
>
<Trash2 size="18" />
</button>
<button v-else class="btn-neon small default-reset" @click="resetToDefault(rule.id)" title="Restore default rule">
<RotateCcw size="16" /> Default
</button>
</div>
</div>
</div>
</div>
</div>
</div>
</Teleport>
</template>
<style scoped>
.modal-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.7);
display: flex;
justify-content: center;
align-items: center;
z-index: 1000;
backdrop-filter: blur(5px);
padding: 1rem;
}
.modal-content {
background: var(--glass-bg);
backdrop-filter: blur(16px);
-webkit-backdrop-filter: blur(16px);
border: 1px solid var(--glass-border);
border-radius: 16px;
padding: 0;
max-width: 800px;
width: 100%;
max-height: 85vh;
display: flex;
flex-direction: column;
box-shadow: var(--glass-shadow);
color: var(--text-color);
}
.modal-header {
padding: 1.5rem;
border-bottom: 1px solid var(--glass-border);
display: flex;
justify-content: space-between;
align-items: center;
}
@media (max-width: 640px) {
.modal-header {
padding: 0.8rem 1.2rem;
}
}
.modal-header h3 {
margin: 0;
font-size: 1.4rem;
background: var(--title-gradient);
-webkit-background-clip: text;
background-clip: text;
color: transparent;
}
.close-btn {
background: none;
border: none;
color: var(--text-secondary);
cursor: pointer;
padding: 0.5rem;
}
.close-btn:hover {
color: var(--text-color);
}
.modal-body {
padding: 1.5rem;
overflow-y: auto;
}
@media (max-width: 640px) {
.modal-body {
padding: 1rem 1.2rem;
}
}
.description {
color: var(--text-secondary);
margin-bottom: 1.5rem;
font-size: 0.95rem;
line-height: 1.5;
}
.add-rule-form {
background: rgba(255, 255, 255, 0.05);
padding: 1rem;
border-radius: 8px;
margin-bottom: 2rem;
border: 1px solid var(--glass-border);
}
.add-rule-form h4, .rules-list h4 {
margin-top: 0;
margin-bottom: 1rem;
font-size: 1rem;
color: var(--text-strong);
}
.form-row {
display: flex;
gap: 0.8rem;
margin-bottom: 0.8rem;
}
@media (max-width: 640px) {
.form-row {
flex-direction: column;
}
.checkbox-row {
flex-direction: column;
align-items: stretch;
gap: 1rem;
}
.checkbox-group {
flex-direction: column;
align-items: flex-start;
gap: 0.8rem;
}
.btn-neon {
width: 100%;
justify-content: center;
}
}
.checkbox-row {
align-items: center;
justify-content: space-between;
margin-bottom: 0;
}
.checkbox-group {
display: flex;
align-items: center;
gap: 1.2rem;
}
.input-field {
flex: 1;
padding: 0.6rem;
border-radius: 6px;
border: 1px solid var(--toggle-border);
background: var(--toggle-bg);
color: var(--text-color);
outline: none;
}
.input-field:focus {
border-color: var(--primary-accent);
}
.checkbox-label {
display: flex;
align-items: center;
gap: 0.5rem;
color: var(--text-secondary);
font-size: 0.9rem;
cursor: pointer;
}
.btn-neon.small {
padding: 0.4rem 1rem;
font-size: 0.9rem;
}
.rules-list {
display: flex;
flex-direction: column;
gap: 0.8rem;
}
.rule-item {
display: flex;
justify-content: space-between;
align-items: center;
padding: 0.8rem;
border: 1px solid var(--glass-border);
border-radius: 8px;
background: rgba(255, 255, 255, 0.03);
transition: opacity 0.3s;
}
.rule-item.disabled {
opacity: 0.6;
}
.rule-info {
flex: 1;
}
.rule-domain {
font-weight: 600;
color: var(--primary-accent);
margin-bottom: 0.3rem;
}
.rule-details {
width: 100%;
}
.params-list {
display: flex;
gap: 0.5rem;
flex-wrap: wrap;
}
.detail-tag {
font-size: 0.8rem;
background: rgba(255, 255, 255, 0.1);
padding: 0.2rem 0.5rem;
border-radius: 4px;
color: var(--text-secondary);
display: flex;
align-items: center;
gap: 0.4rem;
}
.remove-param-btn {
background: none;
border: none;
padding: 0;
display: flex;
align-items: center;
color: var(--text-muted);
cursor: pointer;
transition: color 0.2s;
}
.remove-param-btn:hover {
color: #ef4444;
}
.no-params {
font-size: 0.8rem;
color: var(--text-muted);
font-style: italic;
}
.rule-actions {
display: flex;
align-items: center;
gap: 0.8rem;
}
.toggle-switch {
width: 36px;
height: 20px;
background: rgba(255, 255, 255, 0.2);
border-radius: 10px;
position: relative;
transition: background 0.3s;
}
.toggle-switch::after {
content: '';
position: absolute;
left: 2px;
top: 2px;
width: 16px;
height: 16px;
background: white;
border-radius: 50%;
transition: transform 0.3s;
}
.toggle-switch.active {
background: #4ade80;
}
.toggle-switch.active::after {
transform: translateX(16px);
}
.default-reset {
white-space: nowrap;
}
.empty-rules {
text-align: center;
color: var(--text-muted);
padding: 1rem;
}
.icon-btn {
background: none;
border: none;
cursor: pointer;
padding: 0;
color: var(--text-secondary);
display: flex;
align-items: center;
}
.delete-btn:hover {
color: #ef4444;
}
:global(:root[data-theme="light"]) .delete-btn:hover {
color: #dc2626;
}
</style>

View File

@@ -0,0 +1,192 @@
<script>
export default {
inheritAttrs: false
}
</script>
<script setup>
import { ref } from 'vue'
import { Plug, Plus, X } from 'lucide-vue-next'
const props = defineProps({
isReady: Boolean
})
const showModal = ref(false)
</script>
<template>
<div class="extension-status" v-bind="$attrs" :class="{ 'is-ready': isReady }" @click="showModal = true" :title="isReady ? 'Extension Connected' : 'Extension Not Connected'">
<Plug v-if="isReady" size="18" />
<Plus v-else size="18" />
</div>
<Teleport to="body">
<div v-if="showModal" class="modal-overlay" @click.self="showModal = false">
<div class="modal-content glass-panel">
<button class="close-btn" @click="showModal = false">
<X size="24" />
</button>
<div v-if="!isReady">
<h3>Enhance Your Experience</h3>
<p>
Install our browser extension to enable <strong>background clipboard features</strong>!
</p>
<p class="description">
Without the extension, this tool can only access clipboard when the tab is active.
With the extension, you can capture and process content even when you're working in other apps.
</p>
<div class="modal-actions">
<a href="#" class="btn-neon" @click.prevent>Extension Coming Soon</a>
</div>
</div>
<div v-else>
<h3>Extension Connected!</h3>
<p>
You have successfully enabled <strong>background clipboard integration</strong>.
</p>
<p class="description">
The extension is active and ready to process your clipboard in the background.
</p>
<div class="modal-actions">
<button class="btn-neon" @click="showModal = false">Got it!</button>
</div>
</div>
</div>
</div>
</Teleport>
</template>
<style scoped>
.extension-status {
display: flex;
align-items: center;
justify-content: center;
width: 32px;
height: 32px;
border-radius: 50%;
background: rgba(255, 255, 255, 0.1);
color: var(--text-secondary);
cursor: pointer;
transition: all 0.3s ease;
}
:global(:root[data-theme="light"]) .extension-status {
background: rgba(0, 0, 0, 0.05);
color: var(--text-secondary);
}
.extension-status:hover {
background: rgba(255, 255, 255, 0.2);
color: var(--primary-accent);
}
:global(:root[data-theme="light"]) .extension-status:hover {
background: rgba(0, 0, 0, 0.1);
}
.modal-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.7);
display: flex;
justify-content: center;
align-items: center;
z-index: 1000;
backdrop-filter: blur(5px);
padding: 1rem;
}
.modal-content {
background: var(--glass-bg);
backdrop-filter: blur(16px);
-webkit-backdrop-filter: blur(16px);
border: 1px solid var(--glass-border);
border-radius: 16px;
padding: 2.5rem; /* Większy padding */
max-width: 500px;
width: 90%;
position: relative;
box-shadow: var(--glass-shadow);
text-align: center;
color: var(--text-color); /* Wymuś kolor tekstu */
}
.close-btn {
position: absolute;
top: 1rem;
right: 1rem;
background: none;
border: none;
color: var(--text-secondary);
cursor: pointer;
padding: 0.5rem;
transition: color 0.2s;
}
.close-btn:hover {
color: var(--text-color);
}
h3 {
margin-top: 0;
margin-bottom: 1.5rem;
font-size: 1.8rem; /* Większy nagłówek */
background: var(--title-gradient);
-webkit-background-clip: text;
background-clip: text;
color: transparent;
font-weight: 700;
}
p {
font-size: 1.1rem;
line-height: 1.6;
margin-bottom: 1rem;
color: var(--text-color); /* Wymuś kolor */
}
.description {
color: var(--text-secondary);
margin-bottom: 2rem;
font-size: 1rem;
}
.modal-actions {
display: flex;
justify-content: center;
gap: 1rem;
margin-top: 2rem;
}
.modal-actions .btn-neon {
font-size: 1.1rem;
padding: 0.8rem 2.5rem;
min-width: 140px;
}
strong {
color: var(--primary-accent);
font-weight: 600;
}
.extension-status.is-ready {
color: #4ade80;
background: rgba(74, 222, 128, 0.1);
}
.extension-status.is-ready:hover {
background: rgba(74, 222, 128, 0.2);
color: #4ade80;
}
:global(:root[data-theme="light"]) .extension-status.is-ready {
background: rgba(74, 222, 128, 0.15);
color: #16a34a; /* Darker green for light mode */
}
</style>

View File

@@ -0,0 +1,79 @@
import { ref, onMounted, onUnmounted } from 'vue'
export function useExtension() {
const isExtensionReady = ref(false)
const isListening = ref(false)
const lastClipboardText = ref('')
let extensionCheckInterval = null
let lastPongTime = Date.now()
const PING_INTERVAL = 200
const TIMEOUT_THRESHOLD = 1000
const handleMessage = (event) => {
if (event.source !== window) return
if (event.data.type === 'TOOLS_APP_EXTENSION_READY' || event.data.type === 'TOOLS_APP_PONG') {
isExtensionReady.value = true
lastPongTime = Date.now()
}
if (event.data.type === 'TOOLS_APP_CLIPBOARD_UPDATE') {
const text = event.data.content
// Only update if listening
if (isListening.value && text) {
lastClipboardText.value = text
}
}
}
const startWatchdog = () => {
extensionCheckInterval = setInterval(() => {
window.postMessage({ type: 'TOOLS_APP_PING' }, '*')
if (Date.now() - lastPongTime > TIMEOUT_THRESHOLD) {
isExtensionReady.value = false
}
}, PING_INTERVAL)
}
const startListening = () => {
window.postMessage({ type: 'TOOLS_APP_START_SNIFFING' }, '*')
isListening.value = true
}
const stopListening = () => {
window.postMessage({ type: 'TOOLS_APP_STOP_SNIFFING' }, '*')
isListening.value = false
}
const writeClipboard = (text) => {
// Send to extension (background -> offscreen -> clipboard)
window.postMessage({ type: 'TOOLS_APP_CLIPBOARD_WRITE', content: text }, '*')
// Update local state to avoid echo loop if we are listening
lastClipboardText.value = text
}
onMounted(() => {
window.addEventListener('message', handleMessage)
// Initial check
window.postMessage({ type: 'TOOLS_APP_INIT' }, '*')
startWatchdog()
})
onUnmounted(() => {
if (isListening.value) {
stopListening()
}
if (extensionCheckInterval) clearInterval(extensionCheckInterval)
window.removeEventListener('message', handleMessage)
})
return {
isExtensionReady,
isListening,
lastClipboardText,
startListening,
stopListening,
writeClipboard
}
}

View File

@@ -0,0 +1,47 @@
import { onMounted, onUnmounted, ref, nextTick } from 'vue'
export function useFillHeight(elementRef, marginBottom = 20) {
const height = ref('auto')
const updateHeight = () => {
if (!elementRef.value) return
const rect = elementRef.value.getBoundingClientRect()
const windowHeight = window.innerHeight
// Calculate available space: window height - element top position - margin bottom
// We also need to account for the footer height if it's fixed or layout related
// The user mentioned "margin bottom from footer".
// If footer is in the flow, we might just want to fill the parent container?
// But user asked for JS resizing.
// Let's assume we want to fill down to (windowHeight - marginBottom).
// This assumes the element should stretch to the bottom of the viewport.
const availableHeight = windowHeight - rect.top - marginBottom
// Ensure minimum height
if (availableHeight > 100) {
height.value = `${availableHeight}px`
}
}
onMounted(() => {
// Initial update after render
nextTick(() => {
updateHeight()
window.addEventListener('resize', updateHeight)
})
// Also update after a short delay to ensure layout is settled (e.g. sidebar transitions)
setTimeout(updateHeight, 300)
})
onUnmounted(() => {
window.removeEventListener('resize', updateHeight)
})
return {
height,
updateHeight
}
}

View File

@@ -0,0 +1,18 @@
import { ref, watch } from 'vue'
export function useLocalStorage(key, defaultValue) {
// Initialize state from local storage or default value
const storedValue = localStorage.getItem(key)
const data = ref(storedValue ? JSON.parse(storedValue) : defaultValue)
// Watch for changes and update local storage
watch(data, (newValue) => {
if (newValue === null || newValue === undefined) {
localStorage.removeItem(key)
} else {
localStorage.setItem(key, JSON.stringify(newValue))
}
}, { deep: true })
return data
}

View File

@@ -1,6 +1,9 @@
import { createRouter, createWebHistory } from 'vue-router' import { createRouter, createWebHistory } from 'vue-router'
import Main from '../components/Main.vue' import Main from '../components/Main.vue'
import Passwords from '../components/tools/Passwords.vue' import Passwords from '../components/tools/Passwords.vue'
import ClipboardSniffer from '../components/tools/ClipboardSniffer.vue'
import UrlCleaner from '../components/tools/UrlCleaner.vue'
import PrivacyPolicy from '../views/PrivacyPolicy.vue'
const routes = [ const routes = [
{ {
@@ -12,6 +15,21 @@ const routes = [
path: '/passwords', path: '/passwords',
name: 'Passwords', name: 'Passwords',
component: Passwords component: Passwords
},
{
path: '/clipboard-sniffer',
name: 'ClipboardSniffer',
component: ClipboardSniffer
},
{
path: '/url-cleaner',
name: 'UrlCleaner',
component: UrlCleaner
},
{
path: '/extension-privacy-policy',
name: 'PrivacyPolicy',
component: PrivacyPolicy
} }
] ]

View File

@@ -1,3 +1,9 @@
/* Box sizing reset */
*, *::before, *::after {
box-sizing: border-box;
}
@import 'tailwindcss';
:root { :root {
font-family: system-ui, Avenir, Helvetica, Arial, sans-serif; font-family: system-ui, Avenir, Helvetica, Arial, sans-serif;
line-height: 1.5; line-height: 1.5;
@@ -49,7 +55,7 @@
} }
:root[data-theme="light"] { :root[data-theme="light"] {
--bg-gradient: radial-gradient(circle at center, #ffffff 0%, #cccccc 100%); --bg-gradient: radial-gradient(circle at center, #ffffff 0%, #e5e7eb 100%);
--glass-bg: rgba(255, 255, 255, 0.75); --glass-bg: rgba(255, 255, 255, 0.75);
--glass-border: rgba(15, 23, 42, 0.12); --glass-border: rgba(15, 23, 42, 0.12);
--glass-shadow: 0 8px 32px 0 rgba(15, 23, 42, 0.12); --glass-shadow: 0 8px 32px 0 rgba(15, 23, 42, 0.12);
@@ -85,15 +91,145 @@ body {
display: flex; display: flex;
min-width: 320px; min-width: 320px;
min-height: 100vh; min-height: 100vh;
overflow-x: hidden;
user-select: none;
-webkit-user-select: none;
}
.selectable {
user-select: text;
-webkit-user-select: text;
cursor: text;
} }
#app { #app {
width: 100%; width: 100%;
height: 100vh; min-height: 100vh;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
} }
@media (min-width: 768px) {
body {
height: 100vh;
overflow: hidden;
}
#app {
height: 100vh;
overflow: hidden;
}
}
/* --- Shared styles for all tools (moved from tools.css) --- */
.tool-container {
display: flex;
justify-content: center;
width: 100%;
max-width: 800px;
margin: 0 auto;
height: 100%;
padding: 1rem;
}
.tool-panel {
width: 100%;
padding: 2rem;
border-radius: 16px;
display: flex;
flex-direction: column;
gap: 1.5rem;
max-height: 100%;
overflow-y: auto;
background: var(--glass-bg);
backdrop-filter: blur(12px);
-webkit-backdrop-filter: blur(12px);
border: 1px solid var(--glass-border);
box-shadow: var(--glass-shadow);
}
/* Custom scrollbar for tool panel */
.tool-panel::-webkit-scrollbar {
width: 8px;
}
.tool-panel::-webkit-scrollbar-track {
background: rgba(0, 0, 0, 0.1);
border-radius: 4px;
}
.tool-panel::-webkit-scrollbar-thumb {
background: rgba(255, 255, 255, 0.2);
border-radius: 4px;
}
.tool-panel::-webkit-scrollbar-thumb:hover {
background: rgba(255, 255, 255, 0.3);
}
.tool-title {
margin: 0;
text-align: center;
font-size: 1.5rem;
font-weight: 600;
background: var(--title-gradient);
-webkit-background-clip: text;
background-clip: text;
color: transparent;
-webkit-text-fill-color: transparent;
filter: drop-shadow(0 0 10px var(--title-glow));
}
:root[data-theme="light"] .tool-title {
/* background: none !important;
-webkit-background-clip: unset !important;
background-clip: unset !important;
color: #000000 !important;
-webkit-text-fill-color: #000000 !important;
filter: none !important; */
}
.tool-textarea {
width: 100%;
height: 100%;
padding: 1rem;
background-color: rgba(0, 0, 0, 0.2) !important;
border: 1px solid var(--toggle-border);
border-radius: 12px;
color: #ffffff !important; /* Explicit white color for dark mode */
font-family: monospace;
font-size: 1rem;
line-height: 1.6;
resize: none;
transition: all 0.3s ease;
box-sizing: border-box;
}
:root[data-theme="light"] .tool-textarea {
color: #000000 !important;
background-color: rgba(255, 255, 255, 0.5) !important;
}
.tool-textarea:focus {
outline: none;
border-color: #00f2fe !important; /* Force cyan accent */
box-shadow: 0 0 0 1px #00f2fe !important;
}
.result-area {
display: flex;
flex-direction: column;
flex: 1;
min-height: 200px;
}
.result-area label {
margin-bottom: 0.5rem;
font-weight: 600;
color: var(--text-secondary);
}
/* Common UI Element Styles */ /* Common UI Element Styles */
.glass-panel { .glass-panel {
background: var(--glass-bg); background: var(--glass-bg);
@@ -103,6 +239,16 @@ body {
box-shadow: var(--glass-shadow); box-shadow: var(--glass-shadow);
} }
@media (max-width: 640px) {
.glass-panel:not(.modal-content) {
background: transparent;
backdrop-filter: none;
-webkit-backdrop-filter: none;
border: none;
box-shadow: none;
}
}
.btn-neon { .btn-neon {
background: var(--button-bg); background: var(--button-bg);
border: 1px solid var(--button-border); border: 1px solid var(--button-border);
@@ -120,6 +266,12 @@ body {
outline: none; /* Remove focus outline */ outline: none; /* Remove focus outline */
} }
/* Global button styles */
button {
user-select: none;
-webkit-user-select: none;
}
/* Remove focus outline for all buttons */ /* Remove focus outline for all buttons */
button:focus { button:focus {
outline: none; outline: none;
@@ -137,6 +289,22 @@ button:focus {
box-shadow: var(--button-active-shadow); box-shadow: var(--button-active-shadow);
} }
.btn-neon.danger {
background: rgba(239, 68, 68, 0.2);
border-color: rgba(239, 68, 68, 0.6);
color: #fee2e2;
}
.btn-neon.danger:hover {
background: rgba(239, 68, 68, 0.3);
border-color: rgba(239, 68, 68, 0.85);
box-shadow: 0 8px 20px rgba(239, 68, 68, 0.2);
}
:root[data-theme="light"] .btn-neon.danger {
color: #7f1d1d;
}
.icon-only { .icon-only {
padding: 8px; padding: 8px;
border-radius: 50%; border-radius: 50%;

107
src/styles/tools.css Normal file
View File

@@ -0,0 +1,107 @@
/* Shared styles for all tools */
.tool-container {
display: flex;
justify-content: center;
width: 100%;
max-width: 800px;
margin: 0 auto;
height: 100%;
padding: 1rem;
}
.tool-panel {
width: 100%;
padding: 2rem;
border-radius: 16px;
display: flex;
flex-direction: column;
gap: 1.5rem;
max-height: 100%;
overflow-y: auto;
background: var(--glass-bg);
backdrop-filter: blur(12px);
-webkit-backdrop-filter: blur(12px);
border: 1px solid var(--glass-border);
box-shadow: var(--glass-shadow);
}
/* Custom scrollbar for tool panel */
.tool-panel::-webkit-scrollbar {
width: 8px;
}
.tool-panel::-webkit-scrollbar-track {
background: rgba(0, 0, 0, 0.1);
border-radius: 4px;
}
.tool-panel::-webkit-scrollbar-thumb {
background: rgba(255, 255, 255, 0.2);
border-radius: 4px;
}
.tool-panel::-webkit-scrollbar-thumb:hover {
background: rgba(255, 255, 255, 0.3);
}
.tool-title {
margin: 0;
text-align: center;
font-size: 1.5rem;
font-weight: 600;
background: var(--title-gradient);
-webkit-background-clip: text;
background-clip: text;
color: transparent;
-webkit-text-fill-color: transparent;
filter: drop-shadow(0 0 10px var(--title-glow));
}
:root[data-theme="light"] .tool-title {
background: none;
-webkit-background-clip: unset;
background-clip: unset;
color: #000000;
-webkit-text-fill-color: #000000;
filter: none;
}
.tool-textarea {
width: 100%;
height: 100%;
padding: 1rem;
background-color: rgba(0, 0, 0, 0.2);
border: 1px solid var(--toggle-border);
border-radius: 12px;
color: #ffffff; /* Explicit white color for dark mode */
font-family: monospace;
font-size: 1rem;
line-height: 1.6;
resize: none;
transition: all 0.3s ease;
box-sizing: border-box;
}
:root[data-theme="light"] .tool-textarea {
color: #000000;
}
.tool-textarea:focus {
outline: none;
border-color: #00f2fe; /* Force cyan accent */
box-shadow: 0 0 0 1px #00f2fe;
}
.result-area {
display: flex;
flex-direction: column;
flex: 1;
min-height: 200px;
}
.result-area label {
margin-bottom: 0.5rem;
font-weight: 600;
color: var(--text-secondary);
}

209
src/views/PrivacyPolicy.vue Normal file
View File

@@ -0,0 +1,209 @@
<script setup>
import { ArrowLeft } from 'lucide-vue-next'
</script>
<template>
<div class="tool-container">
<div class="tool-panel privacy-panel">
<header class="privacy-header">
<router-link to="/" class="back-link">
<ArrowLeft size="20" />
Back to Tools
</router-link>
<h1>Privacy Policy</h1>
<p class="last-updated">Last Updated: February 27, 2026</p>
</header>
<div class="privacy-body">
<section>
<h2>1. Introduction</h2>
<p>
Welcome to Tools App ("we," "our," or "us"). We are committed to protecting your privacy.
This Privacy Policy explains how our Chrome Extension ("Tools App Extension") handles your data.
</p>
</section>
<section>
<h2>2. Data Collection and Usage</h2>
<p>
The Tools App Extension is designed with privacy as a priority.
<strong>We do not collect, store, or transmit any of your personal data to external servers.</strong>
</p>
<h3>Clipboard Data</h3>
<p>
The extension requires the <code>clipboardRead</code> permission to function.
It reads text from your clipboard <strong>only</strong> when you explicitly enable the "Clipboard Sniffer" tool in the Tools App web interface.
</p>
<ul>
<li>Clipboard data is processed locally within your browser.</li>
<li>Data is sent directly from the extension to the open Tools App tab via a secure local communication channel.</li>
<li>Once you close the tab or stop the tool, the extension stops monitoring the clipboard immediately.</li>
<li>We do not have access to your clipboard history, and it is never uploaded to any cloud storage or third-party service.</li>
</ul>
</section>
<section>
<h2>3. Permissions</h2>
<p>The extension requests the following permissions for specific functional purposes:</p>
<ul>
<li><strong>clipboardRead:</strong> To detect copied text when the Sniffer tool is active.</li>
<li><strong>scripting:</strong> To communicate with the Tools App web page.</li>
<li><strong>storage:</strong> To save local user preferences (e.g., sound settings).</li>
<li><strong>alarms:</strong> To maintain the background process active during monitoring sessions.</li>
</ul>
</section>
<section>
<h2>4. Third-Party Services</h2>
<p>
Our extension operates independently and does not use any third-party analytics, tracking scripts, or advertising networks.
</p>
</section>
<section>
<h2>5. Changes to This Policy</h2>
<p>
We may update our Privacy Policy from time to time. We will notify you of any changes by posting the new Privacy Policy on this page.
</p>
</section>
<section>
<h2>6. Contact Us</h2>
<p>
If you have any questions about this Privacy Policy, please contact us via the repository or support channels provided in the Chrome Web Store listing.
</p>
</section>
</div>
</div>
</div>
</template>
<style scoped>
.privacy-panel {
max-width: 900px; /* Slightly wider for reading */
margin: 0 auto;
}
.privacy-body {
overflow-y: auto;
padding-right: 0.5rem;
}
/* Custom scrollbar for privacy body */
.privacy-body::-webkit-scrollbar {
width: 8px;
}
.privacy-body::-webkit-scrollbar-track {
background: rgba(0, 0, 0, 0.1);
border-radius: 4px;
}
.privacy-body::-webkit-scrollbar-thumb {
background: rgba(255, 255, 255, 0.2);
border-radius: 4px;
}
.privacy-header {
margin-bottom: 2rem;
border-bottom: 1px solid var(--glass-border);
padding-bottom: 1.5rem;
flex-shrink: 0;
}
.back-link {
display: inline-flex;
align-items: center;
gap: 0.5rem;
color: var(--text-secondary);
text-decoration: none;
font-weight: 500;
margin-bottom: 1.5rem;
transition: color 0.2s;
}
.back-link:hover {
color: var(--primary-accent);
}
h1 {
font-size: 2.5rem;
margin: 0 0 0.5rem 0;
background: var(--title-gradient);
-webkit-background-clip: text;
background-clip: text;
color: transparent;
width: fit-content;
}
.last-updated {
color: var(--text-secondary);
font-size: 0.9rem;
}
section {
margin-bottom: 2.5rem;
}
h2 {
font-size: 1.5rem;
margin-bottom: 1rem;
color: var(--text-strong);
}
h3 {
font-size: 1.1rem;
margin: 1.5rem 0 0.5rem 0;
color: var(--text-strong);
}
p {
line-height: 1.7;
color: var(--text-color);
margin-bottom: 1rem;
}
ul {
padding-left: 1.5rem;
margin-bottom: 1rem;
}
li {
margin-bottom: 0.5rem;
line-height: 1.6;
color: var(--text-color);
}
strong {
color: var(--text-strong);
font-weight: 600;
}
code {
background: rgba(255, 255, 255, 0.1);
padding: 0.2rem 0.4rem;
border-radius: 4px;
font-family: monospace;
font-size: 0.9em;
color: var(--text-strong);
}
:global(html[data-theme="light"]) code {
background: rgba(0, 0, 0, 0.1);
}
@media (max-width: 768px) {
.privacy-container {
padding: 1rem;
}
.privacy-content {
padding: 1.5rem;
}
h1 {
font-size: 2rem;
}
}
</style>

View File

@@ -1,10 +1,46 @@
import { defineConfig } from 'vite' import { defineConfig } from 'vite'
import vue from '@vitejs/plugin-vue' import vue from '@vitejs/plugin-vue'
import { VitePWA } from 'vite-plugin-pwa'
import packageJson from './package.json' import packageJson from './package.json'
// https://vite.dev/config/ // https://vite.dev/config/
export default defineConfig({ export default defineConfig({
plugins: [vue()], plugins: [
vue(),
VitePWA({
registerType: 'autoUpdate',
includeAssets: ['favicon.svg'],
manifest: {
name: 'Tools App',
short_name: 'Tools',
description: 'A collection of useful tools',
theme_color: '#4facfe',
background_color: '#242424',
display: 'standalone',
orientation: 'portrait',
icons: [
{
src: 'favicon.svg',
sizes: 'any',
type: 'image/svg+xml',
purpose: 'any maskable'
}
]
},
workbox: {
globPatterns: ['**/*.{js,css,html,ico,png,svg,json,vue,txt,woff2}'],
cleanupOutdatedCaches: true,
clientsClaim: true,
skipWaiting: true
},
devOptions: {
enabled: true,
/* when using generateSW the PWA plugin will switch to classic */
type: 'module',
navigateFallback: 'index.html',
}
})
],
define: { define: {
'__APP_VERSION__': JSON.stringify(packageJson.version) '__APP_VERSION__': JSON.stringify(packageJson.version)
} }