agentchannel 0.8.1 → 0.9.1

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.
Files changed (72) hide show
  1. package/README.md +116 -77
  2. package/dist/brain.d.ts +78 -0
  3. package/dist/brain.js +271 -0
  4. package/dist/brain.js.map +1 -0
  5. package/dist/cli.js +312 -8
  6. package/dist/cli.js.map +1 -1
  7. package/dist/config.d.ts +25 -1
  8. package/dist/config.js +104 -6
  9. package/dist/config.js.map +1 -1
  10. package/dist/crypto.d.ts +34 -4
  11. package/dist/crypto.js +42 -6
  12. package/dist/crypto.js.map +1 -1
  13. package/dist/distill.d.ts +24 -0
  14. package/dist/distill.js +404 -0
  15. package/dist/distill.js.map +1 -0
  16. package/dist/forwarder.d.ts +11 -0
  17. package/dist/forwarder.js +105 -0
  18. package/dist/forwarder.js.map +1 -0
  19. package/dist/local-store.d.ts +7 -0
  20. package/dist/local-store.js +54 -0
  21. package/dist/local-store.js.map +1 -0
  22. package/dist/mqtt-client.d.ts +11 -0
  23. package/dist/mqtt-client.js +369 -27
  24. package/dist/mqtt-client.js.map +1 -1
  25. package/dist/persistence.d.ts +23 -0
  26. package/dist/persistence.js +61 -0
  27. package/dist/persistence.js.map +1 -1
  28. package/dist/server.js +77 -3
  29. package/dist/server.js.map +1 -1
  30. package/dist/store.d.ts +3 -0
  31. package/dist/store.js +16 -2
  32. package/dist/store.js.map +1 -1
  33. package/dist/tools/brain.d.ts +2 -0
  34. package/dist/tools/brain.js +96 -0
  35. package/dist/tools/brain.js.map +1 -0
  36. package/dist/tools/channel.js +6 -6
  37. package/dist/tools/channel.js.map +1 -1
  38. package/dist/tools/get-message.js +1 -1
  39. package/dist/tools/get-message.js.map +1 -1
  40. package/dist/tools/hooks.d.ts +2 -0
  41. package/dist/tools/hooks.js +99 -0
  42. package/dist/tools/hooks.js.map +1 -0
  43. package/dist/tools/index.js +12 -0
  44. package/dist/tools/index.js.map +1 -1
  45. package/dist/tools/info.js +3 -1
  46. package/dist/tools/info.js.map +1 -1
  47. package/dist/tools/kick.d.ts +3 -0
  48. package/dist/tools/kick.js +52 -0
  49. package/dist/tools/kick.js.map +1 -0
  50. package/dist/tools/members.js +3 -3
  51. package/dist/tools/members.js.map +1 -1
  52. package/dist/tools/read.js +9 -6
  53. package/dist/tools/read.js.map +1 -1
  54. package/dist/tools/registry.d.ts +3 -0
  55. package/dist/tools/registry.js +82 -0
  56. package/dist/tools/registry.js.map +1 -0
  57. package/dist/tools/retract.d.ts +3 -0
  58. package/dist/tools/retract.js +27 -0
  59. package/dist/tools/retract.js.map +1 -0
  60. package/dist/tools/update-channel.d.ts +3 -0
  61. package/dist/tools/update-channel.js +50 -0
  62. package/dist/tools/update-channel.js.map +1 -0
  63. package/dist/types.d.ts +43 -1
  64. package/dist/web.d.ts +1 -0
  65. package/dist/web.js +91 -1
  66. package/dist/web.js.map +1 -1
  67. package/package.json +3 -2
  68. package/ui/app.js +715 -86
  69. package/ui/index.html +21 -11
  70. package/ui/marked.min.js +69 -0
  71. package/ui/mqtt.min.js +19 -0
  72. package/ui/style.css +175 -66
package/ui/style.css CHANGED
@@ -1,110 +1,156 @@
1
1
  *,*::before,*::after{margin:0;padding:0;box-sizing:border-box}
2
2
 
