feat: 初始化 NFC 读卡器项目
- 添加 Electron 主进程和渲染进程 - 实现读卡器连接、断开和读取卡片 ID 功能 - 添加自定义窗口标题栏和窗口控制 - 实现简洁美观的用户界面 - 添加项目文档 README.md
This commit is contained in:
193
.gitignore
vendored
Normal file
193
.gitignore
vendored
Normal file
@@ -0,0 +1,193 @@
|
||||
# File created using '.gitignore Generator' for Visual Studio Code: https://bit.ly/vscode-gig
|
||||
# Created by https://www.toptal.com/developers/gitignore/api/windows,visualstudiocode,node
|
||||
# Edit at https://www.toptal.com/developers/gitignore?templates=windows,visualstudiocode,node
|
||||
|
||||
### Node ###
|
||||
# Logs
|
||||
logs
|
||||
*.log
|
||||
npm-debug.log*
|
||||
yarn-debug.log*
|
||||
yarn-error.log*
|
||||
lerna-debug.log*
|
||||
.pnpm-debug.log*
|
||||
|
||||
# Diagnostic reports (https://nodejs.org/api/report.html)
|
||||
report.[0-9]*.[0-9]*.[0-9]*.[0-9]*.json
|
||||
|
||||
# Runtime data
|
||||
pids
|
||||
*.pid
|
||||
*.seed
|
||||
*.pid.lock
|
||||
|
||||
# Directory for instrumented libs generated by jscoverage/JSCover
|
||||
lib-cov
|
||||
|
||||
# Coverage directory used by tools like istanbul
|
||||
coverage
|
||||
*.lcov
|
||||
|
||||
# nyc test coverage
|
||||
.nyc_output
|
||||
|
||||
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
|
||||
.grunt
|
||||
|
||||
# Bower dependency directory (https://bower.io/)
|
||||
bower_components
|
||||
|
||||
# node-waf configuration
|
||||
.lock-wscript
|
||||
|
||||
# Compiled binary addons (https://nodejs.org/api/addons.html)
|
||||
build/Release
|
||||
|
||||
# Dependency directories
|
||||
node_modules/
|
||||
jspm_packages/
|
||||
|
||||
# Snowpack dependency directory (https://snowpack.dev/)
|
||||
web_modules/
|
||||
|
||||
# TypeScript cache
|
||||
*.tsbuildinfo
|
||||
|
||||
# Optional npm cache directory
|
||||
.npm
|
||||
|
||||
# Optional eslint cache
|
||||
.eslintcache
|
||||
|
||||
# Optional stylelint cache
|
||||
.stylelintcache
|
||||
|
||||
# Microbundle cache
|
||||
.rpt2_cache/
|
||||
.rts2_cache_cjs/
|
||||
.rts2_cache_es/
|
||||
.rts2_cache_umd/
|
||||
|
||||
# Optional REPL history
|
||||
.node_repl_history
|
||||
|
||||
# Output of 'npm pack'
|
||||
*.tgz
|
||||
|
||||
# Yarn Integrity file
|
||||
.yarn-integrity
|
||||
|
||||
# dotenv environment variable files
|
||||
.env
|
||||
.env.development.local
|
||||
.env.test.local
|
||||
.env.production.local
|
||||
.env.local
|
||||
|
||||
# parcel-bundler cache (https://parceljs.org/)
|
||||
.cache
|
||||
.parcel-cache
|
||||
|
||||
# Next.js build output
|
||||
.next
|
||||
out
|
||||
|
||||
# Nuxt.js build / generate output
|
||||
.nuxt
|
||||
dist
|
||||
|
||||
# Gatsby files
|
||||
.cache/
|
||||
# Comment in the public line in if your project uses Gatsby and not Next.js
|
||||
# https://nextjs.org/blog/next-9-1#public-directory-support
|
||||
# public
|
||||
|
||||
# vuepress build output
|
||||
.vuepress/dist
|
||||
|
||||
# vuepress v2.x temp and cache directory
|
||||
.temp
|
||||
|
||||
# Docusaurus cache and generated files
|
||||
.docusaurus
|
||||
|
||||
# Serverless directories
|
||||
.serverless/
|
||||
|
||||
# FuseBox cache
|
||||
.fusebox/
|
||||
|
||||
# DynamoDB Local files
|
||||
.dynamodb/
|
||||
|
||||
# TernJS port file
|
||||
.tern-port
|
||||
|
||||
# Stores VSCode versions used for testing VSCode extensions
|
||||
.vscode-test
|
||||
|
||||
# yarn v2
|
||||
.yarn/cache
|
||||
.yarn/unplugged
|
||||
.yarn/build-state.yml
|
||||
.yarn/install-state.gz
|
||||
.pnp.*
|
||||
|
||||
### Node Patch ###
|
||||
# Serverless Webpack directories
|
||||
.webpack/
|
||||
|
||||
# Optional stylelint cache
|
||||
|
||||
# SvelteKit build / generate output
|
||||
.svelte-kit
|
||||
|
||||
### VisualStudioCode ###
|
||||
.vscode/*
|
||||
!.vscode/settings.json
|
||||
!.vscode/tasks.json
|
||||
!.vscode/launch.json
|
||||
!.vscode/extensions.json
|
||||
!.vscode/*.code-snippets
|
||||
|
||||
# Local History for Visual Studio Code
|
||||
.history/
|
||||
|
||||
# Built Visual Studio Code Extensions
|
||||
*.vsix
|
||||
|
||||
### VisualStudioCode Patch ###
|
||||
# Ignore all local history of files
|
||||
.history
|
||||
.ionide
|
||||
|
||||
### Windows ###
|
||||
# Windows thumbnail cache files
|
||||
Thumbs.db
|
||||
Thumbs.db:encryptable
|
||||
ehthumbs.db
|
||||
ehthumbs_vista.db
|
||||
|
||||
# Dump file
|
||||
*.stackdump
|
||||
|
||||
# Folder config file
|
||||
[Dd]esktop.ini
|
||||
|
||||
# Recycle Bin used on file shares
|
||||
$RECYCLE.BIN/
|
||||
|
||||
# Windows Installer files
|
||||
*.cab
|
||||
*.msi
|
||||
*.msix
|
||||
*.msm
|
||||
*.msp
|
||||
|
||||
# Windows shortcuts
|
||||
*.lnk
|
||||
|
||||
# End of https://www.toptal.com/developers/gitignore/api/windows,visualstudiocode,node
|
||||
|
||||
# Custom rules (everything added below won't be overriden by 'Generate .gitignore File' if you use 'Update' option)
|
||||
|
||||
54
README.md
Normal file
54
README.md
Normal file
@@ -0,0 +1,54 @@
|
||||
# NFC 读卡器
|
||||
|
||||
一个基于 Electron 的 NFC 卡片读取应用程序,支持连接读卡器并读取卡片 ID。
|
||||
|
||||
## 功能特性
|
||||
|
||||
- 连接/断开 NFC 读卡器
|
||||
- 读取 NFC 卡片 ID
|
||||
- 实时状态显示
|
||||
- 简洁美观的用户界面
|
||||
|
||||
## 安装
|
||||
|
||||
```bash
|
||||
npm install
|
||||
```
|
||||
|
||||
## 运行
|
||||
|
||||
```bash
|
||||
npm start
|
||||
```
|
||||
|
||||
## 使用说明
|
||||
|
||||
1. 启动应用后,点击"连接读卡器"按钮连接读卡器
|
||||
2. 连接成功后,将 NFC 卡片放置在读卡器上
|
||||
3. 点击"读取卡片"按钮读取卡片 ID
|
||||
4. 读取结果将显示在结果区域中
|
||||
5. 使用完毕后,点击"断开连接"按钮断开读卡器
|
||||
|
||||
## 项目结构
|
||||
|
||||
```
|
||||
NFCReader/
|
||||
├── index.html # 主页面
|
||||
├── main.js # Electron 主进程
|
||||
├── renderer.js # 渲染进程逻辑
|
||||
├── style.css # 样式文件
|
||||
├── lib/
|
||||
│ └── cardReader.es.js # 读卡器核心库
|
||||
└── package.json # 项目配置
|
||||
```
|
||||
|
||||
## 技术栈
|
||||
|
||||
- Electron
|
||||
- JavaScript (ES6+)
|
||||
- HTML5
|
||||
- CSS3
|
||||
|
||||
## 许可证
|
||||
|
||||
MIT
|
||||
60
index.html
Normal file
60
index.html
Normal file
@@ -0,0 +1,60 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh-CN">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||
<title>NFC 读卡器</title>
|
||||
<link rel="stylesheet" href="style.css">
|
||||
</head>
|
||||
<body>
|
||||
<div class="titlebar">
|
||||
<div class="titlebar-drag-region"></div>
|
||||
<div class="titlebar-title">NFC 读卡器</div>
|
||||
<div class="window-controls">
|
||||
<button class="window-control minimize" id="minimizeBtn">
|
||||
<svg width="10" height="10" viewBox="0 0 10 10">
|
||||
<rect x="0" y="4" width="10" height="2" fill="currentColor"/>
|
||||
</svg>
|
||||
</button>
|
||||
<button class="window-control maximize" id="maximizeBtn">
|
||||
<svg width="10" height="10" viewBox="0 0 10 10">
|
||||
<rect x="0" y="0" width="10" height="10" fill="none" stroke="currentColor" stroke-width="1.5"/>
|
||||
</svg>
|
||||
</button>
|
||||
<button class="window-control close" id="closeBtn">
|
||||
<svg width="10" height="10" viewBox="0 0 10 10">
|
||||
<line x1="0" y1="0" x2="10" y2="10" stroke="currentColor" stroke-width="1.5"/>
|
||||
<line x1="10" y1="0" x2="0" y2="10" stroke="currentColor" stroke-width="1.5"/>
|
||||
</svg>
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
<div class="container">
|
||||
<div class="card">
|
||||
<div class="card-body">
|
||||
<div class="status-section">
|
||||
<div class="status-indicator">
|
||||
<span id="statusDot" class="status-dot disconnected"></span>
|
||||
<span id="statusText">未连接</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="control-buttons">
|
||||
<button id="connectBtn" class="btn btn-primary">连接读卡器</button>
|
||||
<button id="disconnectBtn" class="btn btn-secondary" disabled>断开连接</button>
|
||||
<button id="readBtn" class="btn btn-success" disabled>读取卡片</button>
|
||||
</div>
|
||||
|
||||
<div class="result-section">
|
||||
<h4>读取结果</h4>
|
||||
<div id="resultBox" class="result-box">
|
||||
<p class="placeholder-text">请先连接读卡器,然后点击"读取卡片"按钮</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script type="module" src="renderer.js"></script>
|
||||
</body>
|
||||
</html>
|
||||
120
lib/cardReader.es.js
Normal file
120
lib/cardReader.es.js
Normal file
@@ -0,0 +1,120 @@
|
||||
var v = {
|
||||
OBJ: function() {
|
||||
var r = {}, e = null, t = null;
|
||||
r.onResult = function(s) {
|
||||
t.addEvent("Result", s);
|
||||
};
|
||||
var o = function() {
|
||||
}, l = function(s) {
|
||||
var i = "";
|
||||
i = s.data;
|
||||
var w = {
|
||||
type: "Result",
|
||||
strStatus: i.substr(4, 2),
|
||||
strData: i.substr(6),
|
||||
strCmdCode: i.substr(0, 4)
|
||||
};
|
||||
t != null && t.fireEvent(w);
|
||||
}, d = function() {
|
||||
}, h = function() {
|
||||
alert("Server not running !");
|
||||
};
|
||||
return r.createSocket = function() {
|
||||
try {
|
||||
if ("WebSocket" in window)
|
||||
e = new WebSocket("ws://localhost:39002/RFID Reader Service");
|
||||
else if ("MozWebSocket" in window)
|
||||
e = new MozWebSocket("ws://localhost:39002/RFID Reader Service");
|
||||
else
|
||||
return alert("None"), !1;
|
||||
return e.onopen = o, e.onmessage = l, e.onclose = d, e.onerror = h, t = new u(), !0;
|
||||
} catch {
|
||||
return !1;
|
||||
}
|
||||
}, r.Disconnect = function() {
|
||||
e != null && e.close();
|
||||
}, r.send = function(s) {
|
||||
e.send(s);
|
||||
}, r;
|
||||
}
|
||||
};
|
||||
function u() {
|
||||
this.handlers = {};
|
||||
}
|
||||
u.prototype = {
|
||||
constructor: u,
|
||||
addEvent: function(r, e) {
|
||||
typeof this.handlers[r] > "u" && (this.handlers[r] = []), this.handlers[r].push(e);
|
||||
},
|
||||
fireEvent: function(r) {
|
||||
if (r.target || (r.target = this), this.handlers[r.type] instanceof Array)
|
||||
for (var e = this.handlers[r.type], t = 0; t < e.length; t++)
|
||||
e[t](r);
|
||||
},
|
||||
removeEvent: function(r, e) {
|
||||
if (this.handlers[r] instanceof Array) {
|
||||
for (var t = this.handlers[r], o = 0; o < t.length && t[o] != e; o++)
|
||||
;
|
||||
t.splice(o, 1);
|
||||
}
|
||||
}
|
||||
};
|
||||
try {
|
||||
var f = v.OBJ();
|
||||
} catch {
|
||||
}
|
||||
f.createSocket();
|
||||
let n = "00", c = !1;
|
||||
function a(r) {
|
||||
return new Promise((e, t) => {
|
||||
f.onResult(function(o) {
|
||||
o.strStatus === "00" ? e(o) : t(o.strStatus);
|
||||
}), f.send(r);
|
||||
});
|
||||
}
|
||||
async function y() {
|
||||
await a(n + "010702");
|
||||
}
|
||||
async function E() {
|
||||
await a(n + "010701");
|
||||
}
|
||||
async function S() {
|
||||
if (c)
|
||||
try {
|
||||
await a(n + "0009");
|
||||
} catch (r) {
|
||||
throw new Error(`关闭读卡器失败: EerorCode 0x${r}`);
|
||||
}
|
||||
try {
|
||||
await a(n + "000700"), await a(n + "010941"), await a(n + "010801"), y(), await a(n + "010610"), c = !0;
|
||||
} catch (r) {
|
||||
throw new Error(`连接读卡器失败: EerorCode 0x${r}`);
|
||||
}
|
||||
}
|
||||
async function R() {
|
||||
if (!c)
|
||||
throw new Error("读卡器未连接,请先连接读卡器!");
|
||||
try {
|
||||
await a(n + "100152");
|
||||
} catch {
|
||||
return null;
|
||||
}
|
||||
try {
|
||||
return (await a(n + "1002")).strData;
|
||||
} catch (r) {
|
||||
throw new Error(`读取ID出错: EerorCode 0x${r}`);
|
||||
}
|
||||
}
|
||||
async function g() {
|
||||
if (c)
|
||||
try {
|
||||
await a(n + "0009"), E(), await a(n + "010610"), c = !1;
|
||||
} catch (r) {
|
||||
throw new Error(`关闭读卡器失败: EerorCode 0x${r}`);
|
||||
}
|
||||
}
|
||||
export {
|
||||
g as closeReader,
|
||||
S as connectReader,
|
||||
R as readId
|
||||
};
|
||||
59
main.js
Normal file
59
main.js
Normal file
@@ -0,0 +1,59 @@
|
||||
const { app, BrowserWindow, ipcMain } = require('electron')
|
||||
|
||||
function createWindow() {
|
||||
const win = new BrowserWindow({
|
||||
width: 440,
|
||||
height: 400,
|
||||
frame: false,
|
||||
transparent: false,
|
||||
backgroundColor: '#f9f9f9',
|
||||
roundedCorners: true,
|
||||
webPreferences: {
|
||||
nodeIntegration: true,
|
||||
contextIsolation: false,
|
||||
},
|
||||
})
|
||||
|
||||
win.loadFile('index.html')
|
||||
}
|
||||
|
||||
app.whenReady().then(() => {
|
||||
createWindow()
|
||||
|
||||
app.on('activate', () => {
|
||||
if (BrowserWindow.getAllWindows().length === 0) {
|
||||
createWindow()
|
||||
}
|
||||
})
|
||||
})
|
||||
|
||||
app.on('window-all-closed', () => {
|
||||
if (process.platform !== 'darwin') {
|
||||
app.quit()
|
||||
}
|
||||
})
|
||||
|
||||
ipcMain.on('window-minimize', () => {
|
||||
const focusedWindow = BrowserWindow.getFocusedWindow()
|
||||
if (focusedWindow) {
|
||||
focusedWindow.minimize()
|
||||
}
|
||||
})
|
||||
|
||||
ipcMain.on('window-maximize', () => {
|
||||
const focusedWindow = BrowserWindow.getFocusedWindow()
|
||||
if (focusedWindow) {
|
||||
if (focusedWindow.isMaximized()) {
|
||||
focusedWindow.unmaximize()
|
||||
} else {
|
||||
focusedWindow.maximize()
|
||||
}
|
||||
}
|
||||
})
|
||||
|
||||
ipcMain.on('window-close', () => {
|
||||
const focusedWindow = BrowserWindow.getFocusedWindow()
|
||||
if (focusedWindow) {
|
||||
focusedWindow.close()
|
||||
}
|
||||
})
|
||||
15
package.json
Normal file
15
package.json
Normal file
@@ -0,0 +1,15 @@
|
||||
{
|
||||
"name": "nfc-reader",
|
||||
"version": "1.0.0",
|
||||
"description": "NFC Card Reader Application",
|
||||
"main": "main.js",
|
||||
"scripts": {
|
||||
"start": "electron ."
|
||||
},
|
||||
"keywords": ["nfc", "reader", "electron"],
|
||||
"author": "",
|
||||
"license": "MIT",
|
||||
"devDependencies": {
|
||||
"electron": "^28.0.0"
|
||||
}
|
||||
}
|
||||
106
renderer.js
Normal file
106
renderer.js
Normal file
@@ -0,0 +1,106 @@
|
||||
import { connectReader, readId, closeReader } from './lib/cardReader.es.js';
|
||||
const { ipcRenderer } = require('electron');
|
||||
|
||||
const connectBtn = document.getElementById('connectBtn');
|
||||
const disconnectBtn = document.getElementById('disconnectBtn');
|
||||
const readBtn = document.getElementById('readBtn');
|
||||
const statusDot = document.getElementById('statusDot');
|
||||
const statusText = document.getElementById('statusText');
|
||||
const resultBox = document.getElementById('resultBox');
|
||||
|
||||
const minimizeBtn = document.getElementById('minimizeBtn');
|
||||
const maximizeBtn = document.getElementById('maximizeBtn');
|
||||
const closeBtn = document.getElementById('closeBtn');
|
||||
|
||||
let isConnected = false;
|
||||
|
||||
function updateStatus(status) {
|
||||
statusDot.className = 'status-dot ' + status;
|
||||
|
||||
switch(status) {
|
||||
case 'connected':
|
||||
statusText.textContent = '已连接';
|
||||
isConnected = true;
|
||||
connectBtn.disabled = true;
|
||||
disconnectBtn.disabled = false;
|
||||
readBtn.disabled = false;
|
||||
break;
|
||||
case 'disconnected':
|
||||
statusText.textContent = '未连接';
|
||||
isConnected = false;
|
||||
connectBtn.disabled = false;
|
||||
disconnectBtn.disabled = true;
|
||||
readBtn.disabled = true;
|
||||
break;
|
||||
case 'connecting':
|
||||
statusText.textContent = '连接中...';
|
||||
connectBtn.disabled = true;
|
||||
disconnectBtn.disabled = true;
|
||||
readBtn.disabled = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
function showResult(content, isError = false) {
|
||||
if (isError) {
|
||||
resultBox.innerHTML = '<p class="error-message">' + content + '</p>';
|
||||
} else {
|
||||
resultBox.innerHTML = '<p class="result-content">卡片 ID: ' + content + '</p>';
|
||||
}
|
||||
}
|
||||
|
||||
connectBtn.addEventListener('click', async () => {
|
||||
updateStatus('connecting');
|
||||
resultBox.innerHTML = '<p class="placeholder-text">正在连接读卡器...</p>';
|
||||
|
||||
try {
|
||||
await connectReader();
|
||||
updateStatus('connected');
|
||||
showResult('读卡器连接成功!请将卡片放在读卡器上,然后点击"读取卡片"按钮');
|
||||
} catch (error) {
|
||||
updateStatus('disconnected');
|
||||
showResult('连接失败: ' + error.message, true);
|
||||
}
|
||||
});
|
||||
|
||||
disconnectBtn.addEventListener('click', async () => {
|
||||
try {
|
||||
await closeReader();
|
||||
updateStatus('disconnected');
|
||||
showResult('读卡器已断开连接');
|
||||
} catch (error) {
|
||||
showResult('断开连接失败: ' + error.message, true);
|
||||
}
|
||||
});
|
||||
|
||||
readBtn.addEventListener('click', async () => {
|
||||
if (!isConnected) {
|
||||
showResult('请先连接读卡器', true);
|
||||
return;
|
||||
}
|
||||
|
||||
resultBox.innerHTML = '<p class="placeholder-text">正在读取卡片...</p>';
|
||||
|
||||
try {
|
||||
const cardId = await readId();
|
||||
if (cardId) {
|
||||
showResult(cardId);
|
||||
} else {
|
||||
showResult('未检测到卡片,请确保卡片已放置在读卡器上', true);
|
||||
}
|
||||
} catch (error) {
|
||||
showResult('读取失败: ' + error.message, true);
|
||||
}
|
||||
});
|
||||
|
||||
minimizeBtn.addEventListener('click', () => {
|
||||
ipcRenderer.send('window-minimize');
|
||||
});
|
||||
|
||||
maximizeBtn.addEventListener('click', () => {
|
||||
ipcRenderer.send('window-maximize');
|
||||
});
|
||||
|
||||
closeBtn.addEventListener('click', () => {
|
||||
ipcRenderer.send('window-close');
|
||||
});
|
||||
294
style.css
Normal file
294
style.css
Normal file
@@ -0,0 +1,294 @@
|
||||
@import url('https://fonts.googleapis.com/css2?family=Segoe+UI+Variable:wght@300;400;500;600;700&display=swap');
|
||||
|
||||
:root {
|
||||
--window-bg: #f9f9f9;
|
||||
--border-color: rgba(0, 0, 0, 0.08);
|
||||
--border-hover: rgba(0, 0, 0, 0.12);
|
||||
--text-primary: #1a1a1a;
|
||||
--text-secondary: #5c5c5c;
|
||||
--text-disabled: rgba(0, 0, 0, 0.4);
|
||||
--accent-color: #0067C0;
|
||||
--accent-hover: #005A9E;
|
||||
--success-color: #107C10;
|
||||
--danger-color: #E81123;
|
||||
--surface-bg: rgba(255, 255, 255, 0.6);
|
||||
--surface-hover: rgba(255, 255, 255, 0.9);
|
||||
}
|
||||
|
||||
* {
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
body {
|
||||
font-family: 'Segoe UI Variable', 'Segoe UI', -apple-system, BlinkMacSystemFont, sans-serif;
|
||||
background: var(--window-bg);
|
||||
min-height: 100vh;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
overflow: hidden;
|
||||
-webkit-font-smoothing: antialiased;
|
||||
}
|
||||
|
||||
.titlebar {
|
||||
height: 32px;
|
||||
background: var(--window-bg);
|
||||
display: flex;
|
||||
align-items: center;
|
||||
position: relative;
|
||||
-webkit-app-region: drag;
|
||||
border-bottom: 1px solid var(--border-color);
|
||||
}
|
||||
|
||||
.titlebar-drag-region {
|
||||
position: absolute;
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 138px;
|
||||
height: 32px;
|
||||
-webkit-app-region: drag;
|
||||
}
|
||||
|
||||
.titlebar-title {
|
||||
flex: 1;
|
||||
padding-left: 16px;
|
||||
font-size: 12px;
|
||||
font-weight: 400;
|
||||
color: var(--text-primary);
|
||||
user-select: none;
|
||||
pointer-events: none;
|
||||
opacity: 0.9;
|
||||
}
|
||||
|
||||
.window-controls {
|
||||
display: flex;
|
||||
height: 32px;
|
||||
position: absolute;
|
||||
right: 0;
|
||||
top: 0;
|
||||
}
|
||||
|
||||
.window-control {
|
||||
width: 46px;
|
||||
height: 32px;
|
||||
border: none;
|
||||
background: transparent;
|
||||
cursor: pointer;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
color: var(--text-primary);
|
||||
transition: background-color 0.1s ease;
|
||||
-webkit-app-region: no-drag;
|
||||
opacity: 0.9;
|
||||
}
|
||||
|
||||
.window-control:hover {
|
||||
background-color: rgba(0, 0, 0, 0.04);
|
||||
opacity: 1;
|
||||
}
|
||||
|
||||
.window-control:active {
|
||||
background-color: rgba(0, 0, 0, 0.08);
|
||||
}
|
||||
|
||||
.window-control.close:hover {
|
||||
background-color: var(--danger-color);
|
||||
color: white;
|
||||
}
|
||||
|
||||
.window-control svg {
|
||||
pointer-events: none;
|
||||
}
|
||||
|
||||
.container {
|
||||
flex: 1;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
padding: 16px;
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
.card {
|
||||
background: rgba(255, 255, 255, 0.6);
|
||||
border: 1px solid var(--border-color);
|
||||
box-shadow:
|
||||
0 2px 8px rgba(0, 0, 0, 0.04),
|
||||
0 0 0 1px rgba(255, 255, 255, 0.8) inset;
|
||||
border-radius: 8px;
|
||||
overflow: hidden;
|
||||
width: 100%;
|
||||
max-width: 400px;
|
||||
}
|
||||
|
||||
.card-body {
|
||||
padding: 20px;
|
||||
}
|
||||
|
||||
.status-section {
|
||||
text-align: center;
|
||||
padding: 12px 16px;
|
||||
background: rgba(255, 255, 255, 0.5);
|
||||
border-radius: 6px;
|
||||
border: 1px solid var(--border-color);
|
||||
margin-bottom: 16px;
|
||||
}
|
||||
|
||||
.status-indicator {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
gap: 8px;
|
||||
}
|
||||
|
||||
.status-dot {
|
||||
width: 8px;
|
||||
height: 8px;
|
||||
border-radius: 50%;
|
||||
display: inline-block;
|
||||
}
|
||||
|
||||
.status-dot.connected {
|
||||
background-color: var(--success-color);
|
||||
box-shadow: 0 0 0 2px rgba(16, 124, 16, 0.2);
|
||||
}
|
||||
|
||||
.status-dot.disconnected {
|
||||
background-color: var(--danger-color);
|
||||
box-shadow: 0 0 0 2px rgba(232, 17, 35, 0.2);
|
||||
}
|
||||
|
||||
.status-dot.connecting {
|
||||
background-color: #FF8C00;
|
||||
box-shadow: 0 0 0 2px rgba(255, 140, 0, 0.2);
|
||||
animation: blink 1s infinite;
|
||||
}
|
||||
|
||||
@keyframes blink {
|
||||
0%, 100% { opacity: 1; }
|
||||
50% { opacity: 0.4; }
|
||||
}
|
||||
|
||||
#statusText {
|
||||
font-size: 13px;
|
||||
font-weight: 500;
|
||||
color: var(--text-primary);
|
||||
}
|
||||
|
||||
.control-buttons {
|
||||
display: flex;
|
||||
gap: 8px;
|
||||
justify-content: center;
|
||||
margin-bottom: 16px;
|
||||
}
|
||||
|
||||
.control-buttons button {
|
||||
flex: 1;
|
||||
font-weight: 400;
|
||||
font-size: 13px;
|
||||
padding: 8px 14px;
|
||||
border: 1px solid var(--border-color);
|
||||
border-radius: 4px;
|
||||
transition: all 0.1s ease;
|
||||
cursor: pointer;
|
||||
font-family: inherit;
|
||||
letter-spacing: 0.1px;
|
||||
background: var(--surface-bg);
|
||||
color: var(--text-primary);
|
||||
}
|
||||
|
||||
.control-buttons button:hover:not(:disabled) {
|
||||
background: var(--surface-hover);
|
||||
border-color: var(--border-hover);
|
||||
}
|
||||
|
||||
.control-buttons button:active:not(:disabled) {
|
||||
background: rgba(0, 0, 0, 0.04);
|
||||
transform: scale(0.98);
|
||||
}
|
||||
|
||||
.control-buttons button:disabled {
|
||||
opacity: 0.4;
|
||||
cursor: default;
|
||||
}
|
||||
|
||||
.btn-primary {
|
||||
background: var(--accent-color);
|
||||
color: white;
|
||||
border-color: var(--accent-color);
|
||||
}
|
||||
|
||||
.btn-primary:hover:not(:disabled) {
|
||||
background: var(--accent-hover);
|
||||
border-color: var(--accent-hover);
|
||||
}
|
||||
|
||||
.btn-secondary {
|
||||
background: var(--surface-bg);
|
||||
color: var(--text-primary);
|
||||
border-color: var(--border-color);
|
||||
}
|
||||
|
||||
.btn-secondary:hover:not(:disabled) {
|
||||
background: var(--surface-hover);
|
||||
border-color: var(--border-hover);
|
||||
}
|
||||
|
||||
.btn-success {
|
||||
background: var(--success-color);
|
||||
color: white;
|
||||
border-color: var(--success-color);
|
||||
}
|
||||
|
||||
.btn-success:hover:not(:disabled) {
|
||||
background: #0B5A0B;
|
||||
border-color: #0B5A0B;
|
||||
}
|
||||
|
||||
.result-section {
|
||||
margin-top: 16px;
|
||||
}
|
||||
|
||||
.result-section h4 {
|
||||
margin-bottom: 8px;
|
||||
color: var(--text-primary);
|
||||
font-weight: 600;
|
||||
font-size: 12px;
|
||||
text-transform: uppercase;
|
||||
letter-spacing: 0.5px;
|
||||
}
|
||||
|
||||
.result-box {
|
||||
padding: 12px 16px;
|
||||
background: rgba(255, 255, 255, 0.5);
|
||||
border-radius: 6px;
|
||||
min-height: 60px;
|
||||
border: 1px solid var(--border-color);
|
||||
}
|
||||
|
||||
.placeholder-text {
|
||||
color: var(--text-secondary);
|
||||
margin: 0;
|
||||
text-align: center;
|
||||
font-size: 13px;
|
||||
line-height: 1.5;
|
||||
}
|
||||
|
||||
.result-content {
|
||||
word-break: break-all;
|
||||
font-family: 'Segoe UI Variable', 'Segoe UI', 'Consolas', monospace;
|
||||
font-size: 14px;
|
||||
color: var(--text-primary);
|
||||
font-weight: 400;
|
||||
text-align: center;
|
||||
padding: 4px;
|
||||
}
|
||||
|
||||
.error-message {
|
||||
color: var(--danger-color);
|
||||
font-weight: 500;
|
||||
text-align: center;
|
||||
font-size: 13px;
|
||||
}
|
||||
Reference in New Issue
Block a user