51 Commits
v0.4.4 ... main

Author SHA1 Message Date
e40762873c 0.6.19
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-03-03 14:30:19 +00:00
011db26ec4 fix: normalize font-weight for Length/Count labels in Passwords; refactor QR scanner composables; style fixes 2026-03-03 14:29:58 +00:00
6f95dce55a 0.6.18
All checks were successful
Deploy to Production / deploy (push) Successful in 14s
2026-03-03 13:40:20 +00:00
c5293ca9fe style: standardize UI across tools, optimize QR layout, and configure Husky/ESLint 2026-03-03 13:39:56 +00:00
b1cc8ab5a1 style: unify UI elements and improve light theme visibility 2026-03-03 12:33:06 +00:00
dff8a2a0ab refactor: extract logic from QrScanner and UrlCleaner to composables 2026-03-03 11:19:57 +00:00
ee387d9637 feat: add extension packing script and update build script
All checks were successful
Deploy to Production / deploy (push) Successful in 14s
2026-03-03 11:03:43 +00:00
f2203e896e chore: bump version to 0.6.16
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-03-03 10:03:44 +00:00
cb8d3d01ec fix: update extension link to Chrome Web Store 2026-03-03 10:03:41 +00:00
5b31171964 0.6.15
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-03-03 00:02:34 +00:00
e98761a18e feat: make header title a link to home page 2026-03-03 00:02:17 +00:00
bc8168e724 0.6.14
All checks were successful
Deploy to Production / deploy (push) Successful in 14s
2026-03-02 23:59:04 +00:00
dcde3b0799 feat: add ESC key support to close all modals and fullscreen modes 2026-03-02 23:58:50 +00:00
60fc774586 0.6.13
All checks were successful
Deploy to Production / deploy (push) Successful in 14s
2026-03-02 23:37:12 +00:00
d65c0d0357 docs: replace problematic emoji in README.md with safer alternative 2026-03-02 23:36:57 +00:00
02736ecc70 0.6.12
All checks were successful
Deploy to Production / deploy (push) Successful in 14s
2026-03-02 23:32:21 +00:00
7b1d19ca7a docs: fix corrupted emoji in README.md 2026-03-02 23:32:12 +00:00
4846d0e61c 0.6.11
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-03-02 23:26:45 +00:00
3155e12b84 feat: display README.md on home page using marked 2026-03-02 23:26:36 +00:00
b8bbe84aa9 docs: add descriptions for all new tools (QR Gen, QR Scanner, URL Cleaner) to README 2026-03-02 23:24:03 +00:00
74984caf9e 0.6.10
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-28 18:07:22 +00:00
c8b799b078 fix: mirror background canvas for front camera in fullscreen mode 2026-02-28 18:07:10 +00:00
f3a4c1af05 0.6.9
All checks were successful
Deploy to Production / deploy (push) Successful in 14s
2026-02-28 18:05:14 +00:00
616f615d7c fix: improve front camera detection on macOS by checking video track label 2026-02-28 18:04:28 +00:00
4d572b55ca 0.6.8
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-28 17:51:40 +00:00
9822cab93e feat: restore QR code detection overlay in custom QrScanner 2026-02-28 17:50:29 +00:00
9409bd3e21 0.6.7
All checks were successful
Deploy to Production / deploy (push) Successful in 14s
2026-02-28 17:46:21 +00:00
346ded460a chore: remove wasm file from repo and add to gitignore 2026-02-28 17:45:21 +00:00
170539a62f 0.6.6
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-28 17:41:06 +00:00
cfc1785863 chore: cleanup unused code in QrScanner and remove vue-qrcode-reader dependency 2026-02-28 17:40:42 +00:00
712b1238a5 0.6.5
All checks were successful
Deploy to Production / deploy (push) Successful in 14s
2026-02-28 17:34:35 +00:00
3732d365dd refactor: replace vue-qrcode-reader with custom native QR scanner using local BarcodeDetector polyfill 2026-02-28 17:31:04 +00:00
34fd8bb2b3 0.6.4
All checks were successful
Deploy to Production / deploy (push) Successful in 14s
2026-02-28 17:21:46 +00:00
90dc663393 fix: use local zxing-wasm file to resolve CSP issues in QrScanner 2026-02-28 17:21:29 +00:00
1c4bdeff0e 0.6.3
All checks were successful
Deploy to Production / deploy (push) Successful in 15s
2026-02-28 17:10:17 +00:00
35c5ff4c51 fix: ensure useFillHeight recalculates when element appears (v-if) 2026-02-28 17:09:42 +00:00
c8f9dfb37e 0.6.2
All checks were successful
Deploy to Production / deploy (push) Successful in 15s
2026-02-28 04:19:41 +00:00
70d7c8873e refactor: centralize UI config and inject CSS variables dynamically 2026-02-28 04:18:53 +00:00
d5d3d37804 refactor: simplify useLocalStorage (remove APP_PREFIX) and finalize QR Code tool 2026-02-28 04:05:07 +00:00
2a1897f68d feat: enhance QR generator with responsive layout, custom focus styles, and download options 2026-02-28 03:50:47 +00:00
b2e8f23d60 0.6.1
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-27 19:43:31 +00:00
d2ea9e3fc7 chore: UI spacing tweaks, desktop scrolling fix, QR title alignment 2026-02-27 19:43:12 +00:00
1765742574 0.6.0
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-27 19:03:09 +00:00
5b1a50f148 chore: prepare release; reintroduce front camera CSS mirror, stabilize QR Scanner 2026-02-27 19:02:49 +00:00
613604f3c4 feat(qr-scanner): remove frame, add shape detection overlay, improve fullscreen desktop layout 2026-02-27 17:52:36 +00:00
a699b432d7 0.5.0
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-27 17:00:41 +00:00
839a98a658 feat: add QR Scanner tool with history, export, persistence and fullscreen support 2026-02-27 17:00:11 +00:00
fd23860bcf 0.4.6
All checks were successful
Deploy to Production / deploy (push) Successful in 12s
2026-02-27 09:16:51 +00:00
74c0251535 feat: improve url cleaner modal ui (light mode) and enable rule editing 2026-02-27 09:16:37 +00:00
06b2815dd9 0.4.5
All checks were successful
Deploy to Production / deploy (push) Successful in 13s
2026-02-27 08:51:31 +00:00
1346de684c feat: url cleaner multiline support and ui tweaks 2026-02-27 08:51:05 +00:00
32 changed files with 3490 additions and 666 deletions

7
.gitignore vendored
View File