3
- /* Ghostty-inspired dark theme */
3
+ /* Dark theme (default) — inspired by Linear's dark-native approach */
4
4
  :root {
5
- --bg: #0e1117;
6
- --bg-alt: #1a1f2b;
7
- --bg-sidebar: #060810;
8
- --bg-bubble: #131620;
9
- --bg-bubble-self: #1a1e2a;
10
- --bg-hover: rgba(255,255,255,0.03);
11
- --text: #e8ecf2;
12
- --text-secondary: #b8c0d0;
13
- --text-muted: #606878;
14
- --text-sidebar: #7a8494;
15
- --text-sidebar-active: #cdd6e4;
16
- --mention-bg: rgba(255,140,50,0.12);
17
- --mention-text: #ff8c32;
18
- --border: #1c2030;
19
- --accent: #00e676;
20
- --sidebar-active: #1c2030;
5
+ --bg: #0a0b0f;
6
+ --bg-alt: #14161e;
7
+ --bg-sidebar: #060608;
8
+ --bg-bubble: #111318;
9
+ --bg-bubble-self: #181a22;
10
+ --bg-hover: rgba(255,255,255,0.04);
11
+ --text: #f0f1f3;
12
+ --text-secondary: #a0a8b8;
13
+ --text-body: #bcc3d0;
14
+ --text-muted: #555d6e;
15
+ --text-sidebar: #6b7585;
16
+ --text-sidebar-active: #d0d5de;
17
+ --mention-bg: rgba(0,200,88,0.08);
18
+ --mention-text: #00c858;
19
+ --border: rgba(255,255,255,0.06);
20
+ --accent: #00c858;
21
+ --accent-brand: #00c858;
22
+ --accent-brand-bg: rgba(0,200,88,0.18);
23
+ --sidebar-active: rgba(255,255,255,0.06);
24
+ --tag: #7a8da8;
25
+ --tag-bg: rgba(100,130,180,0.06);
26
+ --icon-btn-bg: rgba(255,255,255,0.08);
27
+ --icon-btn-border: rgba(255,255,255,0.12);
28
+ --icon-btn-hover: rgba(255,255,255,0.16);
29
+ --scrollbar: rgba(255,255,255,0.06);
21
30
  --font: -apple-system, BlinkMacSystemFont, "Segoe UI", "Noto Sans", Helvetica, Arial, sans-serif;
22
31
  }
23
32
 
24
- /* Light mode — follows system preference */
33
+ /* Light mode — warm Notion-inspired neutrals */
25
34
  @media (prefers-color-scheme: light) {
26
35
  :root:not(.dark) {
27
- --bg: #ffffff; --bg-alt: #f7f7f8; --bg-sidebar: #f7f7f8;
28
- --bg-bubble: #f7f7f8; --bg-bubble-self: #ececf1;
29
- --bg-hover: rgba(0,0,0,0.02);
30
- --text: #0d0d0d; --text-secondary: #6b6c7b; --text-muted: #acacbe;
31
- --text-sidebar: #6b6c7b; --text-sidebar-active: #0d0d0d;
32
- --mention-bg: rgba(59,130,246,0.08); --mention-text: #2563eb;
33
- --border: #e5e5e5; --accent: #0d0d0d; --sidebar-active: #ececf1;
36
+ --bg: #fefefe; --bg-alt: #f5f3ef; --bg-sidebar: #f7f5f2;
37
+ --bg-bubble: #f5f3ef; --bg-bubble-self: #edeae5;
38
+ --bg-hover: rgba(0,0,0,0.03);
39
+ --text: rgba(0,0,0,0.88); --text-body: rgba(0,0,0,0.65); --text-secondary: rgba(0,0,0,0.55); --text-muted: rgba(0,0,0,0.35);
40
+ --text-sidebar: rgba(0,0,0,0.50); --text-sidebar-active: rgba(0,0,0,0.85);
41
+ --mention-bg: rgba(0,200,88,0.06); --mention-text: #1a7a42;
42
+ --border: rgba(0,0,0,0.08); --accent: rgba(0,0,0,0.85); --accent-brand: #00c858;
43
+ --accent-brand-bg: rgba(0,200,88,0.08);
44
+ --sidebar-active: rgba(0,0,0,0.05);
45
+ --tag: rgba(0,0,0,0.40); --tag-bg: rgba(0,0,0,0.04);
46
+ --icon-btn-bg: rgba(0,0,0,0.04); --icon-btn-border: rgba(0,0,0,0.08);
47
+ --icon-btn-hover: rgba(0,0,0,0.08);
48
+ --scrollbar: rgba(0,0,0,0.08);
34
49
  }
35
50
  }
