@citadel-labs/beads-ui 2.0.7 → 2.3.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.
@@ -0,0 +1,34 @@
1
+ ---
2
+ name: start
3
+ description: Start the beads-board dashboard server
4
+ ---
5
+
6
+ # Start beads-board
7
+
8
+ Start the beads-board dashboard server and print the URL.
9
+
10
+ ## Steps
11
+
12
+ 1. Check if the server is already running:
13
+
14
+ ```bash
15
+ cat .beads-board.pid 2>/dev/null
16
+ ```
17
+
18
+ If the pidfile exists and contains a valid PID, check if the process is alive:
19
+
20
+ ```bash
21
+ node -e "const d=JSON.parse(require('fs').readFileSync('.beads-board.pid','utf8')); try{process.kill(d.pid,0);console.log('running on port '+d.port)}catch{console.log('stale')}"
22
+ ```
23
+
24
+ If running, tell the user the URL (`http://localhost:<port>`) and stop.
25
+
26
+ 2. If not running, start the server in the background:
27
+
28
+ ```bash
29
+ node server/index.js "$(pwd)" &
30
+ ```
31
+
32
+ 3. The server prints `beads-board server running at http://localhost:<port>`. Tell the user the URL.
33
+
34
+ 4. Tell the user they can stop the server with `/beads-board-stop`.
@@ -0,0 +1,38 @@
1
+ ---
2
+ name: stop
3
+ description: Stop the beads-board dashboard server
4
+ ---
5
+
6
+ # Stop beads-board
7
+
8
+ Stop a running beads-board server.
9
+
10
+ ## Steps
11
+
12
+ 1. Check if the pidfile exists:
13
+
14
+ ```bash
15
+ cat .beads-board.pid 2>/dev/null
16
+ ```
17
+
18
+ If the pidfile doesn't exist, tell the user the server is not running and stop.
19
+
20
+ 2. Kill the server process:
21
+
22
+ ```bash
23
+ kill $(node -e "console.log(JSON.parse(require('fs').readFileSync('.beads-board.pid','utf8')).pid)")
24
+ ```
25
+
26
+ 3. Verify the pidfile was cleaned up:
27
+
28
+ ```bash
29
+ ls .beads-board.pid 2>/dev/null
30
+ ```
31
+
32
+ If it still exists, remove it:
33
+
34
+ ```bash
35
+ rm -f .beads-board.pid
36
+ ```
37
+
38
+ 4. Tell the user the server has been stopped.
package/README.md CHANGED
@@ -9,7 +9,8 @@ A minimal kanban dashboard and git log viewer for [Beads](https://github.com/ste
9
9
  - **Bead ID linking** — Bead IDs in commit messages are highlighted as badges
10
10
  - **Dark/light theme** — Toggle between themes, dark by default
11
11
  - **Auto-refresh** — Polls for updates every 5 seconds
12
- - **Zero runtime dependencies** — Server uses only Node.js stdlib
12
+ - **Integrated terminal** — Built-in terminal panel powered by node-pty and xterm.js
13
+ - **Minimal runtime dependencies** — Server uses Node.js stdlib plus `node-pty` and `ws` for the terminal
13
14
 
14
15
  ## Quick Start
15
16
 
@@ -0,0 +1,224 @@
1
+ body, html {
2
+ margin:0; padding: 0;
3
+ height: 100%;
4
+ }
5
+ body {
6
+ font-family: Helvetica Neue, Helvetica, Arial;
7
+ font-size: 14px;
8
+ color:#333;
9
+ }
10
+ .small { font-size: 12px; }
11
+ *, *:after, *:before {
12
+ -webkit-box-sizing:border-box;
13
+ -moz-box-sizing:border-box;
14
+ box-sizing:border-box;
15
+ }
16
+ h1 { font-size: 20px; margin: 0;}
17
+ h2 { font-size: 14px; }
18
+ pre {
19
+ font: 12px/1.4 Consolas, "Liberation Mono", Menlo, Courier, monospace;
20
+ margin: 0;
21
+ padding: 0;
22
+ -moz-tab-size: 2;
23
+ -o-tab-size: 2;
24
+ tab-size: 2;
25
+ }
26
+ a { color:#0074D9; text-decoration:none; }
27
+ a:hover { text-decoration:underline; }
28
+ .strong { font-weight: bold; }
29
+ .space-top1 { padding: 10px 0 0 0; }
30
+ .pad2y { padding: 20px 0; }
31
+ .pad1y { padding: 10px 0; }
32
+ .pad2x { padding: 0 20px; }
33
+ .pad2 { padding: 20px; }
34
+ .pad1 { padding: 10px; }
35
+ .space-left2 { padding-left:55px; }
36
+ .space-right2 { padding-right:20px; }
37
+ .center { text-align:center; }
38
+ .clearfix { display:block; }
39
+ .clearfix:after {
40
+ content:'';
41
+ display:block;
42
+ height:0;
43
+ clear:both;
44
+ visibility:hidden;
45
+ }
46
+ .fl { float: left; }
47
+ @media only screen and (max-width:640px) {
48
+ .col3 { width:100%; max-width:100%; }
49
+ .hide-mobile { display:none!important; }
50
+ }
51
+
52
+ .quiet {
53
+ color: #7f7f7f;
54
+ color: rgba(0,0,0,0.5);
55
+ }
56
+ .quiet a { opacity: 0.7; }
57
+
58
+ .fraction {
59
+ font-family: Consolas, 'Liberation Mono', Menlo, Courier, monospace;
60
+ font-size: 10px;
61
+ color: #555;
62
+ background: #E8E8E8;
63
+ padding: 4px 5px;
64
+ border-radius: 3px;
65
+ vertical-align: middle;
66
+ }
67
+
68
+ div.path a:link, div.path a:visited { color: #333; }
69
+ table.coverage {
70
+ border-collapse: collapse;
71
+ margin: 10px 0 0 0;
72
+ padding: 0;
73
+ }
74
+
75
+ table.coverage td {
76
+ margin: 0;
77
+ padding: 0;
78
+ vertical-align: top;
79
+ }
80
+ table.coverage td.line-count {
81
+ text-align: right;
82
+ padding: 0 5px 0 20px;
83
+ }
84
+ table.coverage td.line-coverage {
85
+ text-align: right;
86
+ padding-right: 10px;
87
+ min-width:20px;
88
+ }
89
+
90
+ table.coverage td span.cline-any {
91
+ display: inline-block;
92
+ padding: 0 5px;
93
+ width: 100%;
94
+ }
95
+ .missing-if-branch {
96
+ display: inline-block;
97
+ margin-right: 5px;
98
+ border-radius: 3px;
99
+ position: relative;
100
+ padding: 0 4px;
101
+ background: #333;
102
+ color: yellow;
103
+ }
104
+
105
+ .skip-if-branch {
106
+ display: none;
107
+ margin-right: 10px;
108
+ position: relative;
109
+ padding: 0 4px;
110
+ background: #ccc;
111
+ color: white;
112
+ }
113
+ .missing-if-branch .typ, .skip-if-branch .typ {
114
+ color: inherit !important;
115
+ }
116
+ .coverage-summary {
117
+ border-collapse: collapse;
118
+ width: 100%;
119
+ }
120
+ .coverage-summary tr { border-bottom: 1px solid #bbb; }
121
+ .keyline-all { border: 1px solid #ddd; }
122
+ .coverage-summary td, .coverage-summary th { padding: 10px; }
123
+ .coverage-summary tbody { border: 1px solid #bbb; }
124
+ .coverage-summary td { border-right: 1px solid #bbb; }
125
+ .coverage-summary td:last-child { border-right: none; }
126
+ .coverage-summary th {
127
+ text-align: left;
128
+ font-weight: normal;
129
+ white-space: nowrap;
130
+ }
131
+ .coverage-summary th.file { border-right: none !important; }
132
+ .coverage-summary th.pct { }
133
+ .coverage-summary th.pic,
134
+ .coverage-summary th.abs,
135
+ .coverage-summary td.pct,
136
+ .coverage-summary td.abs { text-align: right; }
137
+ .coverage-summary td.file { white-space: nowrap; }
138
+ .coverage-summary td.pic { min-width: 120px !important; }
139
+ .coverage-summary tfoot td { }
140
+
141
+ .coverage-summary .sorter {
142
+ height: 10px;
143
+ width: 7px;
144
+ display: inline-block;
145
+ margin-left: 0.5em;
146
+ background: url(sort-arrow-sprite.png) no-repeat scroll 0 0 transparent;
147
+ }
148
+ .coverage-summary .sorted .sorter {
149
+ background-position: 0 -20px;
150
+ }
151
+ .coverage-summary .sorted-desc .sorter {
152
+ background-position: 0 -10px;
153
+ }
154
+ .status-line { height: 10px; }
155
+ /* yellow */
156
+ .cbranch-no { background: yellow !important; color: #111; }
157
+ /* dark red */
158
+ .red.solid, .status-line.low, .low .cover-fill { background:#C21F39 }
159
+ .low .chart { border:1px solid #C21F39 }
160
+ .highlighted,
161
+ .highlighted .cstat-no, .highlighted .fstat-no, .highlighted .cbranch-no{
162
+ background: #C21F39 !important;
163
+ }
164
+ /* medium red */
165
+ .cstat-no, .fstat-no, .cbranch-no, .cbranch-no { background:#F6C6CE }
166
+ /* light red */
167
+ .low, .cline-no { background:#FCE1E5 }
168
+ /* light green */
169
+ .high, .cline-yes { background:rgb(230,245,208) }
170
+ /* medium green */
171
+ .cstat-yes { background:rgb(161,215,106) }
172
+ /* dark green */
173
+ .status-line.high, .high .cover-fill { background:rgb(77,146,33) }
174
+ .high .chart { border:1px solid rgb(77,146,33) }
175
+ /* dark yellow (gold) */
176
+ .status-line.medium, .medium .cover-fill { background: #f9cd0b; }
177
+ .medium .chart { border:1px solid #f9cd0b; }
178
+ /* light yellow */
179
+ .medium { background: #fff4c2; }
180
+
181
+ .cstat-skip { background: #ddd; color: #111; }
182
+ .fstat-skip { background: #ddd; color: #111 !important; }
183
+ .cbranch-skip { background: #ddd !important; color: #111; }
184
+
185
+ span.cline-neutral { background: #eaeaea; }
186
+
187
+ .coverage-summary td.empty {
188
+ opacity: .5;
189
+ padding-top: 4px;
190
+ padding-bottom: 4px;
191
+ line-height: 1;
192
+ color: #888;
193
+ }
194
+
195
+ .cover-fill, .cover-empty {
196
+ display:inline-block;
197
+ height: 12px;
198
+ }
199
+ .chart {
200
+ line-height: 0;
201
+ }
202
+ .cover-empty {
203
+ background: white;
204
+ }
205
+ .cover-full {
206
+ border-right: none !important;
207
+ }
208
+ pre.prettyprint {
209
+ border: none !important;
210
+ padding: 0 !important;
211
+ margin: 0 !important;
212
+ }
213
+ .com { color: #999 !important; }
214
+ .ignore-none { color: #999; font-weight: normal; }
215
+
216
+ .wrapper {
217
+ min-height: 100%;
218
+ height: auto !important;
219
+ height: 100%;
220
+ margin: 0 auto -48px;
221
+ }
222
+ .footer, .push {
223
+ height: 48px;
224
+ }
@@ -0,0 +1,87 @@
1
+ /* eslint-disable */
2
+ var jumpToCode = (function init() {
3
+ // Classes of code we would like to highlight in the file view
4
+ var missingCoverageClasses = ['.cbranch-no', '.cstat-no', '.fstat-no'];
5
+
6
+ // Elements to highlight in the file listing view
7
+ var fileListingElements = ['td.pct.low'];
8
+
9
+ // We don't want to select elements that are direct descendants of another match
10
+ var notSelector = ':not(' + missingCoverageClasses.join('):not(') + ') > '; // becomes `:not(a):not(b) > `
11
+
12
+ // Selector that finds elements on the page to which we can jump
13
+ var selector =
14
+ fileListingElements.join(', ') +
15
+ ', ' +
16
+ notSelector +
17
+ missingCoverageClasses.join(', ' + notSelector); // becomes `:not(a):not(b) > a, :not(a):not(b) > b`
18
+
19
+ // The NodeList of matching elements
20
+ var missingCoverageElements = document.querySelectorAll(selector);
21
+
22
+ var currentIndex;
23
+
24
+ function toggleClass(index) {
25
+ missingCoverageElements
26
+ .item(currentIndex)
27
+ .classList.remove('highlighted');
28
+ missingCoverageElements.item(index).classList.add('highlighted');
29
+ }
30
+
31
+ function makeCurrent(index) {
32
+ toggleClass(index);
33
+ currentIndex = index;
34
+ missingCoverageElements.item(index).scrollIntoView({
35
+ behavior: 'smooth',
36
+ block: 'center',
37
+ inline: 'center'
38
+ });
39
+ }
40
+
41
+ function goToPrevious() {
42
+ var nextIndex = 0;
43
+ if (typeof currentIndex !== 'number' || currentIndex === 0) {
44
+ nextIndex = missingCoverageElements.length - 1;
45
+ } else if (missingCoverageElements.length > 1) {
46
+ nextIndex = currentIndex - 1;
47
+ }
48
+
49
+ makeCurrent(nextIndex);
50
+ }
51
+
52
+ function goToNext() {
53
+ var nextIndex = 0;
54
+
55
+ if (
56
+ typeof currentIndex === 'number' &&
57
+ currentIndex < missingCoverageElements.length - 1
58
+ ) {
59
+ nextIndex = currentIndex + 1;
60
+ }
61
+
62
+ makeCurrent(nextIndex);
63
+ }
64
+
65
+ return function jump(event) {
66
+ if (
67
+ document.getElementById('fileSearch') === document.activeElement &&
68
+ document.activeElement != null
69
+ ) {
70
+ // if we're currently focused on the search input, we don't want to navigate
71
+ return;
72
+ }
73
+
74
+ switch (event.which) {
75
+ case 78: // n
76
+ case 74: // j
77
+ goToNext();
78
+ break;
79
+ case 66: // b
80
+ case 75: // k
81
+ case 80: // p
82
+ goToPrevious();
83
+ break;
84
+ }
85
+ };
86
+ })();
87
+ window.addEventListener('keydown', jumpToCode);
@@ -0,0 +1,153 @@
1
+ <?xml version="1.0" encoding="UTF-8"?>
2
+ <coverage generated="1773281849276" clover="3.2.0">
3
+ <project timestamp="1773281849276" name="All files">
4
+ <metrics statements="144" coveredstatements="58" conditionals="96" coveredconditionals="33" methods="16" coveredmethods="10" elements="256" coveredelements="101" complexity="0" loc="144" ncloc="144" packages="1" files="1" classes="1"/>
5
+ <file name="handlers.js" path="D:\beads-board\server\handlers.js">
6
+ <metrics statements="144" coveredstatements="58" conditionals="96" coveredconditionals="33" methods="16" coveredmethods="10"/>
7
+ <line num="1" count="1" type="stmt"/>
8
+ <line num="2" count="1" type="stmt"/>
9
+ <line num="3" count="1" type="stmt"/>
10
+ <line num="4" count="1" type="stmt"/>
11
+ <line num="10" count="1" type="stmt"/>
12
+ <line num="25" count="1" type="stmt"/>
13
+ <line num="28" count="1" type="stmt"/>
14
+ <line num="32" count="5" type="stmt"/>
15
+ <line num="33" count="5" type="stmt"/>
16
+ <line num="34" count="5" type="cond" truecount="1" falsecount="1"/>
17
+ <line num="35" count="0" type="cond" truecount="0" falsecount="2"/>
18
+ <line num="36" count="0" type="stmt"/>
19
+ <line num="38" count="5" type="stmt"/>
20
+ <line num="44" count="5" type="stmt"/>
21
+ <line num="45" count="5" type="stmt"/>
22
+ <line num="46" count="5" type="stmt"/>
23
+ <line num="49" count="0" type="stmt"/>
24
+ <line num="54" count="0" type="stmt"/>
25
+ <line num="62" count="2" type="cond" truecount="3" falsecount="1"/>
26
+ <line num="63" count="2" type="stmt"/>
27
+ <line num="65" count="2" type="stmt"/>
28
+ <line num="73" count="6" type="stmt"/>
29
+ <line num="77" count="6" type="stmt"/>
30
+ <line num="81" count="1" type="stmt"/>
31
+ <line num="89" count="1" type="stmt"/>
32
+ <line num="90" count="7" type="stmt"/>
33
+ <line num="91" count="7" type="stmt"/>
34
+ <line num="93" count="7" type="cond" truecount="1" falsecount="1"/>
35
+ <line num="94" count="0" type="stmt"/>
36
+ <line num="99" count="0" type="stmt"/>
37
+ <line num="100" count="0" type="stmt"/>
38
+ <line num="103" count="7" type="stmt"/>
39
+ <line num="104" count="7" type="cond" truecount="2" falsecount="0"/>
40
+ <line num="105" count="1" type="stmt"/>
41
+ <line num="106" count="1" type="cond" truecount="1" falsecount="1"/>
42
+ <line num="107" count="6" type="cond" truecount="2" falsecount="0"/>
43
+ <line num="108" count="1" type="stmt"/>
44
+ <line num="109" count="1" type="cond" truecount="1" falsecount="1"/>
45
+ <line num="110" count="5" type="cond" truecount="2" falsecount="0"/>
46
+ <line num="111" count="1" type="stmt"/>
47
+ <line num="112" count="1" type="cond" truecount="1" falsecount="1"/>
48
+ <line num="113" count="4" type="cond" truecount="2" falsecount="0"/>
49
+ <line num="114" count="3" type="stmt"/>
50
+ <line num="115" count="3" type="cond" truecount="4" falsecount="0"/>
51
+ <line num="116" count="1" type="stmt"/>
52
+ <line num="117" count="1" type="stmt"/>
53
+ <line num="119" count="2" type="stmt"/>
54
+ <line num="120" count="2" type="stmt"/>
55
+ <line num="121" count="1" type="cond" truecount="1" falsecount="1"/>
56
+ <line num="122" count="0" type="cond" truecount="0" falsecount="2"/>
57
+ <line num="123" count="0" type="cond" truecount="0" falsecount="4"/>
58
+ <line num="124" count="0" type="cond" truecount="0" falsecount="4"/>
59
+ <line num="125" count="0" type="stmt"/>
60
+ <line num="126" count="0" type="stmt"/>
61
+ <line num="128" count="0" type="stmt"/>
62
+ <line num="129" count="0" type="stmt"/>
63
+ <line num="130" count="0" type="cond" truecount="0" falsecount="2"/>
64
+ <line num="131" count="0" type="stmt"/>
65
+ <line num="132" count="0" type="stmt"/>
66
+ <line num="133" count="0" type="stmt"/>
67
+ <line num="134" count="0" type="cond" truecount="0" falsecount="2"/>
68
+ <line num="136" count="0" type="stmt"/>
69
+ <line num="137" count="1" type="cond" truecount="1" falsecount="1"/>
70
+ <line num="138" count="0" type="stmt"/>
71
+ <line num="139" count="0" type="stmt"/>
72
+ <line num="140" count="0" type="stmt"/>
73
+ <line num="141" count="0" type="stmt"/>
74
+ <line num="142" count="0" type="cond" truecount="0" falsecount="2"/>
75
+ <line num="144" count="0" type="stmt"/>
76
+ <line num="145" count="1" type="cond" truecount="1" falsecount="1"/>
77
+ <line num="146" count="0" type="stmt"/>
78
+ <line num="147" count="0" type="cond" truecount="0" falsecount="2"/>
79
+ <line num="148" count="0" type="stmt"/>
80
+ <line num="149" count="0" type="stmt"/>
81
+ <line num="150" count="0" type="cond" truecount="0" falsecount="4"/>
82
+ <line num="151" count="0" type="stmt"/>
83
+ <line num="152" count="0" type="cond" truecount="0" falsecount="4"/>
84
+ <line num="153" count="0" type="cond" truecount="0" falsecount="2"/>
85
+ <line num="154" count="0" type="stmt"/>
86
+ <line num="159" count="0" type="stmt"/>
87
+ <line num="160" count="1" type="cond" truecount="1" falsecount="1"/>
88
+ <line num="161" count="0" type="stmt"/>
89
+ <line num="162" count="0" type="stmt"/>
90
+ <line num="163" count="0" type="stmt"/>
91
+ <line num="164" count="0" type="stmt"/>
92
+ <line num="165" count="0" type="stmt"/>
93
+ <line num="166" count="0" type="cond" truecount="0" falsecount="2"/>
94
+ <line num="167" count="0" type="stmt"/>
95
+ <line num="168" count="0" type="stmt"/>
96
+ <line num="169" count="0" type="stmt"/>
97
+ <line num="170" count="0" type="stmt"/>
98
+ <line num="171" count="0" type="cond" truecount="0" falsecount="2"/>
99
+ <line num="172" count="0" type="stmt"/>
100
+ <line num="179" count="0" type="stmt"/>
101
+ <line num="180" count="0" type="stmt"/>
102
+ <line num="182" count="0" type="stmt"/>
103
+ <line num="184" count="1" type="cond" truecount="1" falsecount="1"/>
104
+ <line num="185" count="0" type="stmt"/>
105
+ <line num="186" count="0" type="stmt"/>
106
+ <line num="187" count="0" type="stmt"/>
107
+ <line num="188" count="0" type="cond" truecount="0" falsecount="2"/>
108
+ <line num="189" count="0" type="stmt"/>
109
+ <line num="191" count="0" type="stmt"/>
110
+ <line num="192" count="1" type="cond" truecount="1" falsecount="1"/>
111
+ <line num="193" count="0" type="cond" truecount="0" falsecount="2"/>
112
+ <line num="194" count="0" type="cond" truecount="0" falsecount="4"/>
113
+ <line num="195" count="0" type="stmt"/>
114
+ <line num="196" count="0" type="stmt"/>
115
+ <line num="198" count="0" type="stmt"/>
116
+ <line num="201" count="0" type="stmt"/>
117
+ <line num="202" count="0" type="stmt"/>
118
+ <line num="203" count="0" type="cond" truecount="0" falsecount="2"/>
119
+ <line num="204" count="0" type="stmt"/>
120
+ <line num="207" count="0" type="stmt"/>
121
+ <line num="209" count="0" type="cond" truecount="0" falsecount="2"/>
122
+ <line num="211" count="0" type="stmt"/>
123
+ <line num="212" count="0" type="stmt"/>
124
+ <line num="215" count="0" type="stmt"/>
125
+ <line num="216" count="0" type="stmt"/>
126
+ <line num="218" count="0" type="stmt"/>
127
+ <line num="221" count="0" type="stmt"/>
128
+ <line num="223" count="0" type="stmt"/>
129
+ <line num="225" count="1" type="cond" truecount="1" falsecount="1"/>
130
+ <line num="226" count="0" type="stmt"/>
131
+ <line num="227" count="0" type="stmt"/>
132
+ <line num="228" count="0" type="stmt"/>
133
+ <line num="229" count="0" type="stmt"/>
134
+ <line num="232" count="1" type="cond" truecount="1" falsecount="1"/>
135
+ <line num="233" count="1" type="stmt"/>
136
+ <line num="234" count="1" type="cond" truecount="1" falsecount="1"/>
137
+ <line num="235" count="0" type="stmt"/>
138
+ <line num="236" count="0" type="stmt"/>
139
+ <line num="237" count="0" type="stmt"/>
140
+ <line num="240" count="1" type="cond" truecount="1" falsecount="1"/>
141
+ <line num="241" count="0" type="stmt"/>
142
+ <line num="243" count="1" type="stmt"/>
143
+ <line num="244" count="1" type="stmt"/>
144
+ <line num="245" count="1" type="stmt"/>
145
+ <line num="246" count="1" type="cond" truecount="1" falsecount="1"/>
146
+ <line num="247" count="1" type="stmt"/>
147
+ <line num="249" count="0" type="stmt"/>
148
+ <line num="250" count="0" type="stmt"/>
149
+ <line num="254" count="0" type="stmt"/>
150
+ <line num="259" count="1" type="stmt"/>
151
+ </file>
152
+ </project>
153
+ </coverage>