@@ -11,6 +11,7 @@ node_modules
dist dist
dist-ssr dist-ssr
*.local *.local
public/wasm
# Editor directories and files # Editor directories and files
.vscode/* .vscode/*
@@ -26,3 +27,9 @@ dev-dist
extension-release.zip extension-release.zip
*.zip *.zip
tools-app-extension-*.zip tools-app-extension-*.zip
tools-app-extension-*.crx
# Extension keys and builds
*.pem
*.crx
scripts/*.pub

4
.husky/pre-commit Executable file
View File

@@ -0,0 +1,4 @@
#!/usr/bin/env sh
. "$(dirname -- "$0")/_/husky.sh"
npm run lint

View File

@@ -20,6 +20,26 @@ Monitor and capture your clipboard history in real-time.
- Clears history on demand. - Clears history on demand.
- Privacy-focused: Data is processed locally and never sent to any server. - Privacy-focused: Data is processed locally and never sent to any server.
### 🔗 URL Cleaner
Clean tracking parameters and clutter from URLs.
- **Privacy:** Removes known tracking parameters (UTM, fbclid, gclid, etc.).
- **Bulk Processing:** Clean list of URLs at once.
- **Customizable:** Manage exceptions to keep specific parameters.
- **Smart:** "Direct Clean" mode for quick single-URL cleaning.
### ⬛ QR Generator
Create custom QR codes instantly.
- **Customizable:** Adjustable error correction level and size.
- **Download:** Save as PNG (raster) or SVG (vector) for high-quality printing.
- **Persistent Settings:** Remembers your preferences.
### 📷 QR Scanner
Scan QR codes directly from your camera or device.
- **Privacy First:** Works entirely in the browser using local `BarcodeDetector` API. No images are sent to any server.
- **Multi-Camera:** Support for front and back cameras with easy switching.
- **History:** Keeps a log of scanned codes with options to copy or download as JSON.
- **Responsive:** Fullscreen mode for immersive scanning experience.
--- ---
## Chrome Extension 🧩 ## Chrome Extension 🧩

49
eslint.config.js Normal file
View File

@@ -0,0 +1,49 @@
import pluginVue from 'eslint-plugin-vue'
import js from '@eslint/js'
import globals from 'globals'
export default [
// Base JS recommended rules (all off for now to allow pre-commit)
{
rules: Object.keys(js.configs.recommended.rules).reduce((acc, rule) => {
acc[rule] = 'off';
return acc;
}, {}),
},
// Vue essential rules
...pluginVue.configs['flat/essential'],
{
// Apply to all JS and Vue files
files: ['**/*.js', '**/*.vue'],
languageOptions: {
ecmaVersion: 'latest',
sourceType: 'module',
globals: {
...globals.browser,
...globals.node,
crypto: 'readonly',
BarcodeDetector: 'readonly',
chrome: 'readonly',
__APP_VERSION__: 'readonly',
VITE_APP_VERSION: 'readonly',
},
},
rules: {
'vue/multi-word-component-names': 'off',
'no-unused-vars': 'off',
'no-undef': 'off',
'no-debugger': 'off',
}
},
{
// Global ignores
ignores: [
'dist/**',
'dev-dist/**',
'node_modules/**',
'public/**',
'scripts/pack_crx.js',
'src/app.config.mjs'
]
}
]

1233
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -1,20 +1,32 @@
{ {
"name": "tools-app", "name": "tools-app",
"private": true, "private": true,
"version": "0.4.4", "version": "0.6.19",
"type": "module", "type": "module",
"scripts": { "scripts": {
"dev": "vite", "dev": "vite",
"build": "vite build", "build": "vite build",
"preview": "vite preview" "preview": "vite preview",
"lint": "eslint .",
"pack-extension": "node scripts/pack_crx.js",
"postinstall": "mkdir -p public/wasm && cp node_modules/zxing-wasm/dist/reader/zxing_reader.wasm public/wasm/",
"prepare": "husky"
}, },
"dependencies": { "dependencies": {
"barcode-detector": "^3.1.0",
"lucide-vue-next": "^0.575.0", "lucide-vue-next": "^0.575.0",
"marked": "^17.0.3",
"qrcode": "^1.5.4",
"vue": "^3.5.25", "vue": "^3.5.25",
"vue-router": "^5.0.3" "vue-router": "^5.0.3"
}, },
"devDependencies": { "devDependencies": {
"@eslint/js": "^10.0.1",
"@vitejs/plugin-vue": "^6.0.2", "@vitejs/plugin-vue": "^6.0.2",
"eslint": "^10.0.2",
"eslint-plugin-vue": "^10.8.0",
"globals": "^17.4.0",
"husky": "^9.1.7",
"vite": "^7.3.1", "vite": "^7.3.1",
"vite-plugin-pwa": "^1.2.0" "vite-plugin-pwa": "^1.2.0"
} }

View File

@@ -6,9 +6,14 @@ import zipfile
# Configuration # Configuration
SOURCE_DIR = "extension" SOURCE_DIR = "extension"
BUILD_DIR = "dist-extension" BUILD_DIR = "dist-extension"
OUTPUT_ZIP = "extension-release.zip"
MANIFEST_FILE = "manifest.json" MANIFEST_FILE = "manifest.json"
# Read version to create dynamic zip name
with open(os.path.join(SOURCE_DIR, MANIFEST_FILE), "r") as f:
source_manifest = json.load(f)
version = source_manifest.get("version", "unknown")
OUTPUT_ZIP = f"tools-app-extension-v{version}.zip"
# Remove build directory if exists # Remove build directory if exists
if os.path.exists(BUILD_DIR): if os.path.exists(BUILD_DIR):
shutil.rmtree(BUILD_DIR) shutil.rmtree(BUILD_DIR)

110
scripts/pack_crx.js Normal file
View File

@@ -0,0 +1,110 @@
import { execSync } from 'child_process';
import fs from 'fs';
import path from 'path';
function findKey() {
// 1. Check if key provided via CLI
if (process.argv[3]) return process.argv[3];
// 2. Check local project directory (gitignored)
const localKey = path.join(process.cwd(), 'scripts', 'key.pem');
if (fs.existsSync(localKey)) return localKey;
// 3. Check common SSH key locations in ~/.ssh
const sshDir = path.join(process.env.HOME, '.ssh');
const commonKeys = ['id_rsa', 'id_ecdsa', 'id_ed25519'];
for (const keyName of commonKeys) {
const fullPath = path.join(sshDir, keyName);
if (fs.existsSync(fullPath)) return fullPath;
}
return null;
}
const KEY_PATH = findKey();
const INPUT_SOURCE = process.argv[2] || path.join(process.cwd(), 'extension');
const TEMP_DIR = path.join(process.cwd(), 'temp_extension_build');
function pack() {
console.log('📦 Packing extension to CRX...');
if (!KEY_PATH) {
console.error('❌ Error: No private key found.');
console.log('Tried local scripts/key.pem and common ~/.ssh/id_* keys.');
console.log('You can provide a path manually: npm run pack-extension <source> <path/to/key>');
process.exit(1);
}
let extensionDir = INPUT_SOURCE;
let isTemp = false;
try {
// If input is a zip file, unzip it first
if (INPUT_SOURCE.endsWith('.zip')) {
console.log('🤐 Unzipping extension...');
if (fs.existsSync(TEMP_DIR)) fs.rmSync(TEMP_DIR, { recursive: true, force: true });
fs.mkdirSync(TEMP_DIR);
execSync(`unzip -o "${INPUT_SOURCE}" -d "${TEMP_DIR}"`, { stdio: 'pipe' });
extensionDir = TEMP_DIR;
isTemp = true;
}
// Resolve actual extension directory (handle subdirs in zip)
if (!fs.existsSync(path.join(extensionDir, 'manifest.json'))) {
const subdirs = fs.readdirSync(extensionDir).filter(f => fs.statSync(path.join(extensionDir, f)).isDirectory());
if (subdirs.length === 1 && fs.existsSync(path.join(extensionDir, subdirs[0], 'manifest.json'))) {
extensionDir = path.join(extensionDir, subdirs[0]);
console.log(`📂 Found manifest in subdirectory: ${extensionDir}`);
} else {
console.error(`❌ Error: manifest.json not found in ${extensionDir}`);
process.exit(1);
}
}
// Determine output filename
let outputName;
if (INPUT_SOURCE.endsWith('.zip')) {
outputName = path.basename(INPUT_SOURCE).replace('.zip', '.crx');
} else {
const manifest = JSON.parse(fs.readFileSync(path.join(extensionDir, 'manifest.json'), 'utf8'));
outputName = `tools-app-extension-v${manifest.version}.crx`;
}
const outputFull = path.join(process.cwd(), outputName);
// Get version for logging
const manifest = JSON.parse(fs.readFileSync(path.join(extensionDir, 'manifest.json'), 'utf8'));
const version = manifest.version;
console.log(`🔑 Using key: ${KEY_PATH}`);
console.log(`📂 Source: ${extensionDir} (v${version})`);
console.log(`🚀 Running crx3 via npx...`);
// Command: npx -y crx3 --key <key> --crx <output> <dir>
execSync(`npx -y crx3 --key "${KEY_PATH}" --crx "${outputFull}" "${extensionDir}"`, {
stdio: 'inherit'
});
// Cleanup any file that crx3 might have created automatically based on temp dir name
const unintendedFile = extensionDir + '.crx';
if (fs.existsSync(unintendedFile) && unintendedFile !== outputFull) {
fs.unlinkSync(unintendedFile);
}
console.log(`\n✅ Success! Extension packed to: ${outputFull}`);
} catch (error) {
console.error('\n❌ Failed to pack extension.');
if (error.message.includes('algorithm')) {
console.error('⚠️ Note: Chrome CRX format requires RSA or ECDSA (P-256) keys.');
} else {
console.error('Error details:', error.message);
}
process.exit(1);
} finally {
if (isTemp && fs.existsSync(TEMP_DIR)) {
fs.rmSync(TEMP_DIR, { recursive: true, force: true });
}
}
}
pack();

View File

@@ -6,6 +6,7 @@ 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 InstallPrompt from './components/InstallPrompt.vue'
import ReloadPrompt from './components/ReloadPrompt.vue' import ReloadPrompt from './components/ReloadPrompt.vue'
import { UI_CONFIG } from './config/ui'
const isSidebarOpen = ref(window.innerWidth >= 768) const isSidebarOpen = ref(window.innerWidth >= 768)
const router = useRouter() const router = useRouter()
@@ -31,6 +32,11 @@ const handleResize = () => {
} }
onMounted(() => { onMounted(() => {
// Set global CSS variables from config
document.documentElement.style.setProperty('--header-height', `${UI_CONFIG.headerHeight}px`)
document.documentElement.style.setProperty('--footer-height', `${UI_CONFIG.footerHeight}px`)
document.documentElement.style.setProperty('--page-padding', `${UI_CONFIG.pagePadding}px`)
window.addEventListener('resize', handleResize) window.addEventListener('resize', handleResize)
}) })
@@ -67,29 +73,31 @@ onUnmounted(() => {
.main-content { .main-content {
flex: 1; flex: 1;
padding: 2rem; padding: 1rem;
width: 100%; width: 100%;
max-width: 100%; max-width: 100%;
/* Space for fixed footer on mobile + extra margin (match top padding 2rem + footer height ~40px) */ display: flex;
padding-bottom: calc(2rem + 40px + env(safe-area-inset-bottom)); flex-direction: column;
/* Space for fixed footer on mobile + extra margin */
padding-bottom: calc(1rem + var(--footer-height) + env(safe-area-inset-bottom));
} }
@media (max-width: 640px) { @media (max-width: 640px) {
.main-content { .main-content {
padding: 1rem; padding: 0.5rem;
padding-bottom: calc(1rem + 40px + env(safe-area-inset-bottom)); padding-bottom: calc(0.5rem + var(--footer-height) + env(safe-area-inset-bottom));
} }
} }
@media (min-width: 768px) { @media (min-width: 768px) {
.app-body { .app-body {
overflow: hidden; overflow: visible;
} }
.main-content { .main-content {
overflow-y: auto; overflow: visible;
height: 100%; flex: 1;
padding-bottom: 2rem; padding-bottom: 1rem;
} }
} }

View File