36
51
  /* Manual dark override */
37
52
  :root.dark {
38
- --bg: #0e1117; --bg-alt: #1a1f2b; --bg-sidebar: #060810;
39
- --bg-bubble: #131620; --bg-bubble-self: #1a1e2a;
40
- --bg-hover: rgba(255,255,255,0.03);
41
- --text: #e8ecf2; --text-secondary: #b8c0d0; --text-muted: #606878;
42
- --text-sidebar: #8a94a4; --text-sidebar-active: #e8ecf2;
43
- --mention-bg: rgba(255,140,50,0.12); --mention-text: #ff8c32;
44
- --border: #1c2030; --accent: #00e676; --sidebar-active: #1c2030;
53
+ --bg: #0a0b0f; --bg-alt: #14161e; --bg-sidebar: #060608;
54
+ --bg-bubble: #111318; --bg-bubble-self: #181a22;
55
+ --bg-hover: rgba(255,255,255,0.04);
56
+ --text: #f0f1f3; --text-body: #bcc3d0; --text-secondary: #a0a8b8; --text-muted: #555d6e;
57
+ --text-sidebar: #6b7585; --text-sidebar-active: #d0d5de;
58
+ --mention-bg: rgba(0,200,88,0.08); --mention-text: #00c858;
59
+ --border: rgba(255,255,255,0.06); --accent: #00c858; --accent-brand: #00c858;
60
+ --accent-brand-bg: rgba(0,200,88,0.18);
61
+ --sidebar-active: rgba(255,255,255,0.06);
62
+ --tag: #7a8da8; --tag-bg: rgba(100,130,180,0.06);
63
+ --icon-btn-bg: rgba(255,255,255,0.08); --icon-btn-border: rgba(255,255,255,0.12);
64
+ --icon-btn-hover: rgba(255,255,255,0.16);
65
+ --scrollbar: rgba(255,255,255,0.06);
45
66
  }
46
67
  /* Manual light override */
47
68
  :root.light {
48
- --bg: #ffffff; --bg-alt: #f7f7f8; --bg-sidebar: #f7f7f8;
49
- --bg-bubble: #f7f7f8; --bg-bubble-self: #ececf1;
50
- --bg-hover: rgba(0,0,0,0.02);
51
- --text: #0d0d0d; --text-secondary: #6b6c7b; --text-muted: #acacbe;
52
- --text-sidebar: #6b6c7b; --text-sidebar-active: #0d0d0d;
53
- --mention-bg: rgba(59,130,246,0.08); --mention-text: #2563eb;
54
- --border: #e5e5e5; --accent: #0d0d0d; --sidebar-active: #ececf1;
69
+ --bg: #fefefe; --bg-alt: #f5f3ef; --bg-sidebar: #f7f5f2;
70
+ --bg-bubble: #f5f3ef; --bg-bubble-self: #edeae5;
71
+ --bg-hover: rgba(0,0,0,0.03);
72
+ --text: rgba(0,0,0,0.88); --text-body: rgba(0,0,0,0.65); --text-secondary: rgba(0,0,0,0.55); --text-muted: rgba(0,0,0,0.35);
73
+ --text-sidebar: rgba(0,0,0,0.50); --text-sidebar-active: rgba(0,0,0,0.85);
74
+ --mention-bg: rgba(0,200,88,0.06); --mention-text: #1a7a42;
75
+ --border: rgba(0,0,0,0.08); --accent: rgba(0,0,0,0.85); --accent-brand: #00c858;
76
+ --accent-brand-bg: rgba(0,200,88,0.08);
77
+ --sidebar-active: rgba(0,0,0,0.05);
78
+ --tag: rgba(0,0,0,0.40); --tag-bg: rgba(0,0,0,0.04);
79
+ --icon-btn-bg: rgba(0,0,0,0.04); --icon-btn-border: rgba(0,0,0,0.08);
80
+ --icon-btn-hover: rgba(0,0,0,0.08);
81
+ --scrollbar: rgba(0,0,0,0.08);
55
82
  }
