@qrafty-ai/opencode-kanban 0.2.0 → 0.2.1-linux-x64

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 CHANGED
@@ -88,8 +88,10 @@ cargo build --release
88
88
 
89
89
  ```bash
90
90
  opencode-kanban --theme default
91
+ opencode-kanban --theme light
91
92
  opencode-kanban --theme high-contrast
92
93
  opencode-kanban --theme mono
94
+ opencode-kanban --theme custom
93
95
  ```
94
96
 
95
97
  Each project uses its own SQLite file and board state.
@@ -133,11 +135,88 @@ For full, current bindings, use the in-app help overlay (`?`).
133
135
  ## Configuration
134
136
 
135
137
  - Settings file: `~/.config/opencode-kanban/settings.toml`
136
- - Legacy keybindings file (deprecated): `~/.config/opencode-kanban/keybindings.toml`
137
138
  - Project databases (Linux default): `~/.local/share/opencode-kanban/*.sqlite`
138
139
 
139
140
  The app creates config/data files on demand.
140
141
 
142
+ ### Theme configuration options
143
+
144
+ Theme values live in `~/.config/opencode-kanban/settings.toml`.
145
+
146
+ Top-level option:
147
+
148
+ - `theme`: `default` | `light` | `high-contrast` | `mono` | `custom`
149
+
150
+ When `theme = "custom"`, configure semantic tokens with these sections:
151
+
152
+ - `[custom_theme]`
153
+ - `inherit`: `default` | `light` | `high-contrast` | `mono`
154
+ - `[custom_theme.base]`
155
+ - `canvas`, `surface`, `text`, `text_muted`, `header`, `accent`, `danger`
156
+ - `[custom_theme.interactive]`
157
+ - `focus`, `selected_bg`, `selected_border`, `border`
158
+ - `[custom_theme.status]`
159
+ - `running`, `waiting`, `idle`, `dead`, `broken`, `unavailable`
160
+ - `[custom_theme.tile]`
161
+ - `repo`, `branch`, `todo`
162
+ - `[custom_theme.category]`
163
+ - `primary`, `secondary`, `tertiary`, `success`, `warning`, `danger`
164
+ - `[custom_theme.dialog]`
165
+ - `surface`, `input_bg`, `button_bg`, `button_fg`
166
+
167
+ Example:
168
+
169
+ ```toml
170
+ theme = "custom"
171
+
172
+ [custom_theme]
173
+ inherit = "light"
174
+
175
+ [custom_theme.base]
176
+ canvas = "#E2E7EE"
177
+ surface = "#ECF1F7"
178
+ text = "#222A3A"
179
+ text_muted = "#4E596D"
180
+ header = "#2F66BF"
181
+ accent = "#0E7490"
182
+ danger = "#B02E24"
183
+
184
+ [custom_theme.interactive]
185
+ focus = "#2F66BF"
186
+ selected_bg = "#D6DFED"
187
+ selected_border = "#477ACD"
188
+ border = "#A5B2C6"
189
+
190
+ [custom_theme.status]
191
+ running = "#278449"
192
+ waiting = "#AB781A"
193
+ idle = "#5D687A"
194
+ dead = "#B02E24"
195
+ broken = "#B02E24"
196
+ unavailable = "#B02E24"
197
+
198
+ [custom_theme.tile]
199
+ repo = "#086678"
200
+ branch = "#926614"
201
+ todo = "#4E596D"
202
+
203
+ [custom_theme.category]
204
+ primary = "#2F66BF"
205
+ secondary = "#AB501F"
206
+ tertiary = "#6949AB"
207
+ success = "#278449"
208
+ warning = "#AB781A"
209
+ danger = "#B02E24"
210
+
211
+ [custom_theme.dialog]
212
+ surface = "#ECF1F7"
213
+ input_bg = "#E0E6EF"
214
+ button_bg = "#CDD8E7"
215
+ button_fg = "#FFFFFF"
216
+ ```
217
+
218
+ - Accepted color format: `#RRGGBB` (hex only).
219
+
141
220
  ## Troubleshooting
142
221
 
143
222
  - `tmux is required but not available`:
package/package.json CHANGED
@@ -1,24 +1,22 @@
1
1
  {
2
2
  "name": "@qrafty-ai/opencode-kanban",
3
- "version": "0.2.0",
3
+ "version": "0.2.1-linux-x64",
4
4
  "description": "Terminal kanban board for managing OpenCode tmux sessions",
5
5
  "license": "MIT",
6
- "type": "module",
7
- "bin": {
8
- "opencode-kanban": "bin/opencode-kanban.js"
9
- },
10
- "engines": {
11
- "node": ">=18"
12
- },
6
+ "os": [
7
+ "linux"
8
+ ],
9
+ "cpu": [
10
+ "x64"
11
+ ],
13
12
  "files": [
14
- "bin"
13
+ "vendor"
15
14
  ],
16
15
  "repository": {
17
16
  "type": "git",
18
17
  "url": "https://github.com/qrafty-ai/opencode-kanban.git"
19
18
  },
20
- "optionalDependencies": {
21
- "@qrafty-ai/opencode-kanban-linux-x64": "npm:@qrafty-ai/opencode-kanban@0.2.0-linux-x64",
22
- "@qrafty-ai/opencode-kanban-darwin-arm64": "npm:@qrafty-ai/opencode-kanban@0.2.0-darwin-arm64"
19
+ "engines": {
20
+ "node": ">=18"
23
21
  }
24
22
  }
@@ -1,137 +0,0 @@
1
- #!/usr/bin/env node
2
-
3
- import { spawn } from "node:child_process";
4
- import { accessSync, chmodSync, constants, existsSync } from "node:fs";
5
- import { createRequire } from "node:module";
6
- import path from "node:path";
7
- import { fileURLToPath } from "node:url";
8
-
9
- const __filename = fileURLToPath(import.meta.url);
10
- const __dirname = path.dirname(__filename);
11
- const require = createRequire(import.meta.url);
12
-
13
- const PLATFORM_PACKAGE_BY_TARGET = {
14
- "x86_64-unknown-linux-gnu": "@qrafty-ai/opencode-kanban-linux-x64",
15
- "aarch64-apple-darwin": "@qrafty-ai/opencode-kanban-darwin-arm64",
16
- };
17
-
18
- function detectTargetTriple() {
19
- const { platform, arch } = process;
20
-
21
- if (platform === "linux" && arch === "x64") {
22
- return "x86_64-unknown-linux-gnu";
23
- }
24
-
25
- if (platform === "darwin" && arch === "x64") {
26
- throw new Error("darwin-x64 is not supported. Please use macOS with Apple Silicon (arm64).");
27
- }
28
-
29
- if (platform === "darwin" && arch === "arm64") {
30
- return "aarch64-apple-darwin";
31
- }
32
-
33
- throw new Error(`Unsupported platform: ${platform} (${arch})`);
34
- }
35
-
36
- function detectPackageManager() {
37
- const userAgent = process.env.npm_config_user_agent || "";
38
- if (/\bbun\//.test(userAgent)) {
39
- return "bun";
40
- }
41
- return "npm";
42
- }
43
-
44
- const targetTriple = detectTargetTriple();
45
- const platformPackage = PLATFORM_PACKAGE_BY_TARGET[targetTriple];
46
- const binaryName = process.platform === "win32" ? "opencode-kanban.exe" : "opencode-kanban";
47
- const localVendorRoot = path.join(__dirname, "..", "vendor");
48
- const localBinaryPath = path.join(
49
- localVendorRoot,
50
- targetTriple,
51
- "opencode-kanban",
52
- binaryName,
53
- );
54
-
55
- let vendorRoot;
56
- try {
57
- const packageJsonPath = require.resolve(`${platformPackage}/package.json`);
58
- vendorRoot = path.join(path.dirname(packageJsonPath), "vendor");
59
- } catch {
60
- if (existsSync(localBinaryPath)) {
61
- vendorRoot = localVendorRoot;
62
- }
63
- }
64
-
65
- if (!vendorRoot) {
66
- const packageManager = detectPackageManager();
67
- const updateCommand =
68
- packageManager === "bun"
69
- ? "bun install -g @qrafty-ai/opencode-kanban@latest"
70
- : "npm install -g @qrafty-ai/opencode-kanban@latest";
71
- throw new Error(
72
- `Missing optional dependency ${platformPackage}. Reinstall opencode-kanban: ${updateCommand}`,
73
- );
74
- }
75
-
76
- const binaryPath = path.join(vendorRoot, targetTriple, "opencode-kanban", binaryName);
77
-
78
- function ensureExecutable(pathToBinary) {
79
- if (process.platform === "win32") {
80
- return;
81
- }
82
-
83
- try {
84
- accessSync(pathToBinary, constants.X_OK);
85
- return;
86
- } catch {
87
- }
88
-
89
- try {
90
- chmodSync(pathToBinary, 0o755);
91
- accessSync(pathToBinary, constants.X_OK);
92
- } catch (error) {
93
- throw new Error(`Binary is not executable: ${pathToBinary}`, { cause: error });
94
- }
95
- }
96
-
97
- ensureExecutable(binaryPath);
98
-
99
- const child = spawn(binaryPath, process.argv.slice(2), {
100
- stdio: "inherit",
101
- env: process.env,
102
- });
103
-
104
- child.on("error", (error) => {
105
- console.error(error);
106
- process.exit(1);
107
- });
108
-
109
- const forwardSignal = (signal) => {
110
- if (child.killed) {
111
- return;
112
- }
113
- try {
114
- child.kill(signal);
115
- } catch {
116
- }
117
- };
118
-
119
- ["SIGINT", "SIGTERM", "SIGHUP"].forEach((signal) => {
120
- process.on(signal, () => forwardSignal(signal));
121
- });
122
-
123
- const childResult = await new Promise((resolve) => {
124
- child.on("exit", (code, signal) => {
125
- if (signal) {
126
- resolve({ type: "signal", signal });
127
- return;
128
- }
129
- resolve({ type: "code", exitCode: code ?? 1 });
130
- });
131
- });
132
-
133
- if (childResult.type === "signal") {
134
- process.kill(process.pid, childResult.signal);
135
- } else {
136
- process.exit(childResult.exitCode);
137
- }