klpgit 0.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md ADDED
@@ -0,0 +1,24 @@
1
+ # KlpGit
2
+
3
+ Git GUI that opens right from your terminal. Diff viewer, file tree, commit & push in one click.
4
+
5
+ ## Install
6
+
7
+ ```bash
8
+ npm install -g klpgit
9
+ ```
10
+
11
+ ## Usage
12
+
13
+ ```bash
14
+ cd your-project
15
+ klpgit
16
+ ```
17
+
18
+ A window opens with your project. Select files, view changes or source code, run code checks, commit and push.
19
+
20
+ Supports Russian and English — choose on first launch.
21
+
22
+ ## License
23
+
24
+ MIT
package/dist/cli.d.ts ADDED
@@ -0,0 +1,2 @@
1
+ #!/usr/bin/env node
2
+ export {};
package/dist/cli.js ADDED
@@ -0,0 +1,46 @@
1
+ #!/usr/bin/env node
2
+ import { startServer } from './server.js';
3
+ import { exec } from 'child_process';
4
+ import open from 'open';
5
+ const cwd = process.cwd();
6
+ const port = parseInt(process.env.KLPGIT_PORT || '4219', 10);
7
+ const url = `http://localhost:${port}`;
8
+ process.stdout.write('\x1Bc');
9
+ console.log('');
10
+ console.log(' \x1b[35m\x1b[1mKlpGit\x1b[0m \x1b[90mv0.2.0\x1b[0m');
11
+ console.log('');
12
+ console.log(` \x1b[36m→\x1b[0m ${url}`);
13
+ console.log('');
14
+ const { server } = startServer(cwd, port);
15
+ function openAppWindow() {
16
+ const w = 1280, h = 840;
17
+ const flags = `--app=${url} --window-size=${w},${h} --disable-extensions --new-window`;
18
+ const isWin = process.platform === 'win32';
19
+ const isMac = process.platform === 'darwin';
20
+ if (isWin) {
21
+ exec(`start msedge ${flags}`, (err) => {
22
+ if (err)
23
+ exec(`start chrome ${flags}`, (err2) => {
24
+ if (err2)
25
+ open(url);
26
+ });
27
+ });
28
+ }
29
+ else if (isMac) {
30
+ exec(`open -na "Google Chrome" --args ${flags}`, (err) => {
31
+ if (err)
32
+ exec(`open -na "Microsoft Edge" --args ${flags}`, (err2) => {
33
+ if (err2)
34
+ open(url);
35
+ });
36
+ });
37
+ }
38
+ else {
39
+ exec(`google-chrome ${flags} 2>/dev/null || chromium-browser ${flags} 2>/dev/null || microsoft-edge ${flags} 2>/dev/null`, (err) => {
40
+ if (err)
41
+ open(url);
42
+ });
43
+ }
44
+ }
45
+ server.on('listening', openAppWindow);
46
+ //# sourceMappingURL=cli.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"cli.js","sourceRoot":"","sources":["../src/cli.ts"],"names":[],"mappings":";AACA,OAAO,EAAE,WAAW,EAAE,MAAM,aAAa,CAAC;AAC1C,OAAO,EAAE,IAAI,EAAE,MAAM,eAAe,CAAC;AACrC,OAAO,IAAI,MAAM,MAAM,CAAC;AAExB,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,EAAE,CAAC;AAC1B,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,WAAW,IAAI,MAAM,EAAE,EAAE,CAAC,CAAC;AAC7D,MAAM,GAAG,GAAG,oBAAoB,IAAI,EAAE,CAAC;AAEvC,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;AAE9B,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;AAChB,OAAO,CAAC,GAAG,CAAC,sDAAsD,CAAC,CAAC;AACpE,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;AAChB,OAAO,CAAC,GAAG,CAAC,sBAAsB,GAAG,EAAE,CAAC,CAAC;AACzC,OAAO,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;AAEhB,MAAM,EAAE,MAAM,EAAE,GAAG,WAAW,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;AAE1C,SAAS,aAAa;IACpB,MAAM,CAAC,GAAG,IAAI,EAAE,CAAC,GAAG,GAAG,CAAC;IACxB,MAAM,KAAK,GAAG,SAAS,GAAG,kBAAkB,CAAC,IAAI,CAAC,oCAAoC,CAAC;IACvF,MAAM,KAAK,GAAG,OAAO,CAAC,QAAQ,KAAK,OAAO,CAAC;IAC3C,MAAM,KAAK,GAAG,OAAO,CAAC,QAAQ,KAAK,QAAQ,CAAC;IAE5C,IAAI,KAAK,EAAE,CAAC;QACV,IAAI,CAAC,gBAAgB,KAAK,EAAE,EAAE,CAAC,GAAG,EAAE,EAAE;YACpC,IAAI,GAAG;gBAAE,IAAI,CAAC,gBAAgB,KAAK,EAAE,EAAE,CAAC,IAAI,EAAE,EAAE;oBAC9C,IAAI,IAAI;wBAAE,IAAI,CAAC,GAAG,CAAC,CAAC;gBACtB,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;SAAM,IAAI,KAAK,EAAE,CAAC;QACjB,IAAI,CAAC,mCAAmC,KAAK,EAAE,EAAE,CAAC,GAAG,EAAE,EAAE;YACvD,IAAI,GAAG;gBAAE,IAAI,CAAC,oCAAoC,KAAK,EAAE,EAAE,CAAC,IAAI,EAAE,EAAE;oBAClE,IAAI,IAAI;wBAAE,IAAI,CAAC,GAAG,CAAC,CAAC;gBACtB,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;SAAM,CAAC;QACN,IAAI,CAAC,iBAAiB,KAAK,oCAAoC,KAAK,kCAAkC,KAAK,cAAc,EAAE,CAAC,GAAG,EAAE,EAAE;YACjI,IAAI,GAAG;gBAAE,IAAI,CAAC,GAAG,CAAC,CAAC;QACrB,CAAC,CAAC,CAAC;IACL,CAAC;AACH,CAAC;AAED,MAAM,CAAC,EAAE,CAAC,WAAW,EAAE,aAAa,CAAC,CAAC"}
@@ -0,0 +1,5 @@
1
+ export declare function startServer(cwd: string, port: number): {
2
+ server: import("http").Server<typeof import("http").IncomingMessage, typeof import("http").ServerResponse>;
3
+ watcher: import("chokidar").FSWatcher;
4
+ port: number;
5
+ };
package/dist/server.js ADDED
@@ -0,0 +1,170 @@
1
+ import express from 'express';
2
+ import { createServer } from 'http';
3
+ import { WebSocketServer, WebSocket } from 'ws';
4
+ import { watch } from 'chokidar';
5
+ import { join, dirname, resolve } from 'path';
6
+ import { readFileSync, statSync } from 'fs';
7
+ import { fileURLToPath } from 'url';
8
+ import { getFullStatus, getDiff, commitAndPush, addFiles, initRepo, removeRemote, hasGitRepo } from './services/git.js';
9
+ import { getFileTree } from './services/fileTree.js';
10
+ const __filename = fileURLToPath(import.meta.url);
11
+ const __dirname = dirname(__filename);
12
+ export function startServer(cwd, port) {
13
+ const app = express();
14
+ app.use(express.json());
15
+ const webDir = join(__dirname, '..', 'web');
16
+ app.use(express.static(webDir));
17
+ app.get('/api/status', async (_req, res) => {
18
+ try {
19
+ const status = await getFullStatus(cwd);
20
+ res.json(status);
21
+ }
22
+ catch (err) {
23
+ res.status(500).json({ error: String(err) });
24
+ }
25
+ });
26
+ app.get('/api/tree', async (_req, res) => {
27
+ try {
28
+ const tree = getFileTree(cwd);
29
+ res.json(tree);
30
+ }
31
+ catch (err) {
32
+ res.status(500).json({ error: String(err) });
33
+ }
34
+ });
35
+ app.get('/api/diff', async (req, res) => {
36
+ try {
37
+ const file = req.query.file;
38
+ if (!file) {
39
+ res.json({ diff: '' });
40
+ return;
41
+ }
42
+ const diff = await getDiff(cwd, file);
43
+ res.json({ diff });
44
+ }
45
+ catch (err) {
46
+ res.status(500).json({ error: String(err) });
47
+ }
48
+ });
49
+ app.post('/api/add', async (req, res) => {
50
+ try {
51
+ const { files } = req.body;
52
+ await addFiles(cwd, files || ['.']);
53
+ const status = await getFullStatus(cwd);
54
+ res.json({ ok: true, status });
55
+ }
56
+ catch (err) {
57
+ res.status(500).json({ error: String(err) });
58
+ }
59
+ });
60
+ app.post('/api/submit', async (req, res) => {
61
+ try {
62
+ const { message, files } = req.body;
63
+ if (!message) {
64
+ res.status(400).json({ error: 'Commit message required' });
65
+ return;
66
+ }
67
+ const result = await commitAndPush(cwd, message, files || ['.']);
68
+ res.json({ ok: true, ...result });
69
+ }
70
+ catch (err) {
71
+ res.status(500).json({ error: String(err) });
72
+ }
73
+ });
74
+ app.post('/api/init', async (req, res) => {
75
+ try {
76
+ const { remoteUrl } = req.body;
77
+ await initRepo(cwd, remoteUrl);
78
+ res.json({ ok: true });
79
+ }
80
+ catch (err) {
81
+ res.status(500).json({ error: String(err) });
82
+ }
83
+ });
84
+ app.post('/api/disconnect', async (_req, res) => {
85
+ try {
86
+ await removeRemote(cwd);
87
+ res.json({ ok: true });
88
+ }
89
+ catch (err) {
90
+ res.status(500).json({ error: String(err) });
91
+ }
92
+ });
93
+ app.get('/api/file', (req, res) => {
94
+ try {
95
+ const filePath = req.query.path;
96
+ if (!filePath) {
97
+ res.status(400).json({ error: 'Path required' });
98
+ return;
99
+ }
100
+ const fullPath = join(cwd, filePath);
101
+ if (!resolve(fullPath).startsWith(resolve(cwd))) {
102
+ res.status(403).json({ error: 'Access denied' });
103
+ return;
104
+ }
105
+ const stats = statSync(fullPath);
106
+ if (stats.size > 1024 * 1024) {
107
+ res.json({ content: '', language: '' });
108
+ return;
109
+ }
110
+ const content = readFileSync(fullPath, 'utf-8');
111
+ const ext = filePath.split('.').pop()?.toLowerCase() || '';
112
+ res.json({ content, language: ext });
113
+ }
114
+ catch (err) {
115
+ res.status(500).json({ error: String(err) });
116
+ }
117
+ });
118
+ app.get('/api/info', (_req, res) => {
119
+ res.json({
120
+ cwd,
121
+ hasGit: hasGitRepo(cwd),
122
+ name: cwd.split(/[\\/]/).pop() || cwd,
123
+ });
124
+ });
125
+ const server = createServer(app);
126
+ const wss = new WebSocketServer({ server, path: '/ws' });
127
+ const clients = new Set();
128
+ wss.on('connection', (ws) => {
129
+ clients.add(ws);
130
+ ws.on('close', () => clients.delete(ws));
131
+ });
132
+ function broadcast(event, data) {
133
+ const msg = JSON.stringify({ event, data });
134
+ for (const ws of clients) {
135
+ if (ws.readyState === WebSocket.OPEN) {
136
+ ws.send(msg);
137
+ }
138
+ }
139
+ }
140
+ const watcher = watch(cwd, {
141
+ ignored: [
142
+ /(^|[\/\\])\./,
143
+ '**/node_modules/**',
144
+ '**/dist/**',
145
+ '**/build/**',
146
+ '**/.klpgit/**',
147
+ ],
148
+ persistent: true,
149
+ ignoreInitial: true,
150
+ depth: 5,
151
+ });
152
+ let debounceTimer = null;
153
+ const notifyChange = () => {
154
+ if (debounceTimer)
155
+ clearTimeout(debounceTimer);
156
+ debounceTimer = setTimeout(async () => {
157
+ try {
158
+ const status = await getFullStatus(cwd);
159
+ broadcast('status', status);
160
+ }
161
+ catch { }
162
+ }, 300);
163
+ };
164
+ watcher.on('add', notifyChange);
165
+ watcher.on('change', notifyChange);
166
+ watcher.on('unlink', notifyChange);
167
+ server.listen(port, () => { });
168
+ return { server, watcher, port };
169
+ }
170
+ //# sourceMappingURL=server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"server.js","sourceRoot":"","sources":["../src/server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,7 @@
1
+ export interface TreeEntry {
2
+ name: string;
3
+ path: string;
4
+ type: 'file' | 'dir';
5
+ children?: TreeEntry[];
6
+ }
7
+ export declare function getFileTree(cwd: string): TreeEntry[];
@@ -0,0 +1,76 @@
1
+ import { readdirSync, statSync, readFileSync, existsSync } from 'fs';
2
+ import { join } from 'path';
3
+ const DEFAULT_IGNORE = new Set([
4
+ 'node_modules', '.git', '.klpgit', 'dist', 'build',
5
+ '.next', '.nuxt', 'coverage', '.cache', '__pycache__',
6
+ '.venv', 'venv', '.env', '.idea', '.vscode',
7
+ ]);
8
+ function shouldIgnore(name, gitignorePatterns) {
9
+ if (DEFAULT_IGNORE.has(name))
10
+ return true;
11
+ if (name.startsWith('.') && name !== '.gitignore')
12
+ return true;
13
+ for (const p of gitignorePatterns) {
14
+ if (name === p || name.match(new RegExp('^' + p.replace(/\*/g, '.*') + '$')))
15
+ return true;
16
+ }
17
+ return false;
18
+ }
19
+ function readDir(basePath, relPath, gitignorePatterns) {
20
+ const fullPath = join(basePath, relPath);
21
+ let entries;
22
+ try {
23
+ entries = readdirSync(fullPath);
24
+ }
25
+ catch {
26
+ return [];
27
+ }
28
+ const result = [];
29
+ const sorted = entries
30
+ .filter(e => !shouldIgnore(e, gitignorePatterns))
31
+ .sort((a, b) => {
32
+ const aIsDir = statSync(join(fullPath, a)).isDirectory();
33
+ const bIsDir = statSync(join(fullPath, b)).isDirectory();
34
+ if (aIsDir && !bIsDir)
35
+ return -1;
36
+ if (!aIsDir && bIsDir)
37
+ return 1;
38
+ return a.localeCompare(b);
39
+ });
40
+ for (const entry of sorted) {
41
+ const rel = relPath ? `${relPath}/${entry}` : entry;
42
+ const full = join(basePath, rel);
43
+ let isDir;
44
+ try {
45
+ isDir = statSync(full).isDirectory();
46
+ }
47
+ catch {
48
+ continue;
49
+ }
50
+ const node = {
51
+ name: entry,
52
+ path: rel,
53
+ type: isDir ? 'dir' : 'file',
54
+ };
55
+ if (isDir) {
56
+ node.children = readDir(basePath, rel, gitignorePatterns);
57
+ }
58
+ result.push(node);
59
+ }
60
+ return result;
61
+ }
62
+ export function getFileTree(cwd) {
63
+ let gitignorePatterns = [];
64
+ const gitignorePath = join(cwd, '.gitignore');
65
+ if (existsSync(gitignorePath)) {
66
+ try {
67
+ gitignorePatterns = readFileSync(gitignorePath, 'utf-8')
68
+ .split('\n')
69
+ .map(l => l.trim())
70
+ .filter(l => l && !l.startsWith('#'));
71
+ }
72
+ catch { }
73
+ }
74
+ return readDir(cwd, '', gitignorePatterns);
75
+ }
76
+ //# sourceMappingURL=fileTree.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"fileTree.js","sourceRoot":"","sources":["../../src/services/fileTree.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,24 @@
1
+ import { type SimpleGit } from 'simple-git';
2
+ export declare function getGit(cwd: string): SimpleGit;
3
+ export declare function hasGitRepo(cwd: string): boolean;
4
+ export declare function getFullStatus(cwd: string): Promise<{
5
+ branch: string;
6
+ remoteUrl: string;
7
+ staged: number;
8
+ modified: number;
9
+ total: number;
10
+ files: {
11
+ path: string;
12
+ status: string;
13
+ }[];
14
+ ahead: number;
15
+ behind: number;
16
+ }>;
17
+ export declare function getDiff(cwd: string, file: string): Promise<string>;
18
+ export declare function addFiles(cwd: string, files: string[]): Promise<void>;
19
+ export declare function commitAndPush(cwd: string, message: string, filesToAdd: string[]): Promise<{
20
+ hash: string;
21
+ branch: string;
22
+ }>;
23
+ export declare function initRepo(cwd: string, remoteUrl?: string): Promise<void>;
24
+ export declare function removeRemote(cwd: string): Promise<void>;
@@ -0,0 +1,125 @@
1
+ import simpleGit from 'simple-git';
2
+ import { existsSync } from 'fs';
3
+ import { join } from 'path';
4
+ export function getGit(cwd) {
5
+ return simpleGit({ baseDir: cwd });
6
+ }
7
+ export function hasGitRepo(cwd) {
8
+ return existsSync(join(cwd, '.git'));
9
+ }
10
+ export async function getFullStatus(cwd) {
11
+ const git = getGit(cwd);
12
+ const status = await git.status();
13
+ const branch = status.current || 'main';
14
+ let remoteUrl = '';
15
+ try {
16
+ const remotes = await git.getRemotes(true);
17
+ const origin = remotes.find((r) => r.name === 'origin');
18
+ remoteUrl = origin?.refs?.fetch || origin?.refs?.push || '';
19
+ }
20
+ catch { }
21
+ const files = [];
22
+ const seen = new Set();
23
+ for (const f of status.staged) {
24
+ if (!seen.has(f)) {
25
+ files.push({ path: f, status: 'staged' });
26
+ seen.add(f);
27
+ }
28
+ }
29
+ for (const f of status.modified) {
30
+ if (!seen.has(f)) {
31
+ files.push({ path: f, status: 'modified' });
32
+ seen.add(f);
33
+ }
34
+ }
35
+ for (const f of status.not_added) {
36
+ if (!seen.has(f)) {
37
+ files.push({ path: f, status: 'untracked' });
38
+ seen.add(f);
39
+ }
40
+ }
41
+ for (const f of status.created) {
42
+ if (!seen.has(f)) {
43
+ files.push({ path: f, status: 'untracked' });
44
+ seen.add(f);
45
+ }
46
+ }
47
+ for (const f of status.deleted) {
48
+ if (!seen.has(f)) {
49
+ files.push({ path: f, status: 'deleted' });
50
+ seen.add(f);
51
+ }
52
+ }
53
+ return {
54
+ branch,
55
+ remoteUrl,
56
+ staged: status.staged.length,
57
+ modified: files.filter(f => f.status !== 'staged').length,
58
+ total: files.length,
59
+ files,
60
+ ahead: status.ahead,
61
+ behind: status.behind,
62
+ };
63
+ }
64
+ export async function getDiff(cwd, file) {
65
+ const git = getGit(cwd);
66
+ let unstaged = '';
67
+ let staged = '';
68
+ try {
69
+ unstaged = await git.diff(['--', file]);
70
+ }
71
+ catch { }
72
+ try {
73
+ staged = await git.diff(['--staged', '--', file]);
74
+ }
75
+ catch { }
76
+ return staged || unstaged || '';
77
+ }
78
+ export async function addFiles(cwd, files) {
79
+ const git = getGit(cwd);
80
+ if (files.includes('.')) {
81
+ await git.add('.');
82
+ }
83
+ else {
84
+ await git.add(files);
85
+ }
86
+ }
87
+ export async function commitAndPush(cwd, message, filesToAdd) {
88
+ const git = getGit(cwd);
89
+ if (filesToAdd.length > 0) {
90
+ if (filesToAdd.includes('.')) {
91
+ await git.add('.');
92
+ }
93
+ else {
94
+ await git.add(filesToAdd);
95
+ }
96
+ }
97
+ const result = await git.commit(message);
98
+ const hash = result.commit || '';
99
+ const status = await git.status();
100
+ const branch = status.current || 'main';
101
+ try {
102
+ await git.push('origin', branch);
103
+ }
104
+ catch {
105
+ await git.push(['-u', 'origin', branch]);
106
+ }
107
+ return { hash, branch };
108
+ }
109
+ export async function initRepo(cwd, remoteUrl) {
110
+ const git = getGit(cwd);
111
+ if (!hasGitRepo(cwd)) {
112
+ await git.init();
113
+ }
114
+ if (remoteUrl) {
115
+ try {
116
+ await git.addRemote('origin', remoteUrl);
117
+ }
118
+ catch { }
119
+ }
120
+ }
121
+ export async function removeRemote(cwd) {
122
+ const git = getGit(cwd);
123
+ await git.removeRemote('origin');
124
+ }
125
+ //# sourceMappingURL=git.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"git.js","sourceRoot":"","sources":["../../src/services/git.ts"],"names":[],"mappings":"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"}
package/package.json ADDED
@@ -0,0 +1,54 @@
1
+ {
2
+ "name": "klpgit",
3
+ "version": "0.2.0",
4
+ "description": "Beautiful web-based Git GUI — run klpgit, get a local dashboard in your browser",
5
+ "main": "dist/cli.js",
6
+ "type": "module",
7
+ "bin": {
8
+ "klpgit": "dist/cli.js"
9
+ },
10
+ "scripts": {
11
+ "build": "tsc",
12
+ "lint": "eslint src --ext .ts",
13
+ "build:frontend": "cd frontend && npm install && npm run build",
14
+ "build:all": "npm run build:frontend && npm run build",
15
+ "start": "node dist/cli.js",
16
+ "dev": "npm run build && node dist/cli.js",
17
+ "prepublishOnly": "npm run build:all"
18
+ },
19
+ "keywords": [
20
+ "git",
21
+ "gui",
22
+ "web",
23
+ "cli",
24
+ "dashboard"
25
+ ],
26
+ "author": "",
27
+ "license": "MIT",
28
+ "engines": {
29
+ "node": ">=18.0.0"
30
+ },
31
+ "dependencies": {
32
+ "chokidar": "^3.5.3",
33
+ "express": "^4.18.2",
34
+ "open": "^10.1.0",
35
+ "simple-git": "^3.22.0",
36
+ "ws": "^8.16.0"
37
+ },
38
+ "devDependencies": {
39
+ "@eslint/js": "^10.0.1",
40
+ "@types/express": "^4.17.21",
41
+ "@types/node": "^20.10.0",
42
+ "@types/ws": "^8.5.10",
43
+ "@typescript-eslint/eslint-plugin": "^8.56.0",
44
+ "@typescript-eslint/parser": "^8.56.0",
45
+ "eslint": "^10.0.1",
46
+ "eslint-import-resolver-typescript": "^4.4.4",
47
+ "globals": "^17.3.0",
48
+ "typescript": "^5.3.0"
49
+ },
50
+ "files": [
51
+ "dist",
52
+ "web"
53
+ ]
54
+ }
@@ -0,0 +1 @@
1
+ *,*:before,*:after{box-sizing:border-box;margin:0;padding:0}:root{--bg: #0d1117;--bg2: #161b22;--bg3: #1c2129;--bg4: #242b35;--border: #30363d;--border2: #3d444d;--text: #e6edf3;--text2: #7d8590;--text3: #484f58;--accent: #a855f7;--accent2: #c084fc;--accent-bg: rgba(168, 85, 247, .1);--green: #3fb950;--green2: #56d364;--green-bg: rgba(63, 185, 80, .1);--green-line: rgba(63, 185, 80, .15);--red: #f85149;--red2: #ff7b72;--red-bg: rgba(248, 81, 73, .1);--red-line: rgba(248, 81, 73, .15);--yellow: #d29922;--yellow-bg: rgba(210, 153, 34, .1);--cyan: #58a6ff;--cyan-bg: rgba(88, 166, 255, .1);--radius: 8px;--radius-sm: 5px;--mono: "JetBrains Mono", "Cascadia Code", "Fira Code", "Consolas", monospace;--ui: "Inter", -apple-system, "Segoe UI", system-ui, sans-serif}body{background:var(--bg);color:var(--text);font-family:var(--ui);font-size:13px;height:100vh;overflow:hidden}::selection{background:var(--accent);color:#fff}::-webkit-scrollbar{width:6px;height:6px}::-webkit-scrollbar-track{background:transparent}::-webkit-scrollbar-thumb{background:var(--border);border-radius:3px}::-webkit-scrollbar-thumb:hover{background:var(--border2)}.app{display:flex;flex-direction:column;height:100vh}.center-screen{display:flex;align-items:center;justify-content:center;flex:1}.spinner{width:20px;height:20px;border:2px solid var(--border);border-top-color:var(--accent);border-radius:50%;animation:spin .5s linear infinite}.spin{animation:spin .8s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}.lang-picker{display:flex;align-items:center;justify-content:center;height:100vh;background:var(--bg)}.lang-content{display:flex;flex-direction:column;align-items:center;gap:24px}.lang-logo{font-size:48px;font-weight:800;color:var(--accent);letter-spacing:-1px}.lang-subtitle{color:var(--text2);font-size:15px;font-weight:500}.lang-options{display:flex;gap:12px}.lang-option{display:flex;align-items:center;gap:10px;padding:16px 32px;border-radius:var(--radius);border:1px solid var(--border);background:var(--bg2);color:var(--text);font-size:16px;font-weight:600;cursor:pointer;transition:all .15s;font-family:var(--ui)}.lang-option:hover{border-color:var(--accent);background:var(--accent-bg);transform:translateY(-2px)}.lang-flag{font-size:11px;font-weight:800;background:var(--bg3);padding:4px 8px;border-radius:4px;font-family:var(--mono);color:var(--accent2)}.titlebar{display:flex;align-items:center;height:38px;background:var(--bg);border-bottom:1px solid var(--border);padding:0 16px;-webkit-app-region:drag;-webkit-user-select:none;user-select:none;flex-shrink:0}.tb-dots{display:flex;gap:7px;width:60px}.tb-dot{width:12px;height:12px;border-radius:50%;-webkit-app-region:no-drag}.dot-r{background:#ff5f57}.dot-y{background:#ffbd2e}.dot-g{background:#28c840}.tb-center{flex:1;text-align:center;font-size:12px;color:var(--text2);font-weight:500;letter-spacing:.3px}.tb-right{width:80px;display:flex;justify-content:flex-end;align-items:center;gap:8px}.tb-lang{background:var(--bg3);border:1px solid var(--border);border-radius:4px;padding:2px 6px;font-size:10px;font-weight:800;font-family:var(--mono);color:var(--accent2);cursor:pointer;transition:all .12s;-webkit-app-region:no-drag}.tb-lang:hover{border-color:var(--accent);background:var(--accent-bg)}.tb-live{width:7px;height:7px;border-radius:50%;transition:all .3s}.tb-live.live{background:var(--green);box-shadow:0 0 8px var(--green)}.tb-live.offline{background:var(--red);box-shadow:0 0 8px var(--red)}.toolbar{display:flex;align-items:center;padding:8px 16px;gap:10px;background:var(--bg2);border-bottom:1px solid var(--border);flex-shrink:0}.tool-logo{font-size:15px;font-weight:800;background:linear-gradient(135deg,var(--accent) 0%,var(--accent2) 100%);-webkit-background-clip:text;-webkit-text-fill-color:transparent;background-clip:text;letter-spacing:-.3px}.tool-sep{width:1px;height:20px;background:var(--border)}.tool-branch{display:flex;align-items:center;gap:5px;background:var(--bg3);border:1px solid var(--border);padding:3px 10px;border-radius:14px;font-size:11px;font-family:var(--mono);color:var(--cyan)}.tool-badges{display:flex;gap:6px;margin-left:auto}.badge{padding:2px 10px;border-radius:12px;font-size:10px;font-weight:700;letter-spacing:.3px;display:flex;align-items:center;gap:4px}.badge-staged{background:var(--cyan-bg);color:var(--cyan)}.badge-changed{background:var(--yellow-bg);color:var(--yellow)}.badge-clean{background:var(--green-bg);color:var(--green)}.tool-actions{display:flex;gap:6px;margin-left:8px}.btn{display:inline-flex;align-items:center;gap:5px;padding:5px 12px;border-radius:var(--radius-sm);border:none;font-size:12px;font-weight:600;cursor:pointer;font-family:var(--ui);transition:all .12s}.btn:hover{transform:translateY(-1px)}.btn:active{transform:translateY(0)}.btn-ghost{background:transparent;color:var(--text2);border:1px solid var(--border)}.btn-ghost:hover{background:var(--bg3);color:var(--text)}.btn-primary{background:var(--accent);color:#fff}.btn-primary:hover{background:var(--accent2)}.btn-primary:disabled{opacity:.6;cursor:not-allowed;transform:none}.btn-danger{color:var(--red);border-color:#f851494d}.btn-danger:hover{background:var(--red-bg);color:var(--red);border-color:var(--red)}.main{display:flex;flex:1;overflow:hidden}.sidebar{width:280px;min-width:240px;background:var(--bg2);border-right:1px solid var(--border);display:flex;flex-direction:column;overflow:hidden}.sb-head{padding:10px 14px;display:flex;align-items:center;justify-content:space-between;border-bottom:1px solid var(--border);font-size:11px;font-weight:700;text-transform:uppercase;letter-spacing:.8px;color:var(--text2)}.sb-count{background:var(--accent-bg);color:var(--accent2);padding:1px 7px;border-radius:10px;font-family:var(--mono);font-size:10px;margin-left:6px}.sb-search{padding:8px 10px;border-bottom:1px solid var(--border)}.sb-search input{width:100%;background:var(--bg);border:1px solid var(--border);border-radius:var(--radius-sm);padding:6px 10px;font-size:12px;color:var(--text);font-family:var(--mono);outline:none;transition:border .15s}.sb-search input:focus{border-color:var(--accent)}.sb-search input::placeholder{color:var(--text3)}.file-list{flex:1;overflow-y:auto;padding:4px 0}.file-list-empty{padding:30px;text-align:center;color:var(--text3);font-size:12px}.fi{display:flex;align-items:center;padding:4px 12px;gap:7px;cursor:pointer;transition:background .08s;-webkit-user-select:none;user-select:none;font-size:12px;font-family:var(--mono);border-left:2px solid transparent}.fi:hover{background:var(--bg3)}.fi.sel{background:var(--accent-bg);border-left-color:var(--accent)}.fi>div,.fi>span{transition:transform .12s ease}.fi.sel>div,.fi.sel>span{transform:translate(2px)}.fi-icon{width:18px;height:18px;flex-shrink:0;display:flex;align-items:center;justify-content:center}.fi-icon svg{width:16px;height:16px}.fi-name{flex:1;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.fi-dir .fi-name{color:var(--text);font-weight:600}.fi-check{width:14px;height:14px;border:1.5px solid var(--border2);border-radius:3px;display:flex;align-items:center;justify-content:center;flex-shrink:0;transition:all .12s;cursor:pointer;color:transparent}.fi-check.on{border-color:var(--accent);background:var(--accent);color:#fff}.fi-badge{font-size:9px;font-weight:800;padding:1px 5px;border-radius:3px;flex-shrink:0;letter-spacing:.3px}.fi-badge.M{background:var(--yellow-bg);color:var(--yellow)}.fi-badge.U{background:var(--green-bg);color:var(--green)}.fi-badge.D{background:var(--red-bg);color:var(--red)}.fi-badge.S{background:var(--cyan-bg);color:var(--cyan)}.content{flex:1;display:flex;flex-direction:column;overflow:hidden;background:var(--bg)}.content-header{display:flex;align-items:center;gap:0;background:var(--bg2);border-bottom:1px solid var(--border);height:38px;flex-shrink:0}.content-tabs{display:flex;height:100%}.content-tab{display:flex;align-items:center;gap:5px;padding:0 14px;height:100%;border:none;background:transparent;color:var(--text2);font-size:12px;font-weight:500;font-family:var(--ui);cursor:pointer;border-bottom:2px solid transparent;transition:all .12s}.content-tab:hover{color:var(--text);background:var(--bg3)}.content-tab.active{color:var(--text);border-bottom-color:var(--accent)}.content-filename{margin-left:auto;padding-right:16px;font-size:12px;font-family:var(--mono);color:var(--cyan);font-weight:500}.content-stats{display:flex;gap:12px;padding:6px 16px;font-size:12px;font-weight:700;font-family:var(--mono);background:var(--bg2);border-bottom:1px solid var(--border)}.stat-add{color:var(--green)}.stat-del{color:var(--red)}.content-body{flex:1;overflow:hidden;display:flex;flex-direction:column}.content-empty{display:flex;align-items:center;justify-content:center;flex:1;flex-direction:column;gap:10px;color:var(--text3)}.content-empty svg{opacity:.15}.content-empty p{font-size:14px;font-weight:500}.content-empty span{font-size:12px}.content-empty.small{padding:40px}.content-empty.small p{font-size:13px}.diff-body{flex:1;overflow-y:auto;font-family:var(--mono);font-size:12px;line-height:1.7}.dl{display:flex;min-height:22px;border-bottom:1px solid rgba(48,54,61,.3)}.dl:hover{background:#ffffff05}.dl-num{width:50px;flex-shrink:0;text-align:right;padding:0 8px;color:var(--text3);-webkit-user-select:none;user-select:none;font-size:11px;border-right:1px solid var(--border);display:flex;align-items:center;justify-content:flex-end}.dl-sign{width:20px;flex-shrink:0;text-align:center;-webkit-user-select:none;user-select:none;font-weight:700;display:flex;align-items:center;justify-content:center}.dl-code{flex:1;padding:0 12px;white-space:pre;overflow-x:auto}.dl-add{background:var(--green-line);border-left:3px solid var(--green2)}.dl-add .dl-sign,.dl-add .dl-code{color:var(--green2)}.dl-del{background:var(--red-line);border-left:3px solid var(--red2)}.dl-del .dl-sign,.dl-del .dl-code{color:var(--red2)}.dl-hunk{background:var(--cyan-bg)}.dl-hunk .dl-code{color:var(--cyan);font-weight:600;font-style:italic}.dl-meta .dl-code{color:var(--text3)}.code-body{flex:1;overflow-y:auto;font-family:var(--mono);font-size:12px;line-height:1.7}.cl{display:flex;min-height:22px}.cl:hover{background:var(--bg3)}.cl-num{width:50px;flex-shrink:0;text-align:right;padding:0 8px;color:var(--text3);-webkit-user-select:none;user-select:none;font-size:11px;border-right:1px solid var(--border);display:flex;align-items:center;justify-content:flex-end}.cl-code{flex:1;padding:0 12px;white-space:pre;overflow-x:auto;color:var(--text)}.hl-keyword{color:#ff7b72}.hl-string{color:#a5d6ff}.hl-comment{color:#8b949e;font-style:italic}.hl-number{color:#79c0ff}.hl-bool{color:#ff7b72}.hl-fn{color:#d2a8ff}.hl-type{color:#ffa657}.hl-tag{color:#7ee787}.hl-attr{color:#79c0ff}.val-body{flex:1;overflow-y:auto;padding:16px}.val-header{display:flex;align-items:center;gap:12px;margin-bottom:16px}.val-stat{font-size:12px;color:var(--text2);font-family:var(--mono)}.val-badge{padding:3px 10px;border-radius:12px;font-size:11px;font-weight:700}.val-ok{background:var(--green-bg);color:var(--green)}.val-warn{background:var(--yellow-bg);color:var(--yellow)}.val-list{display:flex;flex-direction:column;gap:6px}.val-item{display:flex;align-items:center;gap:10px;padding:10px 14px;border-radius:var(--radius-sm);font-size:13px;border:1px solid var(--border);background:var(--bg2)}.val-item-icon{display:flex;flex-shrink:0}.val-warning,.val-warning .val-item-icon{color:var(--yellow)}.val-info,.val-info .val-item-icon{color:var(--cyan)}.val-count{font-weight:700;font-family:var(--mono);min-width:20px;color:var(--text)}.val-msg{flex:1;color:var(--text)}.val-lines{font-size:11px;color:var(--text3);font-family:var(--mono)}.overlay{position:fixed;top:0;right:0;bottom:0;left:0;background:#0009;display:flex;align-items:center;justify-content:center;z-index:100;-webkit-backdrop-filter:blur(4px);backdrop-filter:blur(4px);animation:fadeIn .12s}@keyframes fadeIn{0%{opacity:0}to{opacity:1}}@keyframes slideUp{0%{transform:translateY(12px);opacity:0}to{transform:translateY(0);opacity:1}}.modal{background:var(--bg2);border:1px solid var(--border);border-radius:var(--radius);width:440px;max-width:92vw;padding:24px;animation:slideUp .15s;box-shadow:0 20px 60px #00000080}.modal h2{font-size:16px;margin-bottom:4px}.modal-desc{color:var(--text2);font-size:12px;margin-bottom:16px}.modal-input{width:100%;background:var(--bg);border:1px solid var(--border);border-radius:var(--radius-sm);padding:10px 14px;font-size:14px;color:var(--text);font-family:var(--mono);outline:none;transition:border .15s}.modal-input:focus{border-color:var(--accent);box-shadow:0 0 0 3px #a855f71f}.modal-input::placeholder{color:var(--text3)}.modal-actions{display:flex;gap:8px;margin-top:16px;justify-content:flex-end}.modal-option{display:flex;align-items:center;gap:12px;padding:10px 14px;border-radius:var(--radius-sm);cursor:pointer;transition:all .1s;border:1px solid transparent;margin-bottom:6px}.modal-option:hover{background:var(--bg3);border-color:var(--border)}.modal-option.active{background:var(--accent-bg);border-color:var(--accent)}.modal-option-icon{width:32px;height:32px;border-radius:6px;display:flex;align-items:center;justify-content:center;background:var(--bg3);color:var(--accent2)}.modal-option-label{font-weight:600;font-size:13px}.modal-option-desc{color:var(--text2);font-size:11px;margin-top:1px}.toast-container{position:fixed;bottom:16px;right:16px;z-index:200;display:flex;flex-direction:column-reverse;gap:8px}.toast{padding:8px 14px;border-radius:var(--radius-sm);font-size:13px;font-weight:500;animation:slideIn .2s;box-shadow:0 8px 24px #0006;display:flex;align-items:center;gap:8px;cursor:pointer}@keyframes slideIn{0%{transform:translateY(16px);opacity:0}to{transform:translateY(0);opacity:1}}.toast-ok{background:#0d2818;color:var(--green);border:1px solid #1a4028}.toast-error{background:#2a0f0f;color:var(--red);border:1px solid #4a1a1a}.toast-info{background:#0d1f30;color:var(--cyan);border:1px solid #1a3050}.init-screen{display:flex;align-items:center;justify-content:center;flex:1;flex-direction:column;gap:16px}.init-logo{display:flex;flex-direction:column;align-items:center;gap:8px;color:var(--accent)}.init-logo h1{font-size:36px;font-weight:800;letter-spacing:-1px}.init-screen p{color:var(--text2);font-size:14px}.init-screen .modal-input{max-width:380px}