56
83
 
57
84
  html { font-size: 16px; -webkit-font-smoothing: antialiased; }
85
+ input, textarea, select, button { outline: none; -webkit-appearance: none; }
86
+ input:focus, textarea:focus { border-color: var(--accent) !important; }
58
87
  body { font-family: var(--font); background: var(--bg); color: var(--text); height: 100vh; overflow: hidden; }
59
88
 
60
- .app { display: flex; height: 100vh; }
89
+ .app { display: flex; height: 100vh; border-top: 1px solid var(--border); }
61
90
 
62
91
  /* Sidebar */
63
92
  .sidebar { width: 260px; background: var(--bg-sidebar); border-right: 1px solid var(--border); display: flex; flex-direction: column; flex-shrink: 0; }
64
- .sidebar__header { padding: 20px; display: flex; flex-direction: column; gap: 2px; }
93
+ .sidebar__header { padding: 16px 16px 14px 20px; display: flex; align-items: flex-start; justify-content: space-between; }
94
+ .sidebar__brand-group { display: flex; flex-direction: column; gap: 1px; }
95
+ .sidebar__collapse { background: none; border: none; cursor: pointer; color: var(--text-muted); padding: 4px; border-radius: 4px; line-height: 0; }
96
+ .sidebar__collapse:hover { color: var(--text); background: var(--bg-hover); }
65
97
  .sidebar__brand { font-size: 1.05rem; font-weight: 700; color: var(--text); letter-spacing: -0.02em; }
66
98
  .sidebar__tagline { font-size: 0.7rem; color: var(--text-muted); }
67
99
  .sidebar__channels { flex: 1; padding: 0 8px; overflow-y: auto; }
68
- .sidebar__channel { display: flex; align-items: center; padding: 5px 12px; border-radius: 6px; cursor: pointer; color: var(--text-sidebar); font-size: 0.82rem; transition: all 0.1s; margin-bottom: 0; }
100
+ .sidebar__channel { display: flex; align-items: center; padding: 6px 12px; border-radius: 6px; cursor: pointer; color: var(--text-sidebar); font-size: 0.83rem; transition: all 0.1s; margin-bottom: 1px; }
69
101
  .sidebar__channel:hover { background: var(--bg-hover); }
70
102
  .sidebar__channel.active { background: var(--sidebar-active); color: var(--text-sidebar-active); font-weight: 600; }
71
103
  .sidebar__channel .icon { width: 0; margin-right: 0; }
72
104
  .sidebar__channel .badge { margin-left: 4px; background: var(--text-muted); color: var(--bg); font-size: 0.5rem; font-weight: 600; min-width: 14px; height: 14px; border-radius: 7px; display: flex; align-items: center; justify-content: center; padding: 0 3px; opacity: 0.6; }
73
105
  .sidebar__group { padding: 12px 12px 4px; font-size: 0.65rem; font-weight: 600; text-transform: uppercase; color: var(--text-muted); letter-spacing: 0.05em; }
74
- .sidebar__channel.sub { padding-left: 28px; font-size: 0.78rem; }
106
+ .sidebar__channel.sub { padding-left: 22px; font-size: 0.78rem; }
107
+ .sidebar__create { color: var(--text-muted) !important; margin-top: 10px; }
108
+ .sidebar__create:hover { color: var(--accent) !important; }
75
109
  .sidebar__status { padding: 16px 20px; font-size: 0.75rem; color: var(--text-muted); border-top: 1px solid var(--border); }