@@ -14,7 +14,8 @@ const version = __APP_VERSION__;
<style scoped> <style scoped>
.app-footer { .app-footer {
width: 100%; width: 100%;
padding: 0.5rem; height: var(--footer-height);
padding: 0 0.5rem;
/* Background handled by glass-panel */ /* Background handled by glass-panel */
border-left: none; border-left: none;
border-right: none; border-right: none;
@@ -24,12 +25,9 @@ const version = __APP_VERSION__;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
z-index: 10; z-index: 10;
/* Remove fixed height to allow content to dictate size */
/* height: 30px; */
position: fixed; position: fixed;
bottom: 0; bottom: 0;
left: 0; left: 0;
padding-bottom: max(0.5rem, env(safe-area-inset-bottom));
} }
@media (min-width: 768px) { @media (min-width: 768px) {

View File

@@ -44,7 +44,7 @@ onMounted(() => {
<line x1="3" y1="18" x2="21" y2="18"></line> <line x1="3" y1="18" x2="21" y2="18"></line>
</svg> </svg>
</button> </button>
<h1 class="app-title">Tools App</h1> <router-link to="/" class="app-title">Tools App</router-link>
</div> </div>
<button <button
@@ -63,9 +63,12 @@ onMounted(() => {
<style scoped> <style scoped>
.app-header { .app-header {
/* Remove hardcoded colors and use theme variables */ /* Remove hardcoded colors and use theme variables */
background: var(--glass-bg); background: var(--header-bg);
color: var(--text-color); color: var(--text-color);
padding: 1rem; height: var(--header-height);
padding: 0 1rem;
display: flex;
align-items: center;
/* box-shadow handled by glass-panel class */ /* box-shadow handled by glass-panel class */
position: sticky; position: sticky;
top: 0; top: 0;
@@ -77,7 +80,7 @@ onMounted(() => {
} }
.header-content { .header-content {
max-width: 100%; width: 100%;
padding: 0; padding: 0;
margin: 0 auto; margin: 0 auto;
display: flex; display: flex;
@@ -112,10 +115,11 @@ onMounted(() => {
.app-title { .app-title {
margin: 0; margin: 0;
font-size: 1.5rem; font-size: 1.5rem;
font-weight: 600;
background: var(--title-gradient); background: var(--title-gradient);
-webkit-background-clip: text; -webkit-background-clip: text;
-webkit-text-fill-color: transparent;
background-clip: text; background-clip: text;
color: transparent;
text-decoration: none;
font-weight: bold;
} }
</style> </style>

View File

@@ -51,12 +51,20 @@ const dismissPrompt = () => {
deferredPrompt = null deferredPrompt = null
} }
const handleKeydown = (e) => {
if (e.key === 'Escape' && showInstallPrompt.value) {
dismissPrompt()
}
}
onMounted(() => { onMounted(() => {
window.addEventListener('beforeinstallprompt', handleBeforeInstallPrompt) window.addEventListener('beforeinstallprompt', handleBeforeInstallPrompt)
window.addEventListener('keydown', handleKeydown)
}) })
onBeforeUnmount(() => { onBeforeUnmount(() => {
window.removeEventListener('beforeinstallprompt', handleBeforeInstallPrompt) window.removeEventListener('beforeinstallprompt', handleBeforeInstallPrompt)
window.removeEventListener('keydown', handleKeydown)
}) })
</script> </script>

View File

@@ -1,11 +1,99 @@
<script setup> <script setup>
import { ref } from 'vue' import { marked } from 'marked'
import { computed } from 'vue'
import readmeContent from '../../README.md?raw'
const htmlContent = computed(() => {
return marked(readmeContent)
})
</script> </script>
<template> <template>
main <div class="readme-container glass-panel">
<div class="markdown-body" v-html="htmlContent"></div>
</div>
</template> </template>
<style scoped> <style scoped>
.readme-container {
padding: 2rem;
max-width: 800px;
margin: 0 auto;
}
.markdown-body :deep(h1) {
font-size: 2rem;
margin-bottom: 1rem;
color: var(--primary-accent);
}
.markdown-body :deep(h2) {
font-size: 1.5rem;
margin-top: 2rem;
margin-bottom: 1rem;
border-bottom: 1px solid var(--glass-border);
padding-bottom: 0.5rem;
color: var(--text-strong);
}
.markdown-body :deep(h3) {
font-size: 1.25rem;
margin-top: 1.5rem;
margin-bottom: 0.75rem;
color: var(--text-strong);
}
.markdown-body :deep(p) {
margin-bottom: 1rem;
line-height: 1.6;
color: var(--text-color);
}
.markdown-body :deep(ul) {
margin-bottom: 1rem;
padding-left: 1.5rem;
}
.markdown-body :deep(li) {
margin-bottom: 0.5rem;
color: var(--text-color);
}
.markdown-body :deep(a) {
color: var(--primary-accent);
text-decoration: none;
}
.markdown-body :deep(a:hover) {
text-decoration: underline;
}
.markdown-body :deep(code) {
background: rgba(255, 255, 255, 0.1);
padding: 0.2em 0.4em;
border-radius: 4px;
font-family: monospace;
font-size: 0.9em;
}
.markdown-body :deep(pre) {
background: rgba(0, 0, 0, 0.3);
padding: 1rem;
border-radius: 8px;
overflow-x: auto;
margin-bottom: 1rem;
border: 1px solid var(--glass-border);
}
.markdown-body :deep(pre code) {
background: none;
padding: 0;
color: #e6e6e6;
}
.markdown-body :deep(hr) {
border: none;
border-top: 1px solid var(--glass-border);
margin: 2rem 0;
}
</style> </style>

View File

@@ -13,6 +13,8 @@ defineProps({
<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="/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> <router-link to="/url-cleaner" class="nav-item" v-ripple>URL Cleaner</router-link>
<router-link to="/qr-scanner" class="nav-item" v-ripple>QR Scanner</router-link>
<router-link to="/qr-code" class="nav-item" v-ripple>QR Code</router-link>
</nav> </nav>
</aside> </aside>
</template> </template>
@@ -87,6 +89,11 @@ defineProps({
color: var(--text-strong); color: var(--text-strong);
} }
:global(:root[data-theme="light"]) .nav-item:hover {
background-color: rgba(15, 23, 42, 0.1);
color: var(--text-strong);
}
.nav-item.router-link-active { .nav-item.router-link-active {
color: var(--primary-accent); color: var(--primary-accent);
background-color: var(--toggle-bg); background-color: var(--toggle-bg);

View File

@@ -49,9 +49,9 @@ const clearText = () => {
<template> <template>
<div class="tool-container full-width"> <div class="tool-container full-width">
<div class="tool-panel"> <div class="tool-panel">
<div class="tool-header"> <div class="panel-header">
<h2 class="tool-title">Clipboard Sniffer</h2> <h2 class="tool-title">Clipboard Sniffer</h2>
<div class="extension-indicator-wrapper"> <div class="header-actions">
<ExtensionStatus :isReady="isExtensionReady" /> <ExtensionStatus :isReady="isExtensionReady" />
</div> </div>
</div> </div>
@@ -99,19 +99,13 @@ const clearText = () => {
</template> </template>
<style scoped> <style scoped>
.tool-header {
display: flex;
justify-content: center;
align-items: center;
position: relative;
width: 100%;
}
.controls { .controls {
display: flex; display: flex;
justify-content: center; justify-content: center;
gap: 1rem; gap: 1rem;
flex-wrap: wrap; flex-wrap: wrap;
flex-shrink: 0;
} }
.tool-container.full-width { .tool-container.full-width {
@@ -126,6 +120,9 @@ const clearText = () => {
flex-direction: column; flex-direction: column;
height: 100%; height: 100%;
gap: 1.5rem; gap: 1.5rem;
/* Override shared tool-panel scroll for this tool */
max-height: none;
overflow: hidden;
} }
.tool-textarea { .tool-textarea {
@@ -147,11 +144,9 @@ const clearText = () => {
border-color: var(--primary-accent); border-color: var(--primary-accent);
} }
.extension-indicator-wrapper {
position: absolute; .result-area {
right: 0; flex: 1;
top: 50%;
transform: translateY(-50%);
display: flex; display: flex;
} }
</style> </style>

View File

@@ -74,9 +74,9 @@ const generatePasswords = () => {
<div class="tool-container full-width"> <div class="tool-container full-width">
<div class="tool-panel"> <div class="tool-panel">
<div class="panel-header"> <div class="panel-header">
<h2 class="tool-title">Bulk Passwords Generator</h2> <h2 class="tool-title">Passwords Generator</h2>
<div class="action-area desktop-only"> <div class="header-actions">
<button class="btn-neon generate-btn" @click="generatePasswords" v-ripple> <button class="btn-neon generate-btn desktop-only" @click="generatePasswords" v-ripple>
Generate Generate
</button> </button>
</div> </div>
@@ -142,7 +142,6 @@ const generatePasswords = () => {
class="tool-textarea" class="tool-textarea"
v-model="result" v-model="result"
placeholder="Generated passwords will appear here..." placeholder="Generated passwords will appear here..."
readonly
></textarea> ></textarea>
</div> </div>
</div> </div>
@@ -150,32 +149,12 @@ const generatePasswords = () => {
</template> </template>
<style scoped> <style scoped>
.tool-container.full-width { .tool-container {
max-width: 100%;
height: 100%; height: 100%;
display: flex; display: flex;
flex-direction: column; 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 { .options-grid {
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
@@ -209,112 +188,10 @@ const generatePasswords = () => {
min-width: 140px; min-width: 140px;
} }
.checkbox-label { .input-wrapper 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); color: var(--text-color);
font-size: 1.2rem; font-weight: 400;
width: 40px; margin-bottom: 0.2rem;
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 { .number-input:focus {
@@ -323,12 +200,6 @@ const generatePasswords = () => {
background: rgba(0, 0, 0, 0.05); 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 { .result-area {
flex: 1; flex: 1;
display: flex; display: flex;
@@ -336,25 +207,7 @@ const generatePasswords = () => {
min-height: 200px; 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 { .generate-btn {
padding: 0.75rem 2rem;
font-size: 1rem;
font-weight: bold;
text-transform: uppercase; text-transform: uppercase;
letter-spacing: 1px; letter-spacing: 1px;
} }

View File

@@ -0,0 +1,291 @@
<script setup>
import { ref, watch, onMounted } from 'vue'
import { Download } from 'lucide-vue-next'
import QRCode from 'qrcode'
import { useFillHeight } from '../../composables/useFillHeight'
import { useLocalStorage } from '../../composables/useLocalStorage'
const text = useLocalStorage('text', '', 'qr-code')
const ecc = useLocalStorage('ecc', 'M', 'qr-code')
const size = useLocalStorage('size', 300, 'qr-code')
const format = useLocalStorage('format', 'png', 'qr-code')
const svgContent = ref('')
const previewRef = ref(null)
const { height: previewHeight } = useFillHeight(previewRef, 40) // 40px extra margin
const generateQR = async () => {
if (!text.value) {
svgContent.value = ''
return
}
try {
// Generate SVG for preview (always sharp)
svgContent.value = await QRCode.toString(text.value, {
type: 'svg',
errorCorrectionLevel: ecc.value,
margin: 1,
// No fixed width, allow scaling via CSS
})
} catch (err) {
console.error('QR Generation failed', err)
svgContent.value = ''
}
}
// Debounce generation slightly to avoid lag on typing
let timeout
watch([text, ecc], () => { // size is not relevant for preview
clearTimeout(timeout)
timeout = setTimeout(generateQR, 300)
})
onMounted(() => {
if (text.value) generateQR()
})
const downloadFile = async () => {
if (!text.value) return
const filename = `qr-code-${Date.now()}.${format.value}`
if (format.value === 'svg') {
// For SVG download, we might want to inject the size if user specifically requested it,
// but usually raw SVG is better.
// If we want to support the "Size" dropdown for SVG download, we can regenerate with specific width.
const svgWithSize = await QRCode.toString(text.value, {
type: 'svg',
errorCorrectionLevel: ecc.value,
margin: 1,
width: size.value
})
const blob = new Blob([svgWithSize], { type: 'image/svg+xml' })
triggerDownload(blob, filename)
} else {
// For raster formats, render to canvas first
try {
const canvas = document.createElement('canvas')
await QRCode.toCanvas(canvas, text.value, {
errorCorrectionLevel: ecc.value,
margin: 1,
width: size.value
})
const mime = `image/${format.value}`
canvas.toBlob((blob) => {
if (blob) triggerDownload(blob, filename)
}, mime)
} catch (err) {
console.error('Download failed', err)
}
}
}
const triggerDownload = (blob, filename) => {
const url = URL.createObjectURL(blob)
const a = document.createElement('a')
a.href = url
a.download = filename
document.body.appendChild(a)
a.click()
document.body.removeChild(a)
URL.revokeObjectURL(url)
}
</script>
<template>
<div class="tool-container full-width">
<div class="tool-panel">
<div class="panel-header">
<h2 class="tool-title">QR Generator</h2>
<div class="header-actions"></div>
</div>
<div class="input-section">
<textarea
v-model="text"
class="tool-textarea"
placeholder="Enter text to generate QR code..."
rows="3"
></textarea>
</div>
<div class="controls-section">
<div class="control-group">
<label>Error Correction</label>
<select v-model="ecc" class="select-input">
<option value="L">Low (7%)</option>
<option value="M">Medium (15%)</option>
<option value="Q">Quartile (25%)</option>
<option value="H">High (30%)</option>
</select>
</div>
</div>
<div class="preview-section" v-if="text" ref="previewRef" :style="{ height: previewHeight }">
<div class="qr-container">
<div class="qr-frame" v-html="svgContent"></div>
</div>
<div class="download-settings">
<div class="control-group">
<label>Size (px)</label>
<div class="number-control size-control">
<button class="control-btn" @click="size = Math.max(10, size - 100)" title="-100">-100</button>
<button class="control-btn" @click="size = Math.max(10, size - 10)" title="-10">-10</button>
<input type="number" v-model.number="size" class="number-input" />
<button class="control-btn" @click="size += 10" title="+10">+10</button>
<button class="control-btn" @click="size += 100" title="+100">+100</button>
</div>
</div>
<div class="control-group">
<label>Format</label>
<select v-model="format" class="select-input format-select">
<option value="png">PNG</option>
<option value="jpeg">JPG</option>
<option value="webp">WebP</option>
<option value="svg">SVG</option>
</select>
</div>
<button class="action-btn download-btn" @click="downloadFile">
<Download size="18" />
Download
</button>
</div>
</div>
</div>
</div>
</template>
<style scoped>
.tool-container.full-width {
height: 100%;
display: flex;
flex-direction: column;
}
.tool-panel {
padding: 1rem;
display: flex;
flex-direction: column;
gap: 1.5rem;
height: 100%;
overflow: hidden; /* Prevent scrolling, force fit */
}
.tool-title {
margin: 0;
font-size: 1.5rem;
font-weight: 600;
color: var(--primary-accent);
}
.input-section {
display: flex;
flex-direction: column;
flex-shrink: 0;
}
.tool-textarea {
min-height: 80px;
}
.controls-section {
display: flex;
gap: 1.5rem;
flex-wrap: wrap;
justify-content: center;
flex-shrink: 0;
}
.control-group {
display: flex;
flex-direction: column;
gap: 0.5rem;
}
.control-group label {
font-size: 0.85rem;
color: var(--text-secondary);
}
.preview-section {
display: flex;
flex-direction: column;
background: rgba(0, 0, 0, 0.05);
border-radius: 12px;
padding: 0.75rem 1rem 1.25rem;
overflow: hidden;
min-height: 0;
flex: 1;
gap: 1.5rem;
}
.qr-container {
flex: 1;
display: flex;
align-items: center;
justify-content: center;
width: 100%;
min-height: 0;
container-type: size;
}
:root[data-theme="light"] .preview-section {
background: rgba(255, 255, 255, 0.3);
}
.qr-frame {
width: min(100cqw, 100cqh);
height: min(100cqw, 100cqh);
background: white;
padding: 1rem;
border-radius: 8px;
box-shadow: 0 4px 12px rgba(0, 0, 0, 0.2);
display: flex;
justify-content: center;
align-items: center;
overflow: hidden;
}
.download-settings {
display: flex;
gap: 1.5rem;
align-items: flex-end;
flex-wrap: wrap;
justify-content: center;
width: 100%;
}
.size-control {
min-width: 280px;
}
.format-select {
min-width: 100px !important;
}
.download-btn {
height: 40px;
padding: 0 1.5rem;
background: var(--primary-accent);
color: #000;
border: none;
border-radius: 8px;
font-weight: 600;
display: flex;
align-items: center;
gap: 0.5rem;
cursor: pointer;
transition: all 0.2s;
}
.download-btn:hover {
opacity: 0.9;
transform: translateY(-1px);
}
</style>

View File

@@ -0,0 +1,626 @@
<script setup>
import { ref, onMounted, onUnmounted, watch, computed } from 'vue'
import { SwitchCamera, Trash2, Copy, Download, X } from 'lucide-vue-next'
import { useCamera } from '../../composables/useCamera'
import { useQrDetection } from '../../composables/useQrDetection'
const scannedCodes = ref([])
const isFullscreen = ref(false)
const videoAspect = ref(1)
const wrapperRef = ref(null)
const bgCanvas = ref(null)
let bgRafId = null
const videoRef = ref(null)
const overlayCanvas = ref(null)
const {
stream,
facingMode,
hasMultipleCameras,
isMirrored,
error: cameraError,
checkCameras,
startCamera,
stopCamera,
switchCamera: baseSwitchCamera
} = useCamera(videoRef)
const {
error: detectionError,
isDetecting,
startDetection,
stopDetection
} = useQrDetection(videoRef, overlayCanvas)
const error = computed(() => cameraError.value || detectionError.value)
// Background Loop
const updateVideoAspect = () => {
if (videoRef.value && videoRef.value.videoWidth && videoRef.value.videoHeight) {
videoAspect.value = videoRef.value.videoWidth / videoRef.value.videoHeight
}
}
const startBackgroundLoop = () => {
const draw = () => {
const videoEl = videoRef.value
const canvas = bgCanvas.value
if (!videoEl || !canvas || videoEl.paused || videoEl.ended) {
bgRafId = requestAnimationFrame(draw)
return
}
const vw = videoEl.videoWidth || 0
const vh = videoEl.videoHeight || 0
if (!vw || !vh) {
bgRafId = requestAnimationFrame(draw)
return
}
const cw = Math.floor(window.innerWidth)
const ch = Math.floor(window.innerHeight * 0.5)
if (canvas.width !== cw || canvas.height !== ch) {
canvas.width = cw
canvas.height = ch
}
const ctx = canvas.getContext('2d')
if (ctx) {
const scale = cw / vw
const srcH = ch / scale
const sx = 0
const sy = Math.max(0, (vh - srcH) / 2)
ctx.clearRect(0, 0, cw, ch)
ctx.drawImage(videoEl, sx, sy, vw, srcH, 0, 0, cw, ch)
}
bgRafId = requestAnimationFrame(draw)
}
if (bgRafId) cancelAnimationFrame(bgRafId)
bgRafId = requestAnimationFrame(draw)
}
const stopBackgroundLoop = () => {
if (bgRafId) {
cancelAnimationFrame(bgRafId)
bgRafId = null
}
}
// Full screen styles
const desktopFullscreenStyle = computed(() => {
if (!isFullscreen.value) return {}
const isDesktop = window.innerWidth >= 768
if (!isDesktop) return {}
const halfHeight = Math.floor(window.innerHeight * 0.5)
const widthPx = Math.min(window.innerWidth, Math.floor(halfHeight * videoAspect.value))
return { height: `${halfHeight}px`, width: `${widthPx}px`, margin: '0 auto' }
})
const processCodes = (codes) => {
codes.forEach(code => {
const value = code.rawValue
if (value && !scannedCodes.value.some(c => c.value === value)) {
scannedCodes.value.unshift({
id: Date.now() + Math.random(),
value,
format: code.format,
timestamp: new Date().toLocaleTimeString()
})
}
})
}
const onDetect = (detectedCodes) => {
if (isFullscreen.value) {
processCodes(detectedCodes)
return
}
const videoEl = document.querySelector('.camera-wrapper video')
if (!videoEl || !videoEl.videoWidth || !videoEl.videoHeight) {
processCodes(detectedCodes)
return
}
const { videoWidth, videoHeight } = videoEl
const isLandscape = videoWidth > videoHeight
let visibleX, visibleY, visibleW, visibleH
if (isLandscape) {
visibleH = videoHeight
visibleW = videoHeight
visibleX = (videoWidth - videoHeight) / 2
visibleY = 0
} else {
visibleW = videoWidth
visibleH = videoWidth
visibleX = 0
visibleY = (videoHeight - videoWidth) / 2
}
const validCodes = detectedCodes.filter(code => {
if (!code.boundingBox) return true
const { x, y, width, height } = code.boundingBox
const centerX = x + width / 2
const centerY = y + height / 2
return (
centerX >= visibleX &&
centerX <= visibleX + visibleW &&
centerY >= visibleY &&
centerY <= visibleY + visibleH
)
})
processCodes(validCodes)
}
const loadHistory = () => {
try {
const saved = localStorage.getItem('qr-history')
if (saved) {
scannedCodes.value = JSON.parse(saved)
}
} catch (e) {
console.error('Failed to load QR history', e)
}
}
watch(scannedCodes, (newVal) => {
localStorage.setItem('qr-history', JSON.stringify(newVal))
}, { deep: true })
const handleKeydown = (e) => {
if (e.key === 'Escape' && isFullscreen.value) {
toggleFullscreen()
}
}
const startScan = async () => {
try {
await startCamera()
updateVideoAspect()
startBackgroundLoop()
startDetection(onDetect)
} catch (err) {
// Error is handled by error computed property
}
}
onMounted(() => {
checkCameras()
loadHistory()
window.addEventListener('resize', updateVideoAspect)
window.addEventListener('resize', startBackgroundLoop)
window.addEventListener('keydown', handleKeydown)
watch(isFullscreen, (fs) => {
if (fs) {
startBackgroundLoop()
} else {
stopBackgroundLoop()
}
}, { immediate: true })
startScan()
})
onUnmounted(() => {
window.removeEventListener('resize', updateVideoAspect)
window.removeEventListener('resize', startBackgroundLoop)
window.removeEventListener('keydown', handleKeydown)
stopDetection()
stopCamera()
})
const switchCamera = (event) => {
if (event) event.stopPropagation()
baseSwitchCamera()
}
const toggleFullscreen = () => {
isFullscreen.value = !isFullscreen.value
}
const clearHistory = () => {
scannedCodes.value = []
}
const removeCode = (id) => {
scannedCodes.value = scannedCodes.value.filter(c => c.id !== id)
}
const copyAll = async () => {
if (scannedCodes.value.length === 0) return
const text = scannedCodes.value.map(c => c.value).join('\n')
try {
await navigator.clipboard.writeText(text)
} catch (err) {
console.error('Failed to copy', err)
}
}
const downloadJson = () => {
if (scannedCodes.value.length === 0) return
const data = JSON.stringify(scannedCodes.value, null, 2)
const blob = new Blob([data], { type: 'application/json' })
const url = URL.createObjectURL(blob)
const a = document.createElement('a')
a.href = url
a.download = `qr-scan-history-${new Date().toISOString().slice(0, 10)}.json`
document.body.appendChild(a)
a.click()
document.body.removeChild(a)
URL.revokeObjectURL(url)
}
const copyToClipboard = async (text) => {
try {
await navigator.clipboard.writeText(text)
} catch (err) {
console.error('Failed to copy', err)
}
}
const isUrl = (string) => {
try {
return Boolean(new URL(string))
} catch (_) {
return false
}
}
</script>
<template>
<div class="tool-container full-width">
<div class="tool-panel">
<div class="panel-header" v-if="!isFullscreen">
<h2 class="tool-title">QR Scanner</h2>
<div class="header-actions"></div>
</div>
<Teleport to="body" :disabled="!isFullscreen">
<div class="scanner-content" :class="{ 'is-fullscreen': isFullscreen }">
<canvas
v-if="isFullscreen"
ref="bgCanvas"
class="camera-bg"
:class="{ 'is-mirrored': isMirrored }"
></canvas>
<button v-if="isFullscreen" class="close-fullscreen-btn" @click="toggleFullscreen">
<X size="24" />
</button>
<div
class="camera-wrapper"
:class="{ 'clickable': !isFullscreen, 'is-mirrored': isMirrored }"
:style="desktopFullscreenStyle"
ref="wrapperRef"
@click="!isFullscreen && toggleFullscreen()"
>
<video
ref="videoRef"
class="camera-feed"
:class="{ 'is-mirrored': isMirrored }"
autoplay
playsinline
muted
></video>
<canvas ref="overlayCanvas" class="scan-overlay-canvas" :class="{ 'is-mirrored': isMirrored }"></canvas>
<div v-if="error" class="error-overlay">
<p>{{ error }}</p>
<button @click="startScan" class="retry-btn">Retry</button>
</div>
<button
v-if="hasMultipleCameras"
class="switch-camera-btn"
@click.stop="switchCamera"
title="Switch Camera"
>
<SwitchCamera size="24" />
</button>
</div>
<div class="results-section">
<div class="results-header">
<h3>Scanned Codes ({{ scannedCodes.length }})</h3>
<div v-if="scannedCodes.length > 0" class="header-actions">
<button class="icon-btn" @click="copyAll" title="Copy All">
<Copy size="18" />
</button>
<button class="icon-btn" @click="downloadJson" title="Download JSON">
<Download size="18" />
</button>
<button class="icon-btn delete-btn" @click="clearHistory" title="Clear All">
<Trash2 size="18" />
</button>
</div>
</div>
<div v-if="scannedCodes.length > 0" class="codes-list">
<div v-for="code in scannedCodes" :key="code.id" class="code-item">
<div class="code-content">
<div class="code-value">
<a v-if="isUrl(code.value)" :href="code.value" target="_blank" rel="noopener noreferrer">{{ code.value }}</a>
<span v-else>{{ code.value }}</span>
</div>
<div class="code-meta">
<span class="timestamp">{{ code.timestamp }}</span>
<span class="format-badge">{{ code.format }}</span>
</div>
</div>
<div class="item-actions">
<button class="icon-btn" @click="copyToClipboard(code.value)" title="Copy">
<Copy size="18" />
</button>
<button class="icon-btn delete-btn" @click="removeCode(code.id)" title="Remove">
<Trash2 size="18" />
</button>
</div>
</div>
</div>
<div v-else class="empty-state">
Point camera at a QR code to scan
</div>
</div>
</div>
</Teleport>
</div>
</div>
</template>
<style scoped>
.tool-container.full-width {
max-width: 100%;
height: 100%;
display: flex;
flex-direction: column;
}
.scanner-content {
display: flex;
flex-direction: column;
height: 100%;
gap: 1.5rem;
}
.scanner-content.is-fullscreen {
position: fixed;
inset: 0;
z-index: 9999;
background: #000;
gap: 0;
}
:global(:root[data-theme="light"] .scanner-content.is-fullscreen) {
background: #fff;
}
.camera-wrapper {
width: 100%;
max-width: 500px;
aspect-ratio: 1;
margin: 0 auto;
border-radius: 12px;
overflow: hidden;
position: relative;
background: #000;
border: 1px solid var(--glass-border);
box-shadow: var(--glass-shadow);
transition: all 0.3s ease;
}
:global(:root[data-theme="light"] .camera-wrapper) {
background: #f1f5f9;
}
.camera-wrapper.clickable {
cursor: pointer;
}
.scanner-content.is-fullscreen .camera-wrapper {
position: relative;
width: 100%;
aspect-ratio: 1;
max-width: none;
height: auto;
border-radius: 0;
border: none;
margin: 0;
z-index: 1;
}
.camera-bg {
position: absolute;
left: 0;
top: 0;
width: 100vw;
height: 50vh;
filter: blur(16px) saturate(110%);
opacity: 0.9;
z-index: 0;
}
.camera-bg.is-mirrored {
transform: scaleX(-1);
}
.camera-feed {
width: 100%;
height: 100%;
object-fit: cover;
display: block;
}
.camera-feed.is-mirrored {
transform: scaleX(-1);
}
.scan-overlay-canvas {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
pointer-events: none;
z-index: 5;
}
.scan-overlay-canvas.is-mirrored {
transform: scaleX(-1);
}
.error-overlay {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
display: flex;
align-items: center;
justify-content: center;
background: rgba(0, 0, 0, 0.8);
color: #ff4444;
padding: 1rem;
text-align: center;
z-index: 10;
}
.switch-camera-btn {
position: absolute;
top: 1rem;
right: 1rem;
background: rgba(0, 0, 0, 0.4);
border: 1px solid rgba(255, 255, 255, 0.2);
color: #fff;
border-radius: 50%;
width: 44px;
height: 44px;
display: flex;
align-items: center;
justify-content: center;
cursor: pointer;
z-index: 20;
backdrop-filter: blur(4px);
transition: all 0.2s;
}
.switch-camera-btn:hover {
background: rgba(0, 0, 0, 0.6);
transform: scale(1.05);
}
.close-fullscreen-btn {
position: absolute;
top: 1rem;
left: 1rem;
z-index: 20;
background: rgba(0, 0, 0, 0.4);
border: 1px solid rgba(255, 255, 255, 0.2);
color: #fff;
border-radius: 50%;
width: 44px;
height: 44px;
display: flex;
align-items: center;
justify-content: center;
cursor: pointer;
backdrop-filter: blur(4px);
}
.scanner-content.is-fullscreen .results-section {
position: relative;
flex: 1;
width: 100%;
height: auto;
z-index: 2;
border-radius: 0;
background: var(--glass-bg);
backdrop-filter: blur(10px);
border: none;
border-top: 1px solid var(--glass-border);
display: flex;
flex-direction: column;
}
:global(:root[data-theme="light"] .scanner-content.is-fullscreen .results-section) {
background: rgba(255, 255, 255, 0.75);
}
.code-value {
color: var(--primary-accent);
font-family: monospace;
word-break: break-all;
margin-bottom: 0.4rem;
}
.code-value a {
color: var(--primary-accent);
text-decoration: underline;
}
.code-value a:hover {
opacity: 0.8;
}
.code-meta {
display: flex;
gap: 1rem;
font-size: 0.85rem;
color: var(--text-secondary);
}
.format-badge {
background: rgba(255, 255, 255, 0.1);
padding: 0 0.4rem;
border-radius: 4px;
font-size: 0.75rem;
}
:global(:root[data-theme="light"]) .format-badge {
background: rgba(0, 0, 0, 0.1);
}
.empty-state {
flex: 1;
display: flex;
justify-content: center;
align-items: center;
color: var(--text-secondary);
padding: 2rem;
text-align: center;
}
:deep(video) {
object-fit: cover !important;
width: 100% !important;
height: 100% !important;
position: absolute !important;
left: 0 !important;
top: 0 !important;
}
:deep(.qrcode-stream-wrapper),
:deep(.qrcode-stream-overlay) {
width: 100% !important;
height: 100% !important;
position: absolute !important;
inset: 0 !important;
}
/* Front camera mirror (CSS-only) */
.camera-wrapper.is-front :deep(video) {
transform: scaleX(-1);
transform-origin: center;
}
.camera-wrapper.is-front :deep(#qrcode-stream-pause-frame),
.camera-wrapper.is-front :deep(#qrcode-stream-overlay) {
transform: scaleX(-1);
transform-origin: center;
}
@media (min-width: 768px) {
:deep(.scanner-content.is-fullscreen .camera-wrapper video) {
object-fit: contain !important;
}
}
</style>

View File

@@ -1,8 +1,8 @@
<script setup> <script setup>
import { ref, watch, onUnmounted } from 'vue' import { ref, watch, onUnmounted } from 'vue'
import { Copy, Trash2, ExternalLink, Power, Zap, X, Settings } from 'lucide-vue-next' import { Copy, Trash2, ExternalLink, Power, Zap, X, Settings, Download } from 'lucide-vue-next'
import { useExtension } from '../../composables/useExtension' import { useExtension } from '../../composables/useExtension'
import { useLocalStorage } from '../../composables/useLocalStorage' import { useUrlCleaner } from '../../composables/useUrlCleaner'
import ExtensionStatus from './common/ExtensionStatus.vue' import ExtensionStatus from './common/ExtensionStatus.vue'
import UrlCleanerExceptionsModal from './UrlCleanerExceptionsModal.vue' import UrlCleanerExceptionsModal from './UrlCleanerExceptionsModal.vue'
@@ -10,29 +10,22 @@ import UrlCleanerExceptionsModal from './UrlCleanerExceptionsModal.vue'
const { isExtensionReady, isListening, lastClipboardText, startListening, stopListening, writeClipboard } = useExtension() const { isExtensionReady, isListening, lastClipboardText, startListening, stopListening, writeClipboard } = useExtension()
const inputUrl = ref('') 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 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 {
const matchDomain = (pattern, domain) => { cleanedHistory,
// Escape regex chars except * isWatchEnabled,
const regexString = '^' + pattern.replace(/[.+^${}()|[\]\\]/g, '\\$&').replace(/\*/g, '.*') + '$' exceptions,
return new RegExp(regexString, 'i').test(domain) defaultExceptions,
} processUrl: baseProcessUrl,
removeEntry,
clearHistory
} = useUrlCleaner()
// Watch for clipboard changes from extension // Watch for clipboard changes from extension
watch(lastClipboardText, (newText) => { watch(lastClipboardText, (newText) => {
if (isWatchEnabled.value && newText) { if (isWatchEnabled.value && newText) {
processUrl(newText, true) baseProcessUrl(newText, true, writeClipboard)
} }
}) })
@@ -57,116 +50,53 @@ const toggleWatch = () => {
isWatchEnabled.value = !isWatchEnabled.value isWatchEnabled.value = !isWatchEnabled.value
} }
const copyAllUrls = async () => {
if (cleanedHistory.value.length === 0) return
const text = cleanedHistory.value.map(item => item.cleaned).join('\n')
try {
await navigator.clipboard.writeText(text)
} catch (err) {
console.error('Failed to copy URLs', err)
}
}
const downloadJson = () => {
if (cleanedHistory.value.length === 0) return
const exportData = cleanedHistory.value.map(item => ({
url: item.cleaned,
original: item.original,
timestamp: item.timestamp
}))
const data = JSON.stringify(exportData, null, 2)
const blob = new Blob([data], { type: 'application/json' })
const url = URL.createObjectURL(blob)
const a = document.createElement('a')
a.href = url
a.download = `cleaned-urls-${new Date().toISOString().slice(0, 10)}.json`
document.body.appendChild(a)
a.click()
document.body.removeChild(a)
URL.revokeObjectURL(url)
}
// Manual clean // Manual clean
const handleClean = () => { const handleClean = () => {
if (inputUrl.value) { if (inputUrl.value) {
processUrl(inputUrl.value, false) const urls = inputUrl.value.split(/\r?\n/).filter(line => line.trim().length > 0)
urls.forEach(url => {
baseProcessUrl(url.trim(), false, writeClipboard)
})
inputUrl.value = '' 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) => { const copyToClipboard = (text) => {
navigator.clipboard.writeText(text) navigator.clipboard.writeText(text)
} }
const removeEntry = (id) => {
cleanedHistory.value = cleanedHistory.value.filter(item => item.id !== id)
}
const clearHistory = () => {
cleanedHistory.value = []
}
onUnmounted(() => { onUnmounted(() => {
if (isListening.value) { if (isListening.value) {
stopListening() stopListening()
@@ -189,19 +119,19 @@ onUnmounted(() => {
<div class="input-section"> <div class="input-section">
<div class="input-wrapper"> <div class="input-wrapper">
<input <textarea
v-model="inputUrl" v-model="inputUrl"
type="text" placeholder="Paste URL(s) here to clean..."
placeholder="Paste URL here to clean..." class="tool-textarea url-input"
class="url-input" @keydown.enter.prevent="handleClean"
@keyup.enter="handleClean" rows="1"
> ></textarea>
<button class="btn-neon" @click="handleClean">
Clean
</button>
</div> </div>
<div class="watch-toggle"> <div class="watch-toggle">
<button class="btn-neon" @click="handleClean">
Clean
</button>
<button <button
class="btn-neon toggle-btn" class="btn-neon toggle-btn"
:class="{ 'active': isWatchEnabled && isExtensionReady }" :class="{ 'active': isWatchEnabled && isExtensionReady }"
@@ -219,10 +149,18 @@ onUnmounted(() => {
<div class="history-section" v-if="cleanedHistory.length > 0"> <div class="history-section" v-if="cleanedHistory.length > 0">
<div class="history-header"> <div class="history-header">
<h3>Cleaned URLs</h3> <h3>Cleaned URLs</h3>
<button class="icon-btn" @click="clearHistory" title="Clear History"> <div class="history-actions">
<button class="icon-btn" @click="copyAllUrls" title="Copy all URLs">
<Copy size="18" />
</button>
<button class="icon-btn" @click="downloadJson" title="Download JSON">
<Download size="18" />
</button>
<button class="icon-btn delete-btn" @click="clearHistory" title="Clear History">
<Trash2 size="18" /> <Trash2 size="18" />
</button> </button>
</div> </div>
</div>
<div class="history-list"> <div class="history-list">
<div v-for="item in cleanedHistory" :key="item.id" class="history-item"> <div v-for="item in cleanedHistory" :key="item.id" class="history-item">
@@ -274,22 +212,6 @@ onUnmounted(() => {
flex-direction: column; 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 { .input-section {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
@@ -331,24 +253,14 @@ onUnmounted(() => {
.url-input { .url-input {
flex: 1; flex: 1;
padding: 0.8rem 1rem; min-height: 120px;
border-radius: 8px; font-family: inherit;
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 { .watch-toggle {
display: flex; display: flex;
justify-content: flex-end; justify-content: space-between;
gap: 0.75rem;
} }
.toggle-btn { .toggle-btn {
@@ -364,75 +276,6 @@ onUnmounted(() => {
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 { .cleaned-url {
color: var(--primary-accent); color: var(--primary-accent);
font-family: monospace; font-family: monospace;
@@ -456,44 +299,6 @@ onUnmounted(() => {
gap: 0.2rem; 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 { .empty-state {
flex: 1; flex: 1;
display: flex; display: flex;
@@ -504,16 +309,6 @@ onUnmounted(() => {
padding: 2rem; padding: 2rem;
} }
.header-actions {
display: flex;
align-items: center;
gap: 0.8rem;
position: absolute;
right: 0;
top: 50%;
transform: translateY(-50%);
}
.settings-btn { .settings-btn {
background: rgba(255, 255, 255, 0.1); background: rgba(255, 255, 255, 0.1);
width: 32px; width: 32px;
@@ -531,12 +326,4 @@ onUnmounted(() => {
background: rgba(255, 255, 255, 0.2); background: rgba(255, 255, 255, 0.2);
color: var(--primary-accent); 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> </style>

View File

@@ -1,5 +1,5 @@
<script setup> <script setup>
import { ref, computed } from 'vue' import { ref, computed, watch, onUnmounted } from 'vue'
import { X, Plus, Trash2, RotateCcw } from 'lucide-vue-next' import { X, Plus, Trash2, RotateCcw } from 'lucide-vue-next'
const props = defineProps({ const props = defineProps({
@@ -10,6 +10,24 @@ const props = defineProps({
const emit = defineEmits(['close', 'update:exceptions']) const emit = defineEmits(['close', 'update:exceptions'])
const handleKeydown = (e) => {
if (e.key === 'Escape') {
emit('close')
}
}
watch(() => props.isOpen, (isOpen) => {
if (isOpen) {
window.addEventListener('keydown', handleKeydown)
} else {
window.removeEventListener('keydown', handleKeydown)
}
})
onUnmounted(() => {
window.removeEventListener('keydown', handleKeydown)
})
const newRule = ref({ const newRule = ref({
domainPattern: '', domainPattern: '',
keepParams: '', keepParams: '',
@@ -80,6 +98,15 @@ const addRule = () => {
} }
} }
const editRule = (rule) => {
newRule.value = {
domainPattern: rule.domainPattern,
keepParams: Array.isArray(rule.keepParams) ? rule.keepParams.join(', ') : '',
keepHash: !!rule.keepHash,
keepAllParams: !!rule.keepAllParams
}
}
const removeRule = (id) => { const removeRule = (id) => {
localExceptions.value = localExceptions.value.filter(r => r.id !== id) localExceptions.value = localExceptions.value.filter(r => r.id !== id)
} }
@@ -145,7 +172,7 @@ const resetToDefault = (ruleId) => {
<template> <template>
<Teleport to="body"> <Teleport to="body">
<div v-if="isOpen" class="modal-overlay" @click.self="$emit('close')"> <div v-if="isOpen" class="modal-overlay" @click.self="$emit('close')">
<div class="modal-content glass-panel"> <div class="modal-content">
<div class="modal-header"> <div class="modal-header">
<h3>URL Cleaning Exceptions</h3> <h3>URL Cleaning Exceptions</h3>
<button class="close-btn" @click="$emit('close')"> <button class="close-btn" @click="$emit('close')">
@@ -200,7 +227,7 @@ const resetToDefault = (ruleId) => {
<div v-for="rule in localExceptions" :key="rule.id" class="rule-item" :class="{ disabled: !rule.isEnabled }"> <div v-for="rule in localExceptions" :key="rule.id" class="rule-item" :class="{ disabled: !rule.isEnabled }">
<div class="rule-info"> <div class="rule-info">
<div class="rule-domain">{{ rule.domainPattern }}</div> <div class="rule-domain" @click="editRule(rule)" title="Click to edit">{{ rule.domainPattern }}</div>
<div class="rule-details"> <div class="rule-details">
<div class="params-list"> <div class="params-list">
<template v-if="rule.keepAllParams"> <template v-if="rule.keepAllParams">
@@ -291,6 +318,8 @@ const resetToDefault = (ruleId) => {
color: var(--text-color); color: var(--text-color);
} }
.modal-header { .modal-header {
padding: 1.5rem; padding: 1.5rem;
border-bottom: 1px solid var(--glass-border); border-bottom: 1px solid var(--glass-border);
@@ -457,6 +486,13 @@ const resetToDefault = (ruleId) => {
font-weight: 600; font-weight: 600;
color: var(--primary-accent); color: var(--primary-accent);
margin-bottom: 0.3rem; margin-bottom: 0.3rem;
cursor: pointer;
transition: opacity 0.2s;
}
.rule-domain:hover {
opacity: 0.8;
text-decoration: underline;
} }
.rule-details { .rule-details {
@@ -559,8 +595,4 @@ const resetToDefault = (ruleId) => {
.delete-btn:hover { .delete-btn:hover {
color: #ef4444; color: #ef4444;
} }
:global(:root[data-theme="light"]) .delete-btn:hover {
color: #dc2626;
}
</style> </style>

View File

@@ -5,7 +5,7 @@ export default {
</script> </script>
<script setup> <script setup>
import { ref } from 'vue' import { ref, watch, onUnmounted } from 'vue'
import { Plug, Plus, X } from 'lucide-vue-next' import { Plug, Plus, X } from 'lucide-vue-next'
const props = defineProps({ const props = defineProps({
@@ -13,6 +13,24 @@ const props = defineProps({
}) })
const showModal = ref(false) const showModal = ref(false)
const handleKeydown = (e) => {
if (e.key === 'Escape' && showModal.value) {
showModal.value = false
}
}
watch(showModal, (isOpen) => {
if (isOpen) {
window.addEventListener('keydown', handleKeydown)
} else {
window.removeEventListener('keydown', handleKeydown)
}
})
onUnmounted(() => {
window.removeEventListener('keydown', handleKeydown)
})
</script> </script>
<template> <template>
@@ -38,7 +56,14 @@ const showModal = ref(false)
With the extension, you can capture and process content even when you're working in other apps. With the extension, you can capture and process content even when you're working in other apps.
</p> </p>
<div class="modal-actions"> <div class="modal-actions">
<a href="#" class="btn-neon" @click.prevent>Extension Coming Soon</a> <a
href="https://chromewebstore.google.com/detail/tools-app-extension/bhcpbmfncohogehbhebiffcgjcndnneg"
target="_blank"
rel="noopener noreferrer"
class="btn-neon"
>
Install Extension
</a>
</div> </div>
</div> </div>
@@ -102,19 +127,23 @@ const showModal = ref(false)
padding: 1rem; padding: 1rem;
} }
:global(:root[data-theme="light"]) .modal-overlay {
background: rgba(255, 255, 255, 0.3);
}
.modal-content { .modal-content {
background: var(--glass-bg); background: var(--glass-bg);
backdrop-filter: blur(16px); backdrop-filter: blur(16px);
-webkit-backdrop-filter: blur(16px); -webkit-backdrop-filter: blur(16px);
border: 1px solid var(--glass-border); border: 1px solid var(--glass-border);
border-radius: 16px; border-radius: 16px;
padding: 2.5rem; /* Większy padding */ padding: 2.5rem;
max-width: 500px; max-width: 500px;
width: 90%; width: 90%;
position: relative; position: relative;
box-shadow: var(--glass-shadow); box-shadow: var(--glass-shadow);
text-align: center; text-align: center;
color: var(--text-color); /* Wymuś kolor tekstu */ color: var(--text-color);
} }
.close-btn { .close-btn {

View File

@@ -0,0 +1,110 @@
import { ref, watch, onUnmounted } from 'vue'
export function useCamera(videoRef) {
const stream = ref(null)
const facingMode = ref('environment')
const hasMultipleCameras = ref(false)
const isMirrored = ref(false)
const error = ref('')
const checkCameras = async () => {
try {
if (!navigator.mediaDevices || !navigator.mediaDevices.enumerateDevices) {
return
}
const devices = await navigator.mediaDevices.enumerateDevices()
const cameras = devices.filter(d => d.kind === 'videoinput')
hasMultipleCameras.value = cameras.length > 1
} catch (e) {
console.error('Error checking cameras:', e)
}
}
const stopCamera = () => {
if (stream.value) {
stream.value.getTracks().forEach(t => t.stop())
stream.value = null
}
}
const startCamera = async () => {
stopCamera()
error.value = ''
try {
const constraints = {
video: {
facingMode: facingMode.value,
width: { ideal: 1280 },
height: { ideal: 720 }
}
}
const mediaStream = await navigator.mediaDevices.getUserMedia(constraints)
stream.value = mediaStream
// Detect actual facing mode to mirror front camera correctly
const videoTrack = mediaStream.getVideoTracks()[0]
if (videoTrack) {
const settings = videoTrack.getSettings()
if (settings.facingMode) {
isMirrored.value = settings.facingMode === 'user'
} else {
// Fallback: check label for desktop cameras or assume requested mode
const label = videoTrack.label ? videoTrack.label.toLowerCase() : ''
if (label.includes('front') || label.includes('facetime') || label.includes('macbook')) {
isMirrored.value = true
} else {
isMirrored.value = facingMode.value === 'user'
}
}
}
if (videoRef.value) {
videoRef.value.srcObject = mediaStream
return new Promise((resolve) => {
videoRef.value.onloadedmetadata = () => {
videoRef.value.play().catch(e => console.error('Play error', e))
resolve()
}
})
}
} catch (err) {
if (err.name === 'NotAllowedError') {
error.value = 'Camera permission denied'
} else if (err.name === 'NotFoundError') {
error.value = 'No camera found'
} else {
error.value = `Camera error: ${err.name}`
}
throw err // Let caller know it failed
}
}
const switchCamera = () => {
facingMode.value = facingMode.value === 'environment' ? 'user' : 'environment'
}
watch(facingMode, () => {
if (stream.value) {
// Re-start if already running
startCamera().catch(() => { })
}
})
onUnmounted(() => {
stopCamera()
})
return {
stream,
facingMode,
hasMultipleCameras,
isMirrored,
error,
checkCameras,
startCamera,
stopCamera,
switchCamera
}
}

View File

@@ -1,6 +1,7 @@
import { onMounted, onUnmounted, ref, nextTick } from 'vue' import { onMounted, onUnmounted, ref, nextTick, watch } from 'vue'
import { UI_CONFIG } from '../config/ui'
export function useFillHeight(elementRef, marginBottom = 20) { export function useFillHeight(elementRef, extraMargin = 0) {
const height = ref('auto') const height = ref('auto')
const updateHeight = () => { const updateHeight = () => {
@@ -8,16 +9,10 @@ export function useFillHeight(elementRef, marginBottom = 20) {
const rect = elementRef.value.getBoundingClientRect() const rect = elementRef.value.getBoundingClientRect()
const windowHeight = window.innerHeight 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). // Calculate available space: window height - element top position - footer height - padding - extra margin
// This assumes the element should stretch to the bottom of the viewport. const bottomOffset = UI_CONFIG.footerHeight + UI_CONFIG.pagePadding + extraMargin
const availableHeight = windowHeight - rect.top - bottomOffset
const availableHeight = windowHeight - rect.top - marginBottom
// Ensure minimum height // Ensure minimum height
if (availableHeight > 100) { if (availableHeight > 100) {
@@ -34,6 +29,13 @@ export function useFillHeight(elementRef, marginBottom = 20) {
// Also update after a short delay to ensure layout is settled (e.g. sidebar transitions) // Also update after a short delay to ensure layout is settled (e.g. sidebar transitions)
setTimeout(updateHeight, 300) setTimeout(updateHeight, 300)
// Watch for element appearing (v-if) or changing
watch(elementRef, () => {
nextTick(updateHeight)
// Additional update for layout stability
setTimeout(updateHeight, 100)
})
}) })
onUnmounted(() => { onUnmounted(() => {

View File

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

View File

@@ -0,0 +1,174 @@
import { ref, onMounted, onUnmounted } from 'vue'
export function useQrDetection(videoRef, overlayCanvasRef) {
let barcodeDetector = null // must be plain variable, NOT a Vue ref (Proxy breaks native private fields)
const isDetecting = ref(false)
const error = ref('')
let scanRafId = null
// Function to initialize detector
const initDetector = async () => {
if (!barcodeDetector) {
if ('BarcodeDetector' in window) {
try {
const formats = await window.BarcodeDetector.getSupportedFormats()
if (formats.includes('qr_code')) {
barcodeDetector = new window.BarcodeDetector({ formats: ['qr_code'] })
} else {
barcodeDetector = new window.BarcodeDetector()
}
} catch (e) {
barcodeDetector = new window.BarcodeDetector()
}
} else {
error.value = 'Barcode Detection API not supported on this device/browser.'
}
}
}
const paintDetections = (codes) => {
const canvas = overlayCanvasRef.value
const video = videoRef.value
if (!canvas || !video) return
const ctx = canvas.getContext('2d')
const { width, height } = canvas.getBoundingClientRect()
// Update canvas size if needed (to match CSS size)
if (canvas.width !== width || canvas.height !== height) {
canvas.width = width
canvas.height = height
}
ctx.clearRect(0, 0, width, height)
if (!codes || codes.length === 0) return
const vw = video.videoWidth
const vh = video.videoHeight
if (!vw || !vh) return
// Calculate object-fit: cover scaling
const videoRatio = vw / vh
const canvasRatio = width / height
let drawWidth, drawHeight, startX, startY
if (canvasRatio > videoRatio) {
// Canvas is wider than video (video cropped top/bottom)
drawWidth = width
drawHeight = width / videoRatio
startX = 0
startY = (height - drawHeight) / 2
} else {
// Canvas is taller than video (video cropped left/right)
drawHeight = height
drawWidth = height * videoRatio
startY = 0
startX = (width - drawWidth) / 2
}
const scale = drawWidth / vw
// Styles
const styles = getComputedStyle(document.documentElement)
const accent = styles.getPropertyValue('--primary-accent').trim() || '#00f2fe'
ctx.lineWidth = 4
ctx.strokeStyle = accent
ctx.fillStyle = accent
codes.forEach(code => {
const points = code.cornerPoints
if (!points || points.length < 4) return
ctx.beginPath()
const transform = (p) => {
let x = p.x * scale + startX
let y = p.y * scale + startY
return { x, y }
}
const p0 = transform(points[0])
ctx.moveTo(p0.x, p0.y)
for (let i = 1; i < points.length; i++) {
const p = transform(points[i])
ctx.lineTo(p.x, p.y)
}
ctx.closePath()
ctx.stroke()
// Draw corners
points.forEach(p => {
const tp = transform(p)
ctx.beginPath()
ctx.arc(tp.x, tp.y, 4, 0, Math.PI * 2)
ctx.fill()
})
})
}
const startDetection = async (onDetectCallback) => {
error.value = ''
try {
await initDetector()
if (!barcodeDetector) {
if (!error.value) error.value = 'Barcode Detector failed to initialize'
return
}
isDetecting.value = true
const detectLoop = async () => {
const video = videoRef.value
if (!isDetecting.value) return
if (!video || video.readyState < 2) {
scanRafId = requestAnimationFrame(detectLoop)
return
}
try {
const codes = await barcodeDetector.detect(video)
paintDetections(codes)
if (codes.length > 0 && onDetectCallback) {
onDetectCallback(codes)
}
} catch (e) {
// Silent catch for intermittent detection frames failing
}
if (isDetecting.value) {
scanRafId = requestAnimationFrame(detectLoop)
}
}
detectLoop() // start loop
} catch (e) {
error.value = `Detection error: ${e.message}`
}
}
const stopDetection = () => {
isDetecting.value = false
if (scanRafId) cancelAnimationFrame(scanRafId)
// Clear canvas
if (overlayCanvasRef.value) {
const ctx = overlayCanvasRef.value.getContext('2d')
ctx.clearRect(0, 0, overlayCanvasRef.value.width, overlayCanvasRef.value.height)
}
}
onUnmounted(() => {
stopDetection()
})
return {
error,
isDetecting,
startDetection,
stopDetection
}
}

View File

@@ -0,0 +1,115 @@
import { useLocalStorage } from './useLocalStorage'
export function useUrlCleaner() {
const cleanedHistory = useLocalStorage('url-cleaner-history', [])
const isWatchEnabled = useLocalStorage('url-cleaner-watch-enabled', 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)
const matchDomain = (pattern, domain) => {
// Escape regex chars except *
const regexString = '^' + pattern.replace(/[.+^${}()|[\]\\]/g, '\\$&').replace(/\*/g, '.*') + '$'
return new RegExp(regexString, 'i').test(domain)
}
const processUrl = (text, autoClipboard = false, writeClipboardFn = null) => {
try {
// Basic URL validation
if (!text.match(/^https?:\/\//i)) {
if (autoClipboard) return text
}
const originalLength = text.length
let cleanedUrl = text
try {
const urlObj = new URL(text)
const hostname = urlObj.hostname
const matchedRule = exceptions.value.find(rule =>
rule.isEnabled && matchDomain(rule.domainPattern, hostname)
)
if (matchedRule) {
if (!matchedRule.keepAllParams) {
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 {
if (urlObj.search || urlObj.hash) {
urlObj.search = ''
urlObj.hash = ''
}
}
cleanedUrl = urlObj.toString()
} catch (e) {
return text
}
if (cleanedUrl === text && autoClipboard) {
return text
}
const newLength = cleanedUrl.length
const savedChars = originalLength - newLength
const savedPercent = originalLength > 0 ? Math.round((savedChars / originalLength) * 100) : 0
const entry = {
id: Date.now(),
original: text,
cleaned: cleanedUrl,
savedPercent,
timestamp: new Date().toLocaleTimeString()
}
cleanedHistory.value.unshift(entry)
if (cleanedHistory.value.length > 50) {
cleanedHistory.value.pop()
}
if (autoClipboard && savedChars > 0 && writeClipboardFn) {
writeClipboardFn(cleanedUrl)
}
return cleanedUrl
} catch (e) {
console.error('Error processing URL:', e)
return text
}
}
const removeEntry = (id) => {
cleanedHistory.value = cleanedHistory.value.filter(item => item.id !== id)
}
const clearHistory = () => {
cleanedHistory.value = []
}
return {
cleanedHistory,
isWatchEnabled,
exceptions,
defaultExceptions,
processUrl,
removeEntry,
clearHistory
}
}

6
src/config/ui.js Normal file
View File

@@ -0,0 +1,6 @@
export const UI_CONFIG = {
headerHeight: 64,
footerHeight: 50,
pagePadding: 16 // Single side padding (1rem)
}

View File

@@ -3,6 +3,27 @@ import './style.css'
import App from './App.vue' import App from './App.vue'
import router from './router' import router from './router'
import Ripple from './directives/ripple' import Ripple from './directives/ripple'
import { BarcodeDetector, prepareZXingModule } from 'barcode-detector/ponyfill'
// Configure BarcodeDetector polyfill to use local WASM file
try {
prepareZXingModule({
overrides: {
locateFile: (path, prefix) => {
if (path.endsWith('.wasm')) {
return '/wasm/zxing_reader.wasm'
}
return prefix + path
}
}
})
// Force usage of polyfill to avoid CSP issues and ensure consistent behavior
// Native implementation might fail or be missing in some browsers
window.BarcodeDetector = BarcodeDetector
} catch (e) {
console.error('Failed to initialize BarcodeDetector polyfill', e)
}
const app = createApp(App) const app = createApp(App)

View File

@@ -3,6 +3,8 @@ 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 ClipboardSniffer from '../components/tools/ClipboardSniffer.vue'
import UrlCleaner from '../components/tools/UrlCleaner.vue' import UrlCleaner from '../components/tools/UrlCleaner.vue'
import QrScanner from '../components/tools/QrScanner.vue'
import QrCode from '../components/tools/QrCode.vue'
import PrivacyPolicy from '../views/PrivacyPolicy.vue' import PrivacyPolicy from '../views/PrivacyPolicy.vue'
const routes = [ const routes = [
@@ -26,6 +28,16 @@ const routes = [
name: 'UrlCleaner', name: 'UrlCleaner',
component: UrlCleaner component: UrlCleaner
}, },
{
path: '/qr-scanner',
name: 'QrScanner',
component: QrScanner
},
{
path: '/qr-code',
name: 'QrCode',
component: QrCode
},
{ {
path: '/extension-privacy-policy', path: '/extension-privacy-policy',
name: 'PrivacyPolicy', name: 'PrivacyPolicy',

View File

@@ -1,9 +1,10 @@
/* Box sizing reset */ /* Box sizing reset */
*, *::before, *::after { *,
*::before,
*::after {
box-sizing: border-box; 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;
@@ -42,9 +43,13 @@
--title-gradient: linear-gradient(45deg, #00f2fe, #4facfe); --title-gradient: linear-gradient(45deg, #00f2fe, #4facfe);
--ripple-color: rgba(255, 255, 255, 0.3); --ripple-color: rgba(255, 255, 255, 0.3);
--nav-item-weight: 400; --nav-item-weight: 400;
--list-hover-bg: rgba(255, 255, 255, 0.05);
--list-border: rgba(255, 255, 255, 0.12);
--header-bg: rgba(0, 0, 0, 0.6);
color: var(--text-color); color: var(--text-color);
background-color: #242424; /* Fallback */ background-color: #242424;
/* Fallback */
background: var(--bg-gradient); background: var(--bg-gradient);
background-attachment: fixed; background-attachment: fixed;
@@ -55,35 +60,38 @@
} }
:root[data-theme="light"] { :root[data-theme="light"] {
--bg-gradient: radial-gradient(circle at center, #ffffff 0%, #e5e7eb 100%); --bg-gradient: radial-gradient(circle at center, #f8fafc 0%, #e2e8f0 100%);
--glass-bg: rgba(255, 255, 255, 0.75); --glass-bg: rgba(255, 255, 255, 0.85);
--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(30, 41, 59, 0.15);
--text-color: #000000; --text-color: #0f172a;
--text-strong: #000000; --text-strong: #020617;
--text-secondary: #000000; --text-secondary: #334155;
--text-muted: rgba(0, 0, 0, 0.7); --text-muted: #64748b;
--accent-cyan: #0ea5e9; --accent-cyan: #0ea5e9;
--accent-purple: #6366f1; --accent-purple: #6366f1;
--primary-accent: #0ea5e9; --primary-accent: #0ea5e9;
--title-glow: rgba(14, 165, 233, 0.35); --title-glow: rgba(14, 165, 233, 0.35);
--toggle-bg: rgba(255, 255, 255, 0.85); --toggle-bg: rgba(255, 255, 255, 1);
--toggle-border: rgba(15, 23, 42, 0.12); --toggle-border: rgba(15, 23, 42, 0.2);
--toggle-shadow: 0 8px 20px rgba(15, 23, 42, 0.12); --toggle-shadow: 0 4px 12px rgba(15, 23, 42, 0.1);
--toggle-btn-border: rgba(15, 23, 42, 0.18); --toggle-btn-border: rgba(15, 23, 42, 0.15);
--toggle-hover-border: rgba(15, 23, 42, 0.5); --toggle-hover-border: rgba(14, 165, 233, 0.6);
--toggle-active-shadow: 0 0 12px rgba(14, 165, 233, 0.25); --toggle-active-shadow: 0 0 12px rgba(14, 165, 233, 0.25);
--panel-bg: rgba(255, 255, 255, 0.7); --panel-bg: rgba(255, 255, 255, 0.9);
--panel-border: rgba(15, 23, 42, 0.12); --panel-border: rgba(15, 23, 42, 0.12);
--panel-shadow: 0 12px 24px rgba(15, 23, 42, 0.12); --panel-shadow: 0 12px 24px rgba(15, 23, 42, 0.12);
--button-bg: rgba(255, 255, 255, 0.85); --button-bg: rgba(255, 255, 255, 0.7);
--button-border: rgba(15, 23, 42, 0.16); --button-border: rgba(255, 255, 255, 0.9);
--button-text: #0f172a; --button-text: #0f172a;
--button-hover-bg: rgba(0, 0, 0, 0.05); --button-hover-bg: rgba(15, 23, 42, 0.1);
--button-hover-shadow: 0 6px 18px rgba(15, 23, 42, 0.18); --button-hover-shadow: 0 6px 18px rgba(15, 23, 42, 0.18);
--button-active-shadow: 0 0 18px rgba(14, 165, 233, 0.25); --button-active-shadow: 0 0 18px rgba(14, 165, 233, 0.25);
--title-gradient: linear-gradient(45deg, #0ea5e9, #6366f1); --title-gradient: linear-gradient(45deg, #0ea5e9, #6366f1);
--ripple-color: rgba(0, 0, 0, 0.1); --ripple-color: rgba(0, 0, 0, 0.1);
--list-hover-bg: rgba(15, 23, 42, 0.05);
--list-border: rgba(15, 23, 42, 0.08);
--header-bg: rgba(255, 255, 255, 0.6);
} }
body { body {
@@ -111,35 +119,38 @@ body {
@media (min-width: 768px) { @media (min-width: 768px) {
body { body {
height: 100vh; min-height: 100vh;
overflow: hidden; overflow: auto;
} }
#app { #app {
height: 100vh; min-height: 100vh;
overflow: hidden; overflow: auto;
} }
} }
/* Removed global front camera mirror to restore stability */
/* --- Shared styles for all tools (moved from tools.css) --- */ /* --- Shared styles for all tools (moved from tools.css) --- */
.tool-container { .tool-container {
display: flex; display: flex;
justify-content: center; justify-content: center;
width: 100%; width: 100%;
max-width: 800px; max-width: 100%;
margin: 0 auto; margin: 0 auto;
height: 100%; height: 100%;
padding: 1rem; padding: 0.5rem;
} }
.tool-panel { .tool-panel {
width: 100%; width: 100%;
padding: 2rem; padding: 1rem;
border-radius: 16px; border-radius: 16px;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
gap: 1.5rem; gap: 1.5rem;
height: 100%;
max-height: 100%; max-height: 100%;
overflow-y: auto; overflow-y: auto;
background: var(--glass-bg); background: var(--glass-bg);
@@ -170,7 +181,7 @@ body {
.tool-title { .tool-title {
margin: 0; margin: 0;
text-align: center; text-align: left;
font-size: 1.5rem; font-size: 1.5rem;
font-weight: 600; font-weight: 600;
background: var(--title-gradient); background: var(--title-gradient);
@@ -181,40 +192,45 @@ body {
filter: drop-shadow(0 0 10px var(--title-glow)); 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 {
.tool-textarea,
.select-input {
width: 100%; width: 100%;
height: 100%; padding: 0.75rem 1rem;
padding: 1rem; background-color: var(--toggle-bg);
background-color: rgba(0, 0, 0, 0.2) !important;
border: 1px solid var(--toggle-border); border: 1px solid var(--toggle-border);
border-radius: 12px; border-radius: 8px;
color: #ffffff !important; /* Explicit white color for dark mode */ color: var(--text-color);
font-family: monospace; font-family: inherit;
font-size: 1rem; font-size: 0.95rem;
line-height: 1.6;
resize: none;
transition: all 0.3s ease; transition: all 0.3s ease;
box-sizing: border-box; box-sizing: border-box;
} }
:root[data-theme="light"] .tool-textarea { .tool-textarea {
color: #000000 !important; font-family: monospace;
background-color: rgba(255, 255, 255, 0.5) !important; resize: none;
height: 100%;
} }
.tool-textarea:focus { .select-input {
appearance: none;
background-image: url("data:image/svg+xml,%3Csvg 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'%3E%3Cpath d='m6 9 6 6 6-6'%3E%3C/path%3E%3C/svg%3E");
background-repeat: no-repeat;
background-position: right 0.75rem center;
background-size: 1rem;
padding-right: 2.5rem;
height: 40px;
cursor: pointer;
line-height: 1;
}
.tool-textarea:focus,
.select-input:focus {
outline: none; outline: none;
border-color: #00f2fe !important; /* Force cyan accent */ border-color: var(--primary-accent) !important;
box-shadow: 0 0 0 1px #00f2fe !important; box-shadow: 0 0 0 1px var(--primary-accent) !important;
} }
.result-area { .result-area {
@@ -253,9 +269,10 @@ body {
background: var(--button-bg); background: var(--button-bg);
border: 1px solid var(--button-border); border: 1px solid var(--button-border);
color: var(--button-text); color: var(--button-text);
padding: 8px 16px; padding: 0 1.25rem;
height: 40px;
border-radius: 8px; border-radius: 8px;
font-weight: 600; font-weight: 500;
cursor: pointer; cursor: pointer;
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1); transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1); box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);
@@ -263,7 +280,8 @@ body {
align-items: center; align-items: center;
justify-content: center; justify-content: center;
gap: 8px; gap: 8px;
outline: none; /* Remove focus outline */ outline: none;
/* Remove focus outline */
} }
/* Global button styles */ /* Global button styles */
@@ -333,3 +351,304 @@ span.ripple {
opacity: 0; opacity: 0;
} }
} }
/* --- Global Input/Select Focus Styles --- */
input:focus,
select:focus,
textarea:focus,
button:focus {
outline: none;
}
input:focus,
select:focus,
textarea:focus,
.number-control:focus-within {
border-color: var(--primary-accent) !important;
box-shadow: 0 0 0 1px var(--primary-accent) !important;
}
/* --- Global Checkbox Styles --- */
.checkbox-label {
display: flex;
align-items: center;
gap: 10px;
cursor: pointer;
user-select: none;
color: var(--text-secondary);
font-weight: 500;
white-space: nowrap;
transition: color 0.2s;
}
.checkbox-label:hover {
color: var(--text-color);
}
.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;
flex-shrink: 0;
}
.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 white;
border-width: 0 2px 2px 0;
transform: rotate(45deg);
}
:root[data-theme="light"] .checkmark:after {
border-color: white;
/* Keep checkmark white even in light mode if background is primary-accent */
}
/* --- Global Header/Action Patterns --- */
.panel-header {
display: flex;
justify-content: space-between;
align-items: center;
flex-wrap: wrap;
gap: 1rem;
width: 100%;
min-height: 40px;
}
.header-actions {
display: flex;
align-items: center;
gap: 0.8rem;
}
/* --- Global Icon Button Styles --- */
.icon-btn {
background: none;
border: none;
color: var(--text-secondary);
cursor: pointer;
padding: 0.5rem;
border-radius: 8px;
transition: all 0.2s;
display: flex;
align-items: center;
justify-content: center;
}
.icon-btn:hover {
color: var(--text-color);
background: var(--button-hover-bg);
}
.icon-btn.delete-btn:hover {
color: #ef4444;
background: rgba(239, 68, 68, 0.1);
}
:root[data-theme="light"] .icon-btn.delete-btn:hover {
color: #dc2626;
background: rgba(220, 38, 38, 0.05);
}
/* --- Component Specific Theme Overrides (Consolidated) --- */
:root[data-theme="light"] .settings-btn {
background: rgba(0, 0, 0, 0.03);
}
:root[data-theme="light"] .savings {
color: #16a34a;
font-weight: 500;
}
:root[data-theme="light"] .modal-overlay {
background: rgba(255, 255, 255, 0.3);
}
:root[data-theme="light"] .add-rule-form,
:root[data-theme="light"] .rule-item {
background: rgba(0, 0, 0, 0.04);
border: 1px solid rgba(0, 0, 0, 0.06);
}
/* --- Global Number Control Styles --- */
.number-control {
display: flex;
align-items: stretch;
background: var(--toggle-bg);
border: 1px solid var(--toggle-border);
border-radius: 8px;
overflow: hidden;
height: 40px;
}
.control-btn {
background: none;
border: none;
color: var(--text-color);
font-size: 0.9rem;
min-width: 40px;
padding: 0 0.5rem;
cursor: pointer;
transition: all 0.2s;
display: flex;
align-items: center;
justify-content: center;
font-weight: 400;
}
.control-btn:hover {
background: var(--button-hover-bg);
color: var(--primary-accent);
}
.number-input {
background: none;
border: none;
color: var(--text-color);
flex: 1;
min-width: 60px;
text-align: center;
font-size: 1rem;
font-weight: 400;
appearance: textfield;
}
.number-input::-webkit-outer-spin-button,
.number-input::-webkit-inner-spin-button {
-webkit-appearance: none;
margin: 0;
}
/* --- Global List/History Styles --- */
.results-section,
.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,
.results-header {
padding: 1rem;
border-bottom: 1px solid var(--glass-border);
display: flex;
justify-content: space-between;
align-items: center;
}
.history-header h3,
.results-header h3 {
margin: 0;
font-size: 1.1rem;
color: var(--text-strong);
}
.history-actions,
.results-actions,
.header-actions {
display: flex;
gap: 0.5rem;
align-items: center;
}
.history-list,
.codes-list {
flex: 1;
overflow-y: auto;
padding: 0;
}
.history-item,
.code-item {
display: flex;
justify-content: space-between;
align-items: center;
padding: 1rem;
border-bottom: 1px solid var(--list-border);
transition: background 0.2s;
}
.history-item:last-child,
.code-item:last-child {
border-bottom: none;
}
.history-item:hover,
.code-item:hover {
background: var(--list-hover-bg);
}
.item-info,
.code-content {
flex: 1;
overflow: hidden;
padding-right: 1rem;
}
.item-actions {
display: flex;
gap: 0.5rem;
align-items: center;
}
.status-dot {
width: 8px;
height: 8px;
border-radius: 50%;
background-color: #4ade80;
box-shadow: 0 0 8px #4ade80;
animation: pulse 2s infinite;
}
@keyframes pulse {
0% {
opacity: 1;
}
50% {
opacity: 0.5;
}
100% {
opacity: 1;
}
}

View File

@@ -1,107 +0,0 @@
/* 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);
}

View File

@@ -43,5 +43,11 @@ export default defineConfig({
], ],
define: { define: {
'__APP_VERSION__': JSON.stringify(packageJson.version) '__APP_VERSION__': JSON.stringify(packageJson.version)
},
server: {
host: true,
allowedHosts: [
'.trycloudflare.com'
]
} }
}) })