76
- .sidebar__status.connected { color: #22c55e; }
77
- .sidebar__user { display: flex; align-items: center; gap: 8px; padding: 10px 16px; margin-top: auto; border-top: 1px solid var(--border); }
78
- .sidebar__user.connected { background: rgba(0,230,118,0.35); }
110
+ .sidebar__status.connected { color: var(--accent-brand); }
111
+ .sidebar__user { display: flex; align-items: center; gap: 8px; padding: 10px 16px; margin-top: auto; border-top: 1px solid var(--border); position: relative; overflow: hidden; }
112
+ .sidebar__user.connected { background: var(--accent-brand-bg); }
79
113
  .sidebar__user-name { font-size: 0.78rem; font-weight: 600; color: var(--text); white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
80
- .sidebar__icon-btn { background: rgba(255,255,255,0.15); border: 1px solid rgba(255,255,255,0.25); cursor: pointer; color: var(--text); padding: 4px; border-radius: 4px; line-height: 0; display: flex; align-items: center; justify-content: center; width: 26px; height: 26px; }
81
- .sidebar__icon-btn:hover { background: rgba(255,255,255,0.3); }
114
+ /* Connecting progress bar fills up, then becomes the connected bg */
115
+ .sidebar__progress { position: absolute; bottom: 0; left: 0; right: 0; top: 0; pointer-events: none; }
116
+ .sidebar__progress-bar { position: absolute; bottom: 0; left: 0; top: 0; width: 0; background: var(--accent-brand-bg); animation: progress-fill 3s ease-out forwards; }
117
+ @keyframes progress-fill { 0% { width: 0; } 80% { width: 85%; } 100% { width: 95%; } }
118
+ .sidebar__progress.connected .sidebar__progress-bar { width: 100% !important; animation: none; transition: width 0.3s ease, opacity 1s ease 0.5s; opacity: 0; }
119
+
120
+ .sidebar__icon-btn { background: transparent; border: 1px solid transparent; cursor: pointer; color: var(--text); padding: 4px; border-radius: 4px; line-height: 0; display: flex; align-items: center; justify-content: center; width: 26px; height: 26px; }
121
+ .sidebar__icon-btn:hover { background: var(--icon-btn-bg); border-color: var(--icon-btn-border); }
122
+
123
+ /* Resize handles */
124
+ .resize-handle { width: 4px; cursor: col-resize; background: transparent; flex-shrink: 0; position: relative; z-index: 10; }
125
+ .resize-handle:hover, .resize-handle.active { background: var(--border); }
82
126
 
83
127
  /* Main */
84
128
  .main { flex: 1; display: flex; flex-direction: column; min-width: 0; }
85
- .main__header { padding: 16px 24px; border-bottom: 1px solid var(--border); font-weight: 600; font-size: 1rem; display: flex; align-items: center; gap: 8px; }
129
+ .main__header { padding: 16px 24px; border-bottom: 1px solid var(--border); font-weight: 600; font-size: 1rem; display: flex; align-items: baseline; gap: 8px; }
86
130
  .main__header .channel-name { color: var(--text); }
87
131
  .main__header .channel-desc { color: var(--text-muted); font-weight: 400; font-size: 0.85rem; }
132
+ .header__actions { display: flex; align-items: center; gap: 6px; align-self: center; }
133
+ .header__btn { display: flex; align-items: center; gap: 4px; padding: 4px 10px; border: 1px solid var(--icon-btn-border); border-radius: 6px; background: var(--icon-btn-bg); color: var(--text-secondary); font-size: 0.72rem; cursor: pointer; transition: all 0.15s; }
134
+ .header__btn:hover { background: var(--text); color: var(--bg); }
135
+ .header__btn--leave:hover { background: var(--bg-hover); color: var(--text); }
88
136
 
89
137
  /* Messages */
90
138
  .messages { flex: 1; overflow-y: auto; padding: 24px 0 80px; }
91
139
  .messages__inner { max-width: 768px; margin: 0 auto; padding: 0 24px; }
92
140
 
93
- .conversation { margin-top: 16px; position: relative; }
94
- .conversation:first-child { margin-top: 0; }
95
141
  .conversation__label { display: flex; align-items: baseline; gap: 5px; margin-bottom: 1px; }
96
- .conversation__sender { font-weight: 600; font-size: 0.75rem; color: var(--text); }
97
- .conversation__channel { font-size: 0.65rem; color: var(--text-muted); }
98
- .conversation__time { font-size: 0.65rem; color: var(--text-muted); }
142
+ .conversation__sender { font-weight: 600; font-size: 0.84rem; color: var(--text); }
143
+ .conversation__channel { font-size: 0.7rem; color: var(--text-muted); }
144
+ .conversation__time { font-size: 0.7rem; color: var(--text-muted); }
99
145
 
100
- .conversation__text { font-size: 0.85rem; line-height: 1.5; color: var(--text-secondary); word-wrap: break-word; font-weight: 400; }
146
+ .conversation__text { font-size: 0.82rem; line-height: 1.55; color: var(--text-body); word-wrap: break-word; font-weight: 400; }
101
147
  .conversation__text code { background: var(--bg-alt); padding: 1px 4px; border-radius: 3px; font-size: 0.8rem; font-family: "SF Mono","Fira Code",monospace; }
102
148
  .conversation__text pre, .readme-card pre { background: var(--bg-alt); padding: 12px; border-radius: 6px; margin: 6px 0; overflow-x: auto; position: relative; }
103
149
  .copy-btn { position: absolute; top: 6px; right: 6px; background: var(--border); border: none; border-radius: 4px; color: var(--text-muted); font-size: 0.65rem; padding: 2px 6px; cursor: pointer; opacity: 0; transition: opacity 0.15s; }
104
150
  .copy-btn:hover { color: var(--text); }
105
151
  .conversation__text pre:hover .copy-btn, .readme-card pre:hover .copy-btn { opacity: 1; }
106
- .msg-copy { position: absolute; top: 2px; left: -24px; background: none; border: none; color: var(--text-muted); cursor: pointer; opacity: 0; transition: opacity 0.15s; padding: 2px; }
107
- .msg-copy:hover { color: var(--text); }
152
+ .msg-copy { position: absolute; top: 4px; right: 8px; background: var(--bg); border: 1px solid var(--border); border-radius: 4px; color: var(--text-muted); cursor: pointer; opacity: 0; transition: opacity 0.15s; padding: 2px 6px; font-size: 0.65rem; }
153
+ .msg-copy:hover { color: var(--text); background: var(--bg-hover); }
108
154
  .conversation:hover .msg-copy { opacity: 1; }
109
155
  .conversation__text pre code, .readme-card pre code { background: none; padding: 0; font-size: 0.8rem; }
110
156
  .conversation__text p { margin: 0 0 4px; }
@@ -119,7 +165,19 @@ body { font-family: var(--font); background: var(--bg); color: var(--text); heig
119
165
  .conversation__text--grouped { }
120
166
 
121
167
  .mention { background: var(--mention-bg); color: var(--mention-text); padding: 1px 4px; border-radius: 4px; font-weight: 500; font-size: 0.875rem; }
122
- .channel-tag { background: rgba(77,186,135,0.1); color: #4dba87; padding: 1px 4px; border-radius: 4px; font-weight: 500; font-size: inherit; cursor: pointer; }
168
+ .channel-tag { background: var(--mention-bg); color: var(--mention-text); padding: 1px 4px; border-radius: 4px; font-weight: 500; font-size: inherit; cursor: pointer; }
169
+
170
+ /* Subject */
171
+ .conversation__subject { font-weight: 600; font-size: 0.92rem; color: var(--text); margin-bottom: 2px; line-height: 1.4; }
172
+
173
+ /* Tags — inline text style */
174
+ .conversation__tags { margin-bottom: 4px; font-size: 0.78rem; }
175
+ .tag { color: var(--tag); font-size: 0.78rem; font-weight: 400; }
176
+
177
+ /* Message card separation — spacing only, no borders */
178
+ .conversation { margin-top: 2px; position: relative; padding: 8px 14px; border-radius: 8px; }
179
+ .conversation:hover { background: var(--bg-hover); }
180
+ .conversation:first-child { margin-top: 0; }
123
181
 
124
182
  .system-msg { text-align: center; font-size: 0.75rem; color: var(--text-muted); padding: 8px 0; }
125
183
 
@@ -130,21 +188,72 @@ body { font-family: var(--font); background: var(--bg); color: var(--text); heig
130
188
  .members__header { padding: 16px 16px 8px; font-size: 0.7rem; font-weight: 600; text-transform: uppercase; color: var(--text-muted); letter-spacing: 0.05em; }
131
189
  .members__list { flex: 1; padding: 0 8px; overflow-y: auto; }
132
190
  .members__item { display: flex; align-items: center; padding: 4px 8px; border-radius: 6px; font-size: 0.8rem; color: var(--text-secondary); gap: 8px; }
133
- .members__dot { width: 8px; height: 8px; border-radius: 50%; background: #22c55e; flex-shrink: 0; }
191
+ .members__dot { width: 8px; height: 8px; border-radius: 50%; background: var(--accent-brand); flex-shrink: 0; }
134
192
  .members__name { overflow: hidden; text-overflow: ellipsis; white-space: nowrap; }
135
193
  .members__role { font-size: 0.6rem; color: var(--text-muted); margin-left: auto; }
136
194
  .members__actions { padding: 12px 8px; border-top: 1px solid var(--border); margin-top: auto; display: flex; flex-direction: column; gap: 6px; }
137
- .members__btn { display: flex; align-items: center; gap: 6px; padding: 6px 10px; border: none; border-radius: 6px; background: var(--bg-alt); color: var(--text-secondary); font-size: 0.72rem; cursor: pointer; transition: all 0.15s; }
138
- .members__btn:hover { background: var(--border); color: var(--text); }
139
- .members__btn--leave { background: none; color: var(--text-muted); font-size: 0.68rem; }
140
- .members__btn--leave:hover { color: #ef4444; background: rgba(239,68,68,0.08); }
195
+ .members__btn { display: flex; align-items: center; gap: 6px; padding: 6px 10px; border: 1px solid var(--border); border-radius: 6px; background: var(--bg-alt); color: var(--text-secondary); font-size: 0.72rem; cursor: pointer; transition: all 0.15s; }
196
+ .members__btn:hover { background: var(--bg-hover); color: var(--text); border-color: var(--icon-btn-border); }
197
+ .members__btn--leave { border: 1px solid var(--border); background: var(--bg-alt); color: var(--text-muted); font-size: 0.68rem; }
198
+ .members__btn--leave:hover { background: var(--border); color: var(--text); }
141
199
 
142
200
  ::-webkit-scrollbar { width: 6px; }
143
201
  ::-webkit-scrollbar-track { background: transparent; }
144
- ::-webkit-scrollbar-thumb { background: rgba(128,128,128,0.15); border-radius: 3px; }
202
+ ::-webkit-scrollbar-thumb { background: var(--scrollbar); border-radius: 3px; }
203
+
204
+ /* Collapse buttons */
205
+ .collapse-btn { background: none; border: none; cursor: pointer; color: var(--text-muted); font-size: 0.9rem; padding: 6px; border-radius: 6px; line-height: 0; align-self: center; }
206
+ .collapse-btn:hover { color: var(--text); background: var(--bg-hover); }
207
+ .members-toggle { position: relative; }
208
+ .members-badge { position: absolute; top: -4px; right: -6px; min-width: 14px; height: 14px; border-radius: 7px; background: var(--text-muted); color: var(--bg); font-size: 0.5rem; font-weight: 600; display: flex; align-items: center; justify-content: center; padding: 0 3px; }
209
+ .members-badge:empty { display: none; }
210
+ .members-badge.hidden { display: none; }
211
+
212
+ /* Collapsed states */
213
+ .sidebar.collapsed { width: 56px; overflow: hidden; }
214
+ .sidebar.collapsed .sidebar__header { padding: 12px 0; justify-content: center; }
215
+ .sidebar.collapsed .sidebar__brand,
216
+ .sidebar.collapsed .sidebar__tagline,
217
+ .sidebar.collapsed .sidebar__channels,
218
+ .sidebar.collapsed #update-banner { display: none !important; }
219
+ .sidebar.collapsed .sidebar__collapse { display: none; }
220
+ .sidebar__collapse-open { display: none; }
221
+ .sidebar.collapsed .sidebar__collapse-open { display: flex; }
222
+ .sidebar.collapsed .sidebar__collapse-open { display: flex; justify-content: center; background: none; border: none; padding: 0; margin: 4px 0 0; }
223
+ .sidebar.collapsed .sidebar__user { flex-direction: column; align-items: center; gap: 6px; padding: 10px 0; background: none !important; }
224
+ .sidebar.collapsed .sidebar__user span[style*="flex:1"] { display: none; }
225
+ .sidebar.collapsed .sidebar__user-name { display: none; }
226
+ .sidebar.collapsed .sidebar__user-initial { display: flex; }
227
+ .sidebar__user-initial { display: none; width: 28px; height: 28px; border-radius: 50%; background: var(--accent-brand-bg); color: var(--text); font-size: 0.72rem; font-weight: 600; align-items: center; justify-content: center; flex-shrink: 0; }
228
+ .members.collapsed { width: 0; overflow: hidden; border: none; }
229
+ .sidebar, .members { transition: width 0.15s ease; }
230
+
231
+ /* Switch toggle (slider) */
232
+ .switch { position: relative; display: inline-block; width: 32px; height: 18px; flex-shrink: 0; }
233
+ .switch input { opacity: 0; width: 0; height: 0; }
234
+ .switch .slider { position: absolute; cursor: pointer; top: 0; left: 0; right: 0; bottom: 0; background: var(--border); border-radius: 18px; transition: 0.2s; }
235
+ .switch .slider:before { position: absolute; content: ""; height: 14px; width: 14px; left: 2px; bottom: 2px; background: var(--text-muted); border-radius: 50%; transition: 0.2s; }
236
+ .switch input:checked + .slider { background: var(--accent-brand); }
237
+ .switch input:checked + .slider:before { transform: translateX(14px); background: #fff; }
238
+
239
+ /* Channel row action icons — hidden by default, visible on hover */
240
+ .sidebar__channel-actions .sync-toggle,
241
+ .sidebar__channel-actions .sidebar__arrow { opacity: 0; transition: opacity 0.15s; }
242
+ .sidebar__channel:hover .sidebar__channel-actions .sync-toggle,
243
+ .sidebar__channel:hover .sidebar__channel-actions .sidebar__arrow { opacity: 0.6; }
244
+ .sidebar__channel:hover .sidebar__channel-actions .sync-toggle:hover,
245
+ .sidebar__channel:hover .sidebar__channel-actions .sidebar__arrow:hover { opacity: 1; }
246
+ /* Sync ON always shows a subtle indicator */
247
+ .sync-toggle[data-synced="1"] { opacity: 0.25 !important; }
248
+ .sidebar__channel:hover .sync-toggle[data-synced="1"] { opacity: 0.6 !important; }
249
+ .sidebar__channel:hover .sync-toggle[data-synced="1"]:hover { opacity: 1 !important; }
250
+
251
+ /* Retracted messages */
252
+ .retracted { text-decoration: line-through; opacity: 0.4; }
253
+ .retracted-label { font-size: 0.7rem; color: var(--text-muted); font-style: italic; margin-right: 4px; }
145
254
 
146
255
  @media (max-width: 700px) {
147
- .sidebar { width: 0; display: none; }
148
- .members { width: 0; display: none; }
256
+ .sidebar { width: 0; overflow: hidden; border: none; }
257
+ .members { width: 0; overflow: hidden; border: none; }
149
258
  .messages__inner { padding: 0 16px; }
150
259
  }