@memtensor/memos-local-openclaw-plugin 1.0.4-beta.2 → 1.0.4-beta.3

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.
@@ -106,7 +106,7 @@ input,textarea,select{font-family:inherit;font-size:inherit}
106
106
 
107
107
  /* ─── App Layout (dark dashboard, same as www) ─── */
108
108
  .app{display:none;flex-direction:column;min-height:100vh}
109
- .topbar{background:rgba(11,13,17,.88);border-bottom:1px solid var(--border);padding:0 28px;height:56px;display:flex;align-items:center;position:sticky;top:0;z-index:100;backdrop-filter:blur(12px)}
109
+ .topbar{background:rgba(11,13,17,.88);border-bottom:1px solid var(--border);padding:0 max(32px,calc((100% - 1400px)/2 + 32px));height:56px;display:flex;align-items:center;position:sticky;top:0;z-index:100;backdrop-filter:blur(12px)}
110
110
  .topbar .brand{display:flex;align-items:center;gap:8px;font-weight:700;font-size:15px;color:var(--text);letter-spacing:-.02em;flex-shrink:0}
111
111
  .topbar .brand .icon{width:24px;height:24px;display:flex;align-items:center;justify-content:center;font-size:18px;background:none;border-radius:0}
112
112
  .topbar .brand .brand-title{font-size:13px;font-weight:500;opacity:.7}
@@ -712,9 +712,7 @@ input,textarea,select{font-family:inherit;font-size:inherit}
712
712
  .settings-tab-btn.active{background:var(--bg-card);color:var(--text);font-weight:600;box-shadow:0 2px 8px rgba(0,0,0,.06),0 0 0 1px rgba(99,102,241,.1)}
713
713
  .settings-tab-btn .stab-icon{font-size:15px;line-height:1}
714
714
  .settings-tab-btn .stab-dot{width:6px;height:6px;border-radius:50%;flex-shrink:0}
715
- .settings-tab-btn[data-tab="embedding"] .stab-dot{background:#6366f1}
716
- .settings-tab-btn[data-tab="summarizer"] .stab-dot{background:#8b5cf6}
717
- .settings-tab-btn[data-tab="skill"] .stab-dot{background:#f59e0b}
715
+ .settings-tab-btn[data-tab="models"] .stab-dot{background:#6366f1}
718
716
  .settings-tab-btn[data-tab="hub"] .stab-dot{background:#06b6d4}
719
717
  .settings-tab-btn[data-tab="general"] .stab-dot{background:#10b981}
720
718
  .settings-tab-btn.active .stab-dot{box-shadow:0 0 6px currentColor}
@@ -726,9 +724,7 @@ input,textarea,select{font-family:inherit;font-size:inherit}
726
724
  .settings-card::before{content:'';position:absolute;top:0;left:0;right:0;height:3px;opacity:.5;transition:opacity .3s}
727
725
  .settings-card:hover{border-color:rgba(99,102,241,.25);box-shadow:0 0 24px rgba(99,102,241,.06),0 8px 32px rgba(0,0,0,.1)}
728
726
  .settings-card:hover::before{opacity:1}
729
- .settings-card.card-embedding::before{background:linear-gradient(90deg,#6366f1,#818cf8,#a5b4fc)}
730
- .settings-card.card-summarizer::before{background:linear-gradient(90deg,#8b5cf6,#a78bfa,#c4b5fd)}
731
- .settings-card.card-skill::before{background:linear-gradient(90deg,#f59e0b,#fbbf24,#fcd34d)}
727
+ .settings-card.card-models::before{background:linear-gradient(90deg,#6366f1,#8b5cf6,#f59e0b)}
732
728
  .settings-card.card-hub::before{background:linear-gradient(90deg,#06b6d4,#22d3ee,#67e8f9)}
733
729
  .settings-card.card-general::before{background:linear-gradient(90deg,#10b981,#34d399,#6ee7b7)}
734
730
  .settings-card-header{display:flex;align-items:center;gap:14px;padding:22px 28px 0}
@@ -802,6 +798,28 @@ input,textarea,select{font-family:inherit;font-size:inherit}
802
798
  [data-theme="light"] .settings-actions .btn-primary:hover{background:rgba(79,70,229,.1);border-color:#4f46e5}
803
799
  .settings-saved{display:inline-flex;align-items:center;gap:6px;color:var(--green);font-size:12px;font-weight:600;opacity:0;transition:opacity .3s}
804
800
  .settings-saved.show{opacity:1}
801
+ .team-guide{margin-bottom:20px;border:1px solid rgba(6,182,212,.2);border-radius:12px;background:linear-gradient(135deg,rgba(6,182,212,.04),rgba(99,102,241,.03));position:relative;overflow:hidden;padding:20px 22px 18px;display:none}
802
+ .team-guide::before{content:'';position:absolute;top:0;left:0;right:0;height:3px;background:linear-gradient(90deg,#06b6d4,#6366f1,#8b5cf6);opacity:.6}
803
+ .team-guide-title{display:flex;align-items:center;gap:8px;font-size:14px;font-weight:700;color:var(--text);margin-bottom:4px}
804
+ .team-guide-subtitle{font-size:11.5px;color:var(--text-muted);line-height:1.6;margin-bottom:16px}
805
+ .team-guide-options{display:grid;grid-template-columns:1fr 1fr;gap:14px}
806
+ @media(max-width:800px){.team-guide-options{grid-template-columns:1fr}}
807
+ .team-guide-opt{border:1px solid var(--border);border-radius:10px;padding:16px;background:var(--bg-card);transition:border-color .2s,box-shadow .2s;cursor:default}
808
+ .team-guide-opt:hover{border-color:rgba(99,102,241,.3);box-shadow:0 4px 16px rgba(0,0,0,.06)}
809
+ .team-guide-opt-header{display:flex;align-items:center;gap:10px;margin-bottom:8px}
810
+ .team-guide-opt-icon{width:32px;height:32px;display:flex;align-items:center;justify-content:center;border-radius:8px;font-size:16px;flex-shrink:0}
811
+ .team-guide-opt-title{font-size:13px;font-weight:700;color:var(--text)}
812
+ .team-guide-opt-desc{font-size:11px;color:var(--text-sec);line-height:1.7;margin-bottom:12px}
813
+ .team-guide-steps{padding:0 0 0 20px;margin:0 0 12px;counter-reset:none}
814
+ .team-guide-steps li{font-size:11px;color:var(--text-muted);line-height:1.9}
815
+ .team-guide-steps li::marker{color:var(--pri);font-weight:700;font-size:11px}
816
+ .team-guide-opt .btn-guide{font-size:11px;padding:5px 14px;border-radius:6px;font-weight:600;border:1px solid rgba(99,102,241,.25);background:rgba(99,102,241,.08);color:var(--pri);cursor:pointer;transition:background .15s,border-color .15s}
817
+ .team-guide-opt .btn-guide:hover{background:rgba(99,102,241,.14);border-color:var(--pri)}
818
+ .team-guide-dismiss{position:absolute;top:10px;right:12px;background:none;border:none;color:var(--text-muted);font-size:15px;cursor:pointer;padding:4px;line-height:1;opacity:.5;transition:opacity .15s}
819
+ .team-guide-dismiss:hover{opacity:1}
820
+ [data-theme="light"] .team-guide{background:linear-gradient(135deg,rgba(6,182,212,.03),rgba(79,70,229,.02));border-color:rgba(6,182,212,.15)}
821
+ [data-theme="light"] .team-guide-opt{box-shadow:0 1px 3px rgba(0,0,0,.03)}
822
+ [data-theme="light"] .team-guide-opt:hover{box-shadow:0 4px 16px rgba(0,0,0,.04)}
805
823
  .model-health-bar{margin-bottom:20px;border-radius:var(--radius-lg);overflow:visible}
806
824
  .mh-table{width:100%;border-collapse:separate;border-spacing:0;font-size:12px}
807
825
  .mh-table th{text-align:left;padding:6px 12px;font-size:10px;font-weight:600;color:var(--text-muted);text-transform:uppercase;letter-spacing:.05em;background:var(--bg);border-bottom:1px solid var(--border)}
@@ -861,9 +879,9 @@ input,textarea,select{font-family:inherit;font-size:inherit}
861
879
  .analytics-section::before{display:none}
862
880
  .analytics-section h3{font-size:11px;font-weight:600;color:var(--text-muted);text-transform:uppercase;letter-spacing:.08em;margin-bottom:16px;display:flex;align-items:center;gap:8px}
863
881
  .analytics-section h3 .icon{font-size:14px;opacity:.6}
864
- .chart-bars{display:flex;align-items:flex-end;gap:1px;padding:12px 4px 4px;min-height:200px;position:relative;width:100%}
882
+ .chart-bars{display:flex;align-items:flex-end;gap:0;padding:12px 0 4px;min-height:200px;position:relative;width:100%}
865
883
  .chart-bars::before{content:'';position:absolute;left:0;right:0;bottom:24px;height:1px;background:var(--border);opacity:.4}
866
- .chart-bar-wrap{flex:1 1 0;min-width:0;max-width:32px;display:flex;flex-direction:column;align-items:center;gap:4px;position:relative;cursor:pointer}
884
+ .chart-bar-wrap{flex:1 1 0;min-width:0;display:flex;flex-direction:column;align-items:center;gap:4px;position:relative;cursor:pointer}
867
885
  .chart-bar-col{width:100%;height:160px;display:flex;flex-direction:column;justify-content:flex-end;align-items:center}
868
886
  .chart-bar-wrap:hover .chart-bar{opacity:1;filter:brightness(1.2);transform:scaleY(1.02);transform-origin:bottom}
869
887
  .chart-bar-wrap:hover .chart-bar-label{color:var(--text)}
@@ -874,7 +892,7 @@ input,textarea,select{font-family:inherit;font-size:inherit}
874
892
  .chart-bar.violet{background:linear-gradient(180deg,#8b5cf6 0%,#6366f1 100%)}
875
893
  .chart-bar.green{background:linear-gradient(180deg,#34d399 0%,#10b981 100%);box-shadow:0 1px 4px rgba(16,185,129,.12)}
876
894
  .chart-bar.zero{width:100%;max-width:20px;min-width:4px;height:2px!important;background:var(--border);opacity:.25;border-radius:1px;box-shadow:none}
877
- .chart-bar-label{font-size:8px;color:var(--text-muted);white-space:nowrap;overflow:hidden;text-overflow:ellipsis;max-width:40px;text-align:center;transition:color .15s;letter-spacing:0}
895
+ .chart-bar-label{font-size:8px;line-height:1;min-height:10px;color:var(--text-muted);white-space:nowrap;overflow:hidden;text-overflow:ellipsis;max-width:40px;text-align:center;transition:color .15s;letter-spacing:0}
878
896
  .chart-legend{display:flex;gap:14px;margin-top:12px;flex-wrap:wrap;font-size:11px;color:var(--text-sec);font-weight:500}
879
897
  .chart-legend span{display:inline-flex;align-items:center;gap:5px}
880
898
  .chart-legend .dot{width:8px;height:8px;border-radius:2px}
@@ -1025,7 +1043,7 @@ input,textarea,select{font-family:inherit;font-size:inherit}
1025
1043
  <button class="tab" data-view="analytics" onclick="switchView('analytics')" data-i18n="tab.analytics">\u{1F4CA} Analytics</button>
1026
1044
  <button class="tab" data-view="logs" onclick="switchView('logs')" data-i18n="tab.logs">\u{1F4DD} Logs</button>
1027
1045
  <button class="tab" data-view="import" onclick="switchView('import')" data-i18n="tab.import">\u{1F4E5} Import</button>
1028
- <button class="tab" data-view="admin" onclick="switchView('admin')" style="display:none" data-i18n="tab.admin">\u{1F6E1} Admin</button>
1046
+ <button class="tab" data-view="admin" onclick="switchView('admin')" data-i18n="tab.admin">\u{1F6E1} Admin</button>
1029
1047
  <button class="tab" data-view="settings" onclick="switchView('settings')" data-i18n="tab.settings">\u2699 Settings</button>
1030
1048
  </nav>
1031
1049
  </div>
@@ -1265,24 +1283,25 @@ input,textarea,select{font-family:inherit;font-size:inherit}
1265
1283
  <!-- ─── Settings View ─── -->
1266
1284
  <div class="settings-view" id="settingsView">
1267
1285
  <div class="settings-tabs-bar">
1268
- <button class="settings-tab-btn active" data-tab="embedding" onclick="switchSettingsTab('embedding',this)"><span class="stab-dot"></span><span data-i18n="settings.embedding">Embedding Model</span></button>
1269
- <button class="settings-tab-btn" data-tab="summarizer" onclick="switchSettingsTab('summarizer',this)"><span class="stab-dot"></span><span data-i18n="settings.summarizer">Summarizer Model</span></button>
1270
- <button class="settings-tab-btn" data-tab="skill" onclick="switchSettingsTab('skill',this)"><span class="stab-dot"></span><span data-i18n="settings.skill">Skill Evolution</span></button>
1286
+ <button class="settings-tab-btn active" data-tab="models" onclick="switchSettingsTab('models',this)"><span class="stab-dot"></span><span data-i18n="settings.models">AI Models</span></button>
1271
1287
  <button class="settings-tab-btn" data-tab="hub" onclick="switchSettingsTab('hub',this)"><span class="stab-dot"></span><span data-i18n="settings.hub">Hub & Team</span></button>
1272
1288
  <button class="settings-tab-btn" data-tab="general" onclick="switchSettingsTab('general',this)"><span class="stab-dot"></span><span data-i18n="settings.general">General</span></button>
1273
1289
  </div>
1274
1290
  <div class="settings-cards-grid">
1275
1291
 
1276
- <!-- ══ Card: Embedding Model ══ -->
1277
- <div class="settings-card card-embedding stab-active" data-stab="embedding">
1292
+ <!-- ══ Card: AI Models (Embedding + Summarizer + Skill) ══ -->
1293
+ <div class="settings-card card-models stab-active" data-stab="models">
1278
1294
  <div class="settings-card-header">
1279
- <div class="settings-card-icon" style="background:rgba(99,102,241,.1);border-color:rgba(99,102,241,.15)">\u{1F4E1}</div>
1295
+ <div class="settings-card-icon" style="background:rgba(99,102,241,.1);border-color:rgba(99,102,241,.15)">\u{1F9E0}</div>
1280
1296
  <div class="settings-card-title-wrap">
1281
- <div class="settings-card-title" data-i18n="settings.embedding">Embedding Model</div>
1282
- <div class="settings-card-desc" data-i18n="settings.embedding.desc">Vector embedding model for memory search and retrieval</div>
1297
+ <div class="settings-card-title" data-i18n="settings.models">AI Models</div>
1298
+ <div class="settings-card-desc" data-i18n="settings.models.desc">Configure embedding, summarizer and skill evolution models</div>
1283
1299
  </div>
1284
1300
  </div>
1285
1301
  <div class="settings-card-body">
1302
+ <!-- Embedding Model section -->
1303
+ <div class="settings-card-subtitle">\u{1F4E1} <span data-i18n="settings.embedding">Embedding Model</span></div>
1304
+ <div class="field-hint" style="margin-bottom:10px" data-i18n="settings.embedding.desc">Vector embedding model for memory search and retrieval</div>
1286
1305
  <div class="settings-grid">
1287
1306
  <div class="settings-field">
1288
1307
  <label data-i18n="settings.provider">Provider</label>
@@ -1318,19 +1337,12 @@ input,textarea,select{font-family:inherit;font-size:inherit}
1318
1337
  <button class="btn btn-sm btn-ghost" onclick="testModel('embedding')" id="testEmbBtn" data-i18n="settings.test">Test Connection</button>
1319
1338
  <span class="test-result" id="testEmbResult"></span>
1320
1339
  </div>
1321
- </div>
1322
- </div>
1323
1340
 
1324
- <!-- ══ Card: Summarizer Model ══ -->
1325
- <div class="settings-card card-summarizer" data-stab="summarizer">
1326
- <div class="settings-card-header">
1327
- <div class="settings-card-icon" style="background:rgba(139,92,246,.1);border-color:rgba(139,92,246,.15)">\u{1F9E0}</div>
1328
- <div class="settings-card-title-wrap">
1329
- <div class="settings-card-title" data-i18n="settings.summarizer">Summarizer Model</div>
1330
- <div class="settings-card-desc" data-i18n="settings.summarizer.desc">LLM for memory summarization, deduplication and analysis</div>
1331
- </div>
1332
- </div>
1333
- <div class="settings-card-body">
1341
+ <div class="settings-card-divider"></div>
1342
+
1343
+ <!-- Summarizer Model section -->
1344
+ <div class="settings-card-subtitle">\u{1F4DD} <span data-i18n="settings.summarizer">Summarizer Model</span></div>
1345
+ <div class="field-hint" style="margin-bottom:10px" data-i18n="settings.summarizer.desc">LLM for memory summarization, deduplication and analysis</div>
1334
1346
  <div class="settings-grid">
1335
1347
  <div class="settings-field">
1336
1348
  <label data-i18n="settings.provider">Provider</label>
@@ -1370,19 +1382,12 @@ input,textarea,select{font-family:inherit;font-size:inherit}
1370
1382
  <button class="btn btn-sm btn-ghost" onclick="testModel('summarizer')" id="testSumBtn" data-i18n="settings.test">Test Connection</button>
1371
1383
  <span class="test-result" id="testSumResult"></span>
1372
1384
  </div>
1373
- </div>
1374
- </div>
1375
1385
 
1376
- <!-- ══ Card: Skill Evolution ══ -->
1377
- <div class="settings-card card-skill" data-stab="skill">
1378
- <div class="settings-card-header">
1379
- <div class="settings-card-icon" style="background:rgba(245,158,11,.1);border-color:rgba(245,158,11,.15)">\u{1F527}</div>
1380
- <div class="settings-card-title-wrap">
1381
- <div class="settings-card-title" data-i18n="settings.skill">Skill Evolution</div>
1382
- <div class="settings-card-desc" data-i18n="settings.skill.desc">Auto-extract reusable skills from conversation patterns</div>
1383
- </div>
1384
- </div>
1385
- <div class="settings-card-body">
1386
+ <div class="settings-card-divider"></div>
1387
+
1388
+ <!-- Skill Evolution section -->
1389
+ <div class="settings-card-subtitle">\u{1F527} <span data-i18n="settings.skill">Skill Evolution</span></div>
1390
+ <div class="field-hint" style="margin-bottom:10px" data-i18n="settings.skill.desc">Auto-extract reusable skills from conversation patterns</div>
1386
1391
  <div class="settings-grid">
1387
1392
  <div class="settings-toggle">
1388
1393
  <label class="toggle-switch"><input type="checkbox" id="cfgSkillEnabled"><span class="toggle-slider"></span></label>
@@ -1401,45 +1406,54 @@ input,textarea,select{font-family:inherit;font-size:inherit}
1401
1406
  <input type="number" id="cfgSkillMinChunks" placeholder="6">
1402
1407
  </div>
1403
1408
  </div>
1404
- <div class="settings-card-divider"></div>
1405
- <div class="settings-card-subtitle" data-i18n="settings.skill.model">Skill Dedicated Model</div>
1406
- <div class="field-hint" style="margin-bottom:12px" data-i18n="settings.skill.model.hint">If not configured, the main Summarizer Model above will be used for skill generation. Configure a dedicated model here for higher quality skill output.</div>
1407
- <div class="settings-grid">
1408
- <div class="settings-field">
1409
- <label data-i18n="settings.provider">Provider</label>
1410
- <select id="cfgSkillProvider" onchange="onProviderChange('skill')">
1411
- <option value="">\u2014 <span data-i18n="settings.skill.usemain">Use main summarizer</span> \u2014</option>
1412
- <option value="openai_compatible">OpenAI Compatible</option>
1413
- <option value="openai">OpenAI</option>
1414
- <option value="siliconflow">SiliconFlow (\u7845\u57FA\u6D41\u52A8)</option>
1415
- <option value="zhipu">Zhipu AI (\u667A\u8C31)</option>
1416
- <option value="deepseek">DeepSeek</option>
1417
- <option value="bailian">Alibaba Bailian (\u767E\u70BC)</option>
1418
- <option value="moonshot">Moonshot (Kimi)</option>
1419
- <option value="anthropic">Anthropic</option>
1420
- <option value="gemini">Gemini</option>
1421
- <option value="azure_openai">Azure OpenAI</option>
1422
- <option value="bedrock">Bedrock</option>
1423
- <option value="openclaw">OpenClaw Host</option>
1424
- </select>
1425
- </div>
1426
- <div class="settings-field">
1427
- <label data-i18n="settings.model">Model</label>
1428
- <input type="text" id="cfgSkillModel" placeholder="e.g. claude-4.6-opus">
1429
- </div>
1430
- <div class="settings-field full-width">
1431
- <label>Endpoint</label>
1432
- <input type="text" id="cfgSkillEndpoint" placeholder="https://...">
1409
+ <div style="margin-top:14px">
1410
+ <div class="settings-card-subtitle" style="margin-bottom:4px" data-i18n="settings.skill.model">Skill Dedicated Model</div>
1411
+ <div class="field-hint" style="margin-bottom:12px" data-i18n="settings.skill.model.hint">If not configured, the main Summarizer Model above will be used for skill generation. Configure a dedicated model here for higher quality skill output.</div>
1412
+ <div class="settings-grid">
1413
+ <div class="settings-field">
1414
+ <label data-i18n="settings.provider">Provider</label>
1415
+ <select id="cfgSkillProvider" onchange="onProviderChange('skill')">
1416
+ <option value="">\u2014 <span data-i18n="settings.skill.usemain">Use main summarizer</span> \u2014</option>
1417
+ <option value="openai_compatible">OpenAI Compatible</option>
1418
+ <option value="openai">OpenAI</option>
1419
+ <option value="siliconflow">SiliconFlow (\u7845\u57FA\u6D41\u52A8)</option>
1420
+ <option value="zhipu">Zhipu AI (\u667A\u8C31)</option>
1421
+ <option value="deepseek">DeepSeek</option>
1422
+ <option value="bailian">Alibaba Bailian (\u767E\u70BC)</option>
1423
+ <option value="moonshot">Moonshot (Kimi)</option>
1424
+ <option value="anthropic">Anthropic</option>
1425
+ <option value="gemini">Gemini</option>
1426
+ <option value="azure_openai">Azure OpenAI</option>
1427
+ <option value="bedrock">Bedrock</option>
1428
+ <option value="openclaw">OpenClaw Host</option>
1429
+ </select>
1430
+ </div>
1431
+ <div class="settings-field">
1432
+ <label data-i18n="settings.model">Model</label>
1433
+ <input type="text" id="cfgSkillModel" placeholder="e.g. claude-4.6-opus">
1434
+ </div>
1435
+ <div class="settings-field full-width">
1436
+ <label>Endpoint</label>
1437
+ <input type="text" id="cfgSkillEndpoint" placeholder="https://...">
1438
+ </div>
1439
+ <div class="settings-field">
1440
+ <label>API Key</label>
1441
+ <input type="password" id="cfgSkillApiKey" placeholder="\u2022\u2022\u2022\u2022\u2022\u2022\u2022\u2022">
1442
+ </div>
1433
1443
  </div>
1434
- <div class="settings-field">
1435
- <label>API Key</label>
1436
- <input type="password" id="cfgSkillApiKey" placeholder="\u2022\u2022\u2022\u2022\u2022\u2022\u2022\u2022">
1444
+ <div class="test-conn-row">
1445
+ <button class="btn btn-sm btn-ghost" onclick="testModel('skill')" id="testSkillBtn" data-i18n="settings.test">Test Connection</button>
1446
+ <span class="test-result" id="testSkillResult"></span>
1437
1447
  </div>
1438
1448
  </div>
1439
- <div class="test-conn-row">
1440
- <button class="btn btn-sm btn-ghost" onclick="testModel('skill')" id="testSkillBtn" data-i18n="settings.test">Test Connection</button>
1441
- <span class="test-result" id="testSkillResult"></span>
1449
+
1450
+ <div class="settings-card-divider"></div>
1451
+ <div class="settings-actions">
1452
+ <span class="settings-saved" id="modelsSaved">\u2713 <span data-i18n="settings.saved">Saved</span></span>
1453
+ <button class="btn btn-ghost" onclick="loadConfig()" data-i18n="settings.reset">Reset</button>
1454
+ <button class="btn btn-primary" onclick="saveModelsConfig()" data-i18n="settings.save">Save Settings</button>
1442
1455
  </div>
1456
+ <div style="font-size:11px;color:var(--text-muted);text-align:right;margin-top:4px" data-i18n="settings.restart.hint">Some changes require restarting the OpenClaw gateway to take effect.</div>
1443
1457
  </div>
1444
1458
  </div>
1445
1459
 
@@ -1453,6 +1467,43 @@ input,textarea,select{font-family:inherit;font-size:inherit}
1453
1467
  </div>
1454
1468
  </div>
1455
1469
  <div class="settings-card-body">
1470
+ <!-- team setup guide (inside Hub card) -->
1471
+ <div class="team-guide" id="teamSetupGuide">
1472
+ <button class="team-guide-dismiss" onclick="dismissTeamGuide()" title="Dismiss">&times;</button>
1473
+ <div class="team-guide-title">\u{1F680} <span data-i18n="guide.title">Get Started with Team Collaboration</span></div>
1474
+ <div class="team-guide-subtitle" data-i18n="guide.subtitle">MemOS supports team memory sharing. Choose one of the following options to enable collaboration, or continue using local-only mode.</div>
1475
+ <div class="team-guide-options">
1476
+ <div class="team-guide-opt">
1477
+ <div class="team-guide-opt-header">
1478
+ <div class="team-guide-opt-icon" style="background:rgba(6,182,212,.1);border:1px solid rgba(6,182,212,.15)">\u{1F310}</div>
1479
+ <div class="team-guide-opt-title" data-i18n="guide.join.title">Join a Remote Team</div>
1480
+ </div>
1481
+ <div class="team-guide-opt-desc" data-i18n="guide.join.desc">Your team already has a Hub running? Join it to share memories, tasks and skills with team members.</div>
1482
+ <ol class="team-guide-steps">
1483
+ <li><span data-i18n="guide.join.s1">Ask your Hub admin for the Hub Address and Team Token</span></li>
1484
+ <li><span data-i18n="guide.join.s2">Enable sharing above, select "Client" mode</span></li>
1485
+ <li><span data-i18n="guide.join.s3">Fill in Hub Address and Team Token, click "Test Connection"</span></li>
1486
+ <li><span data-i18n="guide.join.s4">Save settings and restart the OpenClaw gateway</span></li>
1487
+ </ol>
1488
+ <button class="btn-guide" onclick="guideGoToHub('client')" data-i18n="guide.join.btn">\u2192 Configure Client Mode</button>
1489
+ </div>
1490
+ <div class="team-guide-opt">
1491
+ <div class="team-guide-opt-header">
1492
+ <div class="team-guide-opt-icon" style="background:rgba(139,92,246,.1);border:1px solid rgba(139,92,246,.15)">\u{1F5A5}\uFE0F</div>
1493
+ <div class="team-guide-opt-title" data-i18n="guide.hub.title">Start Your Own Hub</div>
1494
+ </div>
1495
+ <div class="team-guide-opt-desc" data-i18n="guide.hub.desc">Be the team server. Run a Hub on this device so others can connect and share memories with you.</div>
1496
+ <ol class="team-guide-steps">
1497
+ <li><span data-i18n="guide.hub.s1">Enable sharing above, select "Hub" mode</span></li>
1498
+ <li><span data-i18n="guide.hub.s2">Set a team name, save settings, and restart the gateway</span></li>
1499
+ <li><span data-i18n="guide.hub.s3">Share the Hub Address and Team Token with your team members</span></li>
1500
+ <li><span data-i18n="guide.hub.s4">Approve join requests in the Admin Panel</span></li>
1501
+ </ol>
1502
+ <button class="btn-guide" onclick="guideGoToHub('hub')" data-i18n="guide.hub.btn">\u2192 Configure Hub Mode</button>
1503
+ </div>
1504
+ </div>
1505
+ </div>
1506
+
1456
1507
  <div class="field-hint" style="margin-bottom:12px" data-i18n="settings.hub.enable.hint">Enable to share memories, tasks and skills with your team. When disabled, all features work normally in local-only mode.</div>
1457
1508
  <div class="settings-toggle" style="margin-bottom:16px">
1458
1509
  <label class="toggle-switch">
@@ -1526,6 +1577,14 @@ input,textarea,select{font-family:inherit;font-size:inherit}
1526
1577
  <div id="sharingStatusPanel"></div>
1527
1578
  <div id="sharingTeamPanel"></div>
1528
1579
  <div id="sharingAdminPanel"></div>
1580
+
1581
+ <div class="settings-card-divider"></div>
1582
+ <div class="settings-actions">
1583
+ <span class="settings-saved" id="hubSaved">\u2713 <span data-i18n="settings.saved">Saved</span></span>
1584
+ <button class="btn btn-ghost" onclick="loadConfig()" data-i18n="settings.reset">Reset</button>
1585
+ <button class="btn btn-primary" onclick="saveHubConfig()" data-i18n="settings.save">Save Settings</button>
1586
+ </div>
1587
+ <div style="font-size:11px;color:var(--text-muted);text-align:right;margin-top:4px" data-i18n="settings.restart.hint">Some changes require restarting the OpenClaw gateway to take effect.</div>
1529
1588
  </div>
1530
1589
  </div>
1531
1590
 
@@ -1557,39 +1616,44 @@ input,textarea,select{font-family:inherit;font-size:inherit}
1557
1616
  <label data-i18n="settings.telemetry.enabled">Enable Anonymous Telemetry</label>
1558
1617
  </div>
1559
1618
  <div class="field-hint" style="margin-top:6px" data-i18n="settings.telemetry.hint">Anonymous usage analytics to help improve the plugin. Only sends tool names, latencies, and version info. No memory content, queries, or personal data is ever sent.</div>
1619
+
1620
+ <div class="settings-card-divider"></div>
1621
+ <div class="settings-actions">
1622
+ <span class="settings-saved" id="generalSaved">\u2713 <span data-i18n="settings.saved">Saved</span></span>
1623
+ <button class="btn btn-ghost" onclick="loadConfig()" data-i18n="settings.reset">Reset</button>
1624
+ <button class="btn btn-primary" onclick="saveGeneralConfig()" data-i18n="settings.save">Save Settings</button>
1625
+ </div>
1560
1626
  </div>
1561
1627
  </div>
1562
1628
 
1563
1629
  </div>
1564
1630
 
1565
- <div class="settings-actions">
1566
- <span class="settings-saved" id="settingsSaved">\u2713 <span data-i18n="settings.saved">Saved</span></span>
1567
- <button class="btn btn-ghost" onclick="loadConfig()" data-i18n="settings.reset">Reset</button>
1568
- <button class="btn btn-primary" onclick="saveConfig()" data-i18n="settings.save">Save Settings</button>
1569
- </div>
1570
- <div style="font-size:11px;color:var(--text-muted);text-align:right;margin-top:4px" data-i18n="settings.restart.hint">Some changes require restarting the OpenClaw gateway to take effect.</div>
1631
+
1571
1632
  </div>
1572
1633
 
1573
1634
  <!-- ─── Admin Page ─── -->
1574
1635
  <div class="admin-view" id="adminView">
1575
- <div class="admin-header">
1576
- <div class="admin-header-top">
1577
- <h2><span class="ah-icon">\u{1F6E1}</span> <span data-i18n="admin.title">Hub Admin Panel</span></h2>
1578
- <button class="btn btn-sm btn-ghost" onclick="loadAdminData()" style="backdrop-filter:blur(8px)" data-i18n="admin.refresh">\u21BB Refresh</button>
1636
+ <div id="adminNotEnabled" style="display:none"></div>
1637
+ <div id="adminMainContent">
1638
+ <div class="admin-header">
1639
+ <div class="admin-header-top">
1640
+ <h2><span class="ah-icon">\u{1F6E1}</span> <span data-i18n="admin.title">Hub Admin Panel</span></h2>
1641
+ <button class="btn btn-sm btn-ghost" onclick="loadAdminData()" style="backdrop-filter:blur(8px)" data-i18n="admin.refresh">\u21BB Refresh</button>
1642
+ </div>
1643
+ <div class="admin-header-sub" data-i18n="admin.subtitle">Manage team members, groups, and shared resources</div>
1644
+ <div class="admin-stat-row" id="adminStats"></div>
1579
1645
  </div>
1580
- <div class="admin-header-sub" data-i18n="admin.subtitle">Manage team members, groups, and shared resources</div>
1581
- <div class="admin-stat-row" id="adminStats"></div>
1582
- </div>
1583
- <div class="admin-tabs" id="adminTabsBar">
1584
- <button class="admin-tab active" onclick="switchAdminTab('users',this)"><span class="at-icon">\u{1F465}</span> <span data-i18n="admin.tab.users">Users</span> <span class="at-count" id="adminTabCountUsers">0</span></button>
1585
- <button class="admin-tab" onclick="switchAdminTab('sharedMemories',this)"><span class="at-icon">\u{1F4AD}</span> <span data-i18n="admin.tab.sharedMemories">Shared Memories</span> <span class="at-count" id="adminTabCountMemories">0</span></button>
1586
- <button class="admin-tab" onclick="switchAdminTab('memories',this)"><span class="at-icon">\u{1F4CB}</span> <span data-i18n="admin.tab.memories">Shared Tasks</span> <span class="at-count" id="adminTabCountTasks">0</span></button>
1587
- <button class="admin-tab" onclick="switchAdminTab('skills',this)"><span class="at-icon">\u{1F9E0}</span> <span data-i18n="admin.tab.skills">Shared Skills</span> <span class="at-count" id="adminTabCountSkills">0</span></button>
1646
+ <div class="admin-tabs" id="adminTabsBar">
1647
+ <button class="admin-tab active" onclick="switchAdminTab('users',this)"><span class="at-icon">\u{1F465}</span> <span data-i18n="admin.tab.users">Users</span> <span class="at-count" id="adminTabCountUsers">0</span></button>
1648
+ <button class="admin-tab" onclick="switchAdminTab('sharedMemories',this)"><span class="at-icon">\u{1F4AD}</span> <span data-i18n="admin.tab.sharedMemories">Shared Memories</span> <span class="at-count" id="adminTabCountMemories">0</span></button>
1649
+ <button class="admin-tab" onclick="switchAdminTab('memories',this)"><span class="at-icon">\u{1F4CB}</span> <span data-i18n="admin.tab.memories">Shared Tasks</span> <span class="at-count" id="adminTabCountTasks">0</span></button>
1650
+ <button class="admin-tab" onclick="switchAdminTab('skills',this)"><span class="at-icon">\u{1F9E0}</span> <span data-i18n="admin.tab.skills">Shared Skills</span> <span class="at-count" id="adminTabCountSkills">0</span></button>
1651
+ </div>
1652
+ <div class="admin-panel active" id="adminUsersPanel"></div>
1653
+ <div class="admin-panel" id="adminSharedMemoriesPanel"></div>
1654
+ <div class="admin-panel" id="adminMemoriesPanel"></div>
1655
+ <div class="admin-panel" id="adminSkillsPanel"></div>
1588
1656
  </div>
1589
- <div class="admin-panel active" id="adminUsersPanel"></div>
1590
- <div class="admin-panel" id="adminSharedMemoriesPanel"></div>
1591
- <div class="admin-panel" id="adminMemoriesPanel"></div>
1592
- <div class="admin-panel" id="adminSkillsPanel"></div>
1593
1657
  </div>
1594
1658
 
1595
1659
  <!-- ─── Import Page ─── -->
@@ -1958,6 +2022,8 @@ const I18N={
1958
2022
  'tab.import':'\u{1F4E5} Import',
1959
2023
  'tab.settings':'\u2699 Settings',
1960
2024
  'settings.modelconfig':'Model Configuration',
2025
+ 'settings.models':'AI Models',
2026
+ 'settings.models.desc':'Configure embedding, summarizer and skill evolution models',
1961
2027
  'settings.modelhealth':'Model Health',
1962
2028
  'settings.embedding':'Embedding Model',
1963
2029
  'settings.summarizer':'Summarizer Model',
@@ -2344,7 +2410,36 @@ const I18N={
2344
2410
  'update.success':'Updated!',
2345
2411
  'update.failed':'Update failed',
2346
2412
  'update.restarting':'Restarting service...',
2347
- 'update.dismiss':'Dismiss'
2413
+ 'update.dismiss':'Dismiss',
2414
+ 'sharing.disable.confirm.hub':'You are about to shut down the Hub server.\\n\\nWhat will happen:\\n\\u2022 All connected team members will be disconnected\\n\\u2022 They will no longer be able to sync memories, tasks, or skills\\n\\u2022 Shared data is preserved and will be available when you re-enable\\n\\nAre you sure?',
2415
+ 'sharing.disable.confirm.client':'You are about to disconnect from the team Hub.\\n\\nWhat will happen:\\n\\u2022 You will no longer receive shared memories, tasks, or skills from the team\\n\\u2022 Your local data is preserved and will not be affected\\n\\u2022 You can reconnect later by re-enabling sharing\\n\\nAre you sure?',
2416
+ 'sharing.disable.restartAlert':'Sharing has been disabled. Please restart the OpenClaw gateway for the change to take effect.\\n\\nRun: openclaw gateway stop && openclaw gateway start',
2417
+ 'admin.notEnabled.title':'Team sharing is not enabled',
2418
+ 'admin.notEnabled.desc':'The Admin Panel is used to manage team members, shared memories, tasks, and skills. To use this feature, you need to enable Hub sharing first.',
2419
+ 'admin.notEnabled.setupHub':'Set Up as Hub Server',
2420
+ 'admin.notEnabled.joinTeam':'Join an Existing Team',
2421
+ 'admin.notEnabled.hint':'If you have previously configured sharing, your data is still preserved. Re-enabling sharing will restore access to all shared content.',
2422
+ 'sharing.disconnected.hint':'Unable to reach the Hub server. The Hub may be offline or the network is unavailable.',
2423
+ 'sharing.retryConnection':'Retry Connection',
2424
+ 'sharing.retryConnection.loading':'Connecting...',
2425
+ 'sharing.retryConnection.success':'Connected successfully!',
2426
+ 'sharing.retryConnection.fail':'Still unable to connect. Check if the Hub is online.',
2427
+ 'guide.title':'Get Started with Team Collaboration',
2428
+ 'guide.subtitle':'MemOS supports team memory sharing. Choose one of the following options to enable collaboration, or continue using local-only mode.',
2429
+ 'guide.join.title':'Join a Remote Team',
2430
+ 'guide.join.desc':'Your team already has a Hub running? Join it to share memories, tasks and skills with team members.',
2431
+ 'guide.join.s1':'Ask your Hub admin for the Hub Address and Team Token',
2432
+ 'guide.join.s2':'Go to Settings \u2192 Hub & Team, enable sharing, select "Client" mode',
2433
+ 'guide.join.s3':'Fill in Hub Address and Team Token, click "Test Connection"',
2434
+ 'guide.join.s4':'Save settings and restart the OpenClaw gateway',
2435
+ 'guide.join.btn':'\u2192 Configure Client Mode',
2436
+ 'guide.hub.title':'Start Your Own Hub',
2437
+ 'guide.hub.desc':'Be the team server. Run a Hub on this device so others can connect and share memories with you.',
2438
+ 'guide.hub.s1':'Go to Settings \u2192 Hub & Team, enable sharing, select "Hub" mode',
2439
+ 'guide.hub.s2':'Set a team name, save settings, and restart the gateway',
2440
+ 'guide.hub.s3':'Share the Hub Address and Team Token with your team members',
2441
+ 'guide.hub.s4':'Approve join requests in the Admin Panel',
2442
+ 'guide.hub.btn':'\u2192 Configure Hub Mode'
2348
2443
  },
2349
2444
  zh:{
2350
2445
  'title':'MemOS 记忆',
@@ -2514,6 +2609,8 @@ const I18N={
2514
2609
  'tab.import':'\u{1F4E5} 导入',
2515
2610
  'tab.settings':'\u2699 设置',
2516
2611
  'settings.modelconfig':'模型配置',
2612
+ 'settings.models':'AI 模型',
2613
+ 'settings.models.desc':'配置嵌入模型、摘要模型和技能进化模型',
2517
2614
  'settings.modelhealth':'模型健康',
2518
2615
  'settings.embedding':'嵌入模型',
2519
2616
  'settings.summarizer':'摘要模型',
@@ -2900,7 +2997,36 @@ const I18N={
2900
2997
  'update.success':'更新完成',
2901
2998
  'update.failed':'更新失败',
2902
2999
  'update.restarting':'正在重启服务...',
2903
- 'update.dismiss':'关闭'
3000
+ 'update.dismiss':'关闭',
3001
+ 'sharing.disable.confirm.hub':'你即将关闭 Hub 服务。\\n\\n关闭后将会:\\n\\u2022 所有已连接的团队成员将断开连接\\n\\u2022 他们将无法继续同步记忆、任务和技能\\n\\u2022 已共享的数据会保留,重新开启后仍可使用\\n\\n确定要关闭吗?',
3002
+ 'sharing.disable.confirm.client':'你即将断开与团队 Hub 的连接。\\n\\n断开后将会:\\n\\u2022 你将无法再接收团队共享的记忆、任务和技能\\n\\u2022 你的本地数据不受影响,会完整保留\\n\\u2022 之后可以随时重新开启共享来恢复连接\\n\\n确定要断开吗?',
3003
+ 'sharing.disable.restartAlert':'共享已关闭。请重启 OpenClaw gateway 使更改生效。\\n\\n执行命令:openclaw gateway stop && openclaw gateway start',
3004
+ 'admin.notEnabled.title':'团队共享尚未开启',
3005
+ 'admin.notEnabled.desc':'管理面板用于管理团队成员、共享的记忆、任务和技能。使用此功能前,需要先开启 Hub 共享。',
3006
+ 'admin.notEnabled.setupHub':'配置为 Hub 服务端',
3007
+ 'admin.notEnabled.joinTeam':'加入已有团队',
3008
+ 'admin.notEnabled.hint':'如果之前配置过共享,你的数据仍然保留。重新开启共享即可恢复访问所有共享内容。',
3009
+ 'sharing.disconnected.hint':'无法连接到 Hub 服务器,Hub 可能已下线或网络不可用。',
3010
+ 'sharing.retryConnection':'重试连接',
3011
+ 'sharing.retryConnection.loading':'连接中...',
3012
+ 'sharing.retryConnection.success':'连接成功!',
3013
+ 'sharing.retryConnection.fail':'仍然无法连接,请检查 Hub 是否在线。',
3014
+ 'guide.title':'开始团队协作',
3015
+ 'guide.subtitle':'MemOS 支持团队记忆共享。选择以下方式之一开启协作,或继续使用纯本地模式。',
3016
+ 'guide.join.title':'加入远程团队',
3017
+ 'guide.join.desc':'你的团队已有 Hub 在运行?加入即可与团队成员共享记忆、任务和技能。',
3018
+ 'guide.join.s1':'向 Hub 管理员索取 Hub 地址和 Team Token',
3019
+ 'guide.join.s2':'前往「设置 → Hub & Team」,开启共享,选择「Client」模式',
3020
+ 'guide.join.s3':'填写 Hub 地址和 Team Token,点击「测试连接」',
3021
+ 'guide.join.s4':'保存设置并重启 OpenClaw gateway',
3022
+ 'guide.join.btn':'\u2192 配置 Client 模式',
3023
+ 'guide.hub.title':'自建 Hub 服务',
3024
+ 'guide.hub.desc':'将本机作为团队服务端,让其他成员连接过来共享记忆。',
3025
+ 'guide.hub.s1':'前往「设置 → Hub & Team」,开启共享,选择「Hub」模式',
3026
+ 'guide.hub.s2':'设置团队名称,保存设置后重启 gateway',
3027
+ 'guide.hub.s3':'将 Hub 地址和 Team Token 分享给团队成员',
3028
+ 'guide.hub.s4':'在管理面板中审批加入请求',
3029
+ 'guide.hub.btn':'\u2192 配置 Hub 模式'
2904
3030
  }
2905
3031
  };
2906
3032
  const LANG_KEY='memos-viewer-lang';
@@ -3025,8 +3151,6 @@ function selectSharingRole(role){
3025
3151
  if(sp) sp.style.display='none';
3026
3152
  if(tp) tp.style.display='none';
3027
3153
  if(ap) ap.style.display='none';
3028
- var adminTab=document.querySelector('.tab[data-view="admin"]');
3029
- if(adminTab) adminTab.style.display='none';
3030
3154
  }else{
3031
3155
  if(sp) sp.style.display='';
3032
3156
  if(tp) tp.style.display='';
@@ -3183,10 +3307,12 @@ async function loadSharingStatus(forcePending){
3183
3307
  sharingStatusCache=d;
3184
3308
  renderSharingSidebar(d);
3185
3309
  renderSharingSettings(d);
3310
+ updateTeamGuide(d);
3186
3311
  if(forcePending && d && d.admin && d.admin.canManageUsers) loadSharingPendingUsers();
3187
3312
  }catch(e){
3188
3313
  renderSharingSidebar(null);
3189
3314
  renderSharingSettings(null);
3315
+ updateTeamGuide(null);
3190
3316
  }
3191
3317
  }
3192
3318
 
@@ -3211,8 +3337,6 @@ function renderSharingSidebar(data){
3211
3337
  setBadge('#34d399',t('sharing.sidebar.connected'),true);
3212
3338
  statusEl.innerHTML='';
3213
3339
  hintEl.textContent='';
3214
- var adminTab=document.querySelector('.tab[data-view="admin"]');
3215
- if(adminTab) adminTab.style.display='';
3216
3340
  }else if(conn.pendingApproval&&conn.user){
3217
3341
  setBadge('#fbbf24',t('sharing.sidebar.pending'),false);
3218
3342
  var html='<div class="info-grid">';
@@ -3221,8 +3345,6 @@ function renderSharingSidebar(data){
3221
3345
  html+='</div>';
3222
3346
  statusEl.innerHTML=html;
3223
3347
  hintEl.textContent=t('sharing.pendingApproval.hint');
3224
- var adminTab=document.querySelector('.tab[data-view="admin"]');
3225
- if(adminTab) adminTab.style.display='none';
3226
3348
  }else if(conn.rejected&&conn.user){
3227
3349
  setBadge('#ef4444',t('sharing.sidebar.rejected'),false);
3228
3350
  var html='<div class="info-grid">';
@@ -3231,8 +3353,6 @@ function renderSharingSidebar(data){
3231
3353
  html+='</div>';
3232
3354
  statusEl.innerHTML=html;
3233
3355
  hintEl.textContent=t('sharing.rejected.hint');
3234
- var adminTab=document.querySelector('.tab[data-view="admin"]');
3235
- if(adminTab) adminTab.style.display='none';
3236
3356
  }else if(conn.connected&&conn.user){
3237
3357
  var isAdmin=conn.user.role==='admin';
3238
3358
  setBadge('#34d399',t('sharing.sidebar.connected'),true);
@@ -3242,12 +3362,10 @@ function renderSharingSidebar(data){
3242
3362
  html+='</div>';
3243
3363
  statusEl.innerHTML=html;
3244
3364
  hintEl.innerHTML='';
3245
- var adminTab=document.querySelector('.tab[data-view="admin"]');
3246
- if(adminTab) adminTab.style.display=isAdmin?'':'none';
3247
3365
  }else if(data.clientConfigured){
3248
3366
  setBadge('#ef4444',t('sharing.sidebar.disconnected'),false);
3249
3367
  statusEl.innerHTML='<div style="font-size:11px;color:var(--text-muted)">'+t('sharing.sidebar.targetHub')+' '+esc(data.hubUrl||'')+'</div>';
3250
- hintEl.textContent=t('sharing.clientDisconnected.hint');
3368
+ hintEl.innerHTML=esc(t('sharing.clientDisconnected.hint'))+'<br><a href="#" onclick="retryConnection();return false;" style="color:var(--pri);font-size:11px;text-decoration:none">'+t('sharing.retryConnection')+'</a>';
3251
3369
  }else{
3252
3370
  setBadge('#888',t('sharing.sidebar.notConfigured'),false);
3253
3371
  statusEl.innerHTML='';
@@ -3270,8 +3388,6 @@ function renderSharingSettings(data){
3270
3388
  if(data.role) _sharingRole=data.role;
3271
3389
  var isAdmin=(data.admin&&data.admin.canManageUsers)||(conn.connected&&user.role==='admin')||(actualRole==='hub');
3272
3390
  window._isHubAdmin=isAdmin;
3273
- var adminTab=document.querySelector('.tab[data-view="admin"]');
3274
- if(adminTab) adminTab.style.display=isAdmin?'':'none';
3275
3391
  var hubAdminBtn=document.getElementById('hubAdminEntryBtn');
3276
3392
 
3277
3393
  if(actualRole==='hub'){
@@ -3282,7 +3398,6 @@ function renderSharingSettings(data){
3282
3398
 
3283
3399
  if(actualRole==='client'){
3284
3400
  statusEl.style.display='none';teamEl.style.display='none';adminEl.style.display='none';
3285
- if(adminTab) adminTab.style.display='none';
3286
3401
  if(hubAdminBtn) hubAdminBtn.style.display='none';
3287
3402
 
3288
3403
  var connBadge;
@@ -3313,7 +3428,9 @@ function renderSharingSettings(data){
3313
3428
  sh+='<span class="hic-label">'+t('sharing.team')+'</span><span class="hic-value">'+esc(conn.teamName||'-')+'</span>';
3314
3429
  sh+='</div></div>';
3315
3430
  }else{
3316
- sh+='</div></div>';
3431
+ sh+='</div><div class="hic-empty" style="color:var(--text-muted)">'+t('sharing.disconnected.hint')+'</div>'+
3432
+ '<div style="margin-top:10px;padding:0 16px 14px"><button class="btn btn-sm btn-primary" id="btnRetryConn" onclick="retryConnection()">'+t('sharing.retryConnection')+'</button>'+
3433
+ '<span id="retryConnResult" style="margin-left:10px;font-size:11px"></span></div></div>';
3317
3434
  }
3318
3435
  statusEl.innerHTML=sh;
3319
3436
  teamEl.innerHTML='';adminEl.innerHTML='';
@@ -3323,6 +3440,26 @@ function renderSharingSettings(data){
3323
3440
  statusEl.innerHTML='';teamEl.innerHTML='';adminEl.innerHTML='';
3324
3441
  }
3325
3442
 
3443
+ async function retryConnection(){
3444
+ var btn=document.getElementById('btnRetryConn');
3445
+ var result=document.getElementById('retryConnResult');
3446
+ if(btn){btn.disabled=true;btn.textContent=t('sharing.retryConnection.loading');}
3447
+ if(result) result.innerHTML='<span style="color:var(--text-muted)">'+t('sharing.retryConnection.loading')+'</span>';
3448
+ try{
3449
+ await loadSharingStatus(false);
3450
+ var d=sharingStatusCache;
3451
+ if(d&&d.connection&&d.connection.connected){
3452
+ toast(t('sharing.retryConnection.success'),'success');
3453
+ if(result) result.innerHTML='<span style="color:#22c55e">\\u2705 '+t('sharing.retryConnection.success')+'</span>';
3454
+ }else{
3455
+ if(result) result.innerHTML='<span style="color:#ef4444">'+t('sharing.retryConnection.fail')+'</span>';
3456
+ }
3457
+ }catch(e){
3458
+ if(result) result.innerHTML='<span style="color:#ef4444">'+t('sharing.retryConnection.fail')+'</span>';
3459
+ }
3460
+ if(btn){btn.disabled=false;btn.textContent=t('sharing.retryConnection');}
3461
+ }
3462
+
3326
3463
  async function retryHubJoin(){
3327
3464
  if(!confirm(t('sharing.retryJoin.confirm'))) return;
3328
3465
  try{
@@ -3421,6 +3558,29 @@ async function rejectSharingUser(userId,username){
3421
3558
  }catch(e){toast(t('toast.rejectFail')+': '+e.message,'error');}
3422
3559
  }
3423
3560
 
3561
+ /* ─── Team Setup Guide ─── */
3562
+ var TEAM_GUIDE_DISMISSED_KEY='memos-team-guide-dismissed';
3563
+ function updateTeamGuide(sharingData){
3564
+ var el=document.getElementById('teamSetupGuide');
3565
+ if(!el) return;
3566
+ if(localStorage.getItem(TEAM_GUIDE_DISMISSED_KEY)==='1'){el.style.display='none';return;}
3567
+ var isConfigured=sharingData&&sharingData.enabled;
3568
+ el.style.display=isConfigured?'none':'block';
3569
+ }
3570
+ function dismissTeamGuide(){
3571
+ localStorage.setItem(TEAM_GUIDE_DISMISSED_KEY,'1');
3572
+ var el=document.getElementById('teamSetupGuide');
3573
+ if(el) el.style.display='none';
3574
+ }
3575
+ function guideGoToHub(role){
3576
+ switchSettingsTab('hub',document.querySelector('.settings-tab-btn[data-tab="hub"]'));
3577
+ var chk=document.getElementById('cfgSharingEnabled');
3578
+ if(chk&&!chk.checked){chk.checked=true;onSharingToggle();}
3579
+ selectSharingRole(role);
3580
+ var card=document.getElementById('settingsSharingConfig');
3581
+ if(card) card.scrollIntoView({behavior:'smooth',block:'start'});
3582
+ }
3583
+
3424
3584
  /* ─── Group Manager ─── */
3425
3585
  var groupManagerUsers=[];
3426
3586
  async function loadGroupManager(){
@@ -3575,7 +3735,35 @@ function switchAdminTab(tab,btn){
3575
3735
  if(panel) panel.classList.add('active');
3576
3736
  }
3577
3737
 
3738
+ function adminGoSetup(role){
3739
+ switchView('settings');
3740
+ setTimeout(function(){guideGoToHub(role);},200);
3741
+ }
3742
+
3578
3743
  async function loadAdminData(){
3744
+ var notEnabledEl=document.getElementById('adminNotEnabled');
3745
+ var mainEl=document.getElementById('adminMainContent');
3746
+ var sharingOn=sharingStatusCache&&sharingStatusCache.enabled;
3747
+ if(!sharingOn){
3748
+ if(mainEl) mainEl.style.display='none';
3749
+ if(notEnabledEl){
3750
+ notEnabledEl.style.display='block';
3751
+ notEnabledEl.innerHTML=
3752
+ '<div style="text-align:center;padding:60px 32px;max-width:520px;margin:0 auto">'+
3753
+ '<div style="font-size:48px;margin-bottom:16px">\u{1F6E1}</div>'+
3754
+ '<div style="font-size:18px;font-weight:700;color:var(--text);margin-bottom:8px">'+t('admin.notEnabled.title')+'</div>'+
3755
+ '<div style="font-size:13px;color:var(--text-sec);line-height:1.7;margin-bottom:24px">'+t('admin.notEnabled.desc')+'</div>'+
3756
+ '<div style="display:flex;gap:12px;justify-content:center;flex-wrap:wrap">'+
3757
+ '<button class="btn btn-primary" style="padding:8px 20px;font-size:13px" onclick="adminGoSetup(&quot;hub&quot;)">'+t('admin.notEnabled.setupHub')+'</button>'+
3758
+ '<button class="btn btn-ghost" style="padding:8px 20px;font-size:13px" onclick="adminGoSetup(&quot;client&quot;)">'+t('admin.notEnabled.joinTeam')+'</button>'+
3759
+ '</div>'+
3760
+ '<div style="font-size:11px;color:var(--text-muted);margin-top:20px;line-height:1.6">'+t('admin.notEnabled.hint')+'</div>'+
3761
+ '</div>';
3762
+ }
3763
+ return;
3764
+ }
3765
+ if(notEnabledEl) notEnabledEl.style.display='none';
3766
+ if(mainEl) mainEl.style.display='';
3579
3767
  if(!window._isHubAdmin){
3580
3768
  var statsEl=document.getElementById('adminStats');
3581
3769
  if(statsEl) statsEl.innerHTML='<div class="admin-empty">'+t('admin.noPermission')+'</div>';
@@ -5271,9 +5459,36 @@ function onProviderChange(section){
5271
5459
  if(m[2]==='chat'&&def.chatModel&&!mdEl.value.trim()) mdEl.value=def.chatModel;
5272
5460
  }
5273
5461
 
5274
- async function saveConfig(){
5275
- var saveBtn=document.querySelector('.settings-actions .btn-primary');
5462
+ function flashSaved(id){
5463
+ var el=document.getElementById(id);
5464
+ if(!el)return;
5465
+ el.classList.add('show');
5466
+ setTimeout(function(){el.classList.remove('show');},2500);
5467
+ }
5468
+
5469
+ async function doSaveConfig(cfg, btnEl, savedId){
5470
+ btnEl.disabled=true;btnEl.textContent=t('settings.test.loading');
5471
+ function done(){btnEl.disabled=false;btnEl.textContent=t('settings.save');}
5472
+ try{
5473
+ const r=await fetch('/api/config',{method:'PUT',headers:{'Content-Type':'application/json'},body:JSON.stringify(cfg)});
5474
+ if(r.status===401){done();toast(t('settings.session.expired'),'error');return false;}
5475
+ if(!r.ok) throw new Error(await r.text());
5476
+ flashSaved(savedId);
5477
+ toast(t('settings.saved'),'success');
5478
+ done();
5479
+ return true;
5480
+ }catch(e){
5481
+ toast(t('settings.save.fail')+': '+e.message,'error');
5482
+ done();
5483
+ return false;
5484
+ }
5485
+ }
5486
+
5487
+ async function saveModelsConfig(){
5488
+ var card=document.querySelector('.card-models');
5489
+ var saveBtn=card.querySelector('.settings-actions .btn-primary');
5276
5490
  saveBtn.disabled=true;saveBtn.textContent=t('settings.test.loading');
5491
+ function done(){saveBtn.disabled=false;saveBtn.textContent=t('settings.save');}
5277
5492
 
5278
5493
  const cfg={};
5279
5494
  const embP=document.getElementById('cfgEmbProvider').value;
@@ -5314,52 +5529,8 @@ async function saveConfig(){
5314
5529
  if(skApiKey) cfg.skillEvolution.summarizer.apiKey=skApiKey;
5315
5530
  }
5316
5531
 
5317
- const vp=document.getElementById('cfgViewerPort').value.trim();
5318
- if(vp) cfg.viewerPort=Number(vp);
5319
- cfg.telemetry={enabled:document.getElementById('cfgTelemetryEnabled').checked};
5320
-
5321
- function done(){saveBtn.disabled=false;saveBtn.textContent=t('settings.save');}
5322
-
5323
- var sharingEnabled=document.getElementById('cfgSharingEnabled').checked;
5324
- cfg.sharing={
5325
- enabled:sharingEnabled,
5326
- role:_sharingRole,
5327
- capabilities:{}
5328
- };
5329
- if(sharingEnabled&&_sharingRole==='hub'){
5330
- var hubPort=document.getElementById('cfgHubPort').value.trim();
5331
- var hubTeamName=document.getElementById('cfgHubTeamName').value.trim();
5332
- var hubTeamToken=document.getElementById('cfgHubTeamToken').value.trim();
5333
- cfg.sharing.hub={port:hubPort?Number(hubPort):18800};
5334
- if(hubTeamName) cfg.sharing.hub.teamName=hubTeamName;
5335
- if(hubTeamToken) cfg.sharing.hub.teamToken=hubTeamToken;
5336
- cfg.sharing.client={hubAddress:'',userToken:'',teamToken:''};
5337
- }
5338
- if(sharingEnabled&&_sharingRole==='client'){
5339
- var clientAddr=document.getElementById('cfgClientHubAddress').value.trim();
5340
- var clientTeamToken=document.getElementById('cfgClientTeamToken').value.trim();
5341
- var clientUserToken=document.getElementById('cfgClientUserToken').value.trim();
5342
- cfg.sharing.client={};
5343
- if(clientAddr) cfg.sharing.client.hubAddress=clientAddr;
5344
- if(clientTeamToken) cfg.sharing.client.teamToken=clientTeamToken;
5345
- if(clientUserToken) cfg.sharing.client.userToken=clientUserToken;
5346
- cfg.sharing.hub={port:18800,teamName:'',teamToken:''};
5347
- if(clientAddr){
5348
- try{
5349
- var ips=await fetch('/api/local-ips').then(function(r){return r.json();});
5350
- var localAddrs=['127.0.0.1','localhost','0.0.0.0'].concat(ips.ips||[]);
5351
- var parsed=new URL(clientAddr.indexOf('://')>-1?clientAddr:'http://'+clientAddr);
5352
- if(localAddrs.indexOf(parsed.hostname)>=0){
5353
- done();toast(t('sharing.cannotJoinSelf'),'error');return;
5354
- }
5355
- }catch(e){}
5356
- }
5357
- }
5358
-
5359
- // 1) Embedding model is required
5360
5532
  if(!embP||embP===''){done();toast(t('settings.save.emb.required'),'error');return;}
5361
5533
 
5362
- // 2) Test embedding
5363
5534
  try{
5364
5535
  var er=await fetch('/api/test-model',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({type:'embedding',provider:cfg.embedding.provider,model:cfg.embedding.model||'',endpoint:cfg.embedding.endpoint||'',apiKey:cfg.embedding.apiKey||''})});
5365
5536
  if(er.status===401){done();toast(t('settings.session.expired'),'error');return;}
@@ -5368,7 +5539,6 @@ async function saveConfig(){
5368
5539
  document.getElementById('testEmbResult').className='test-result ok';document.getElementById('testEmbResult').innerHTML='\\u2705 '+t('settings.test.ok');
5369
5540
  }catch(e){done();toast(t('settings.save.emb.fail')+': '+e.message,'error');return;}
5370
5541
 
5371
- // 3) Test summarizer if user filled it
5372
5542
  if(hasSumConfig&&cfg.summarizer){
5373
5543
  try{
5374
5544
  var sr=await fetch('/api/test-model',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({type:'summarizer',provider:cfg.summarizer.provider,model:cfg.summarizer.model||'',endpoint:cfg.summarizer.endpoint||'',apiKey:cfg.summarizer.apiKey||''})});
@@ -5379,7 +5549,6 @@ async function saveConfig(){
5379
5549
  }catch(e){done();toast(t('settings.save.sum.fail')+': '+e.message,'error');return;}
5380
5550
  }
5381
5551
 
5382
- // 4) Test skill model if user filled it
5383
5552
  if(hasSkillConfig&&cfg.skillEvolution.summarizer){
5384
5553
  try{
5385
5554
  var kr=await fetch('/api/test-model',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({type:'summarizer',provider:cfg.skillEvolution.summarizer.provider,model:cfg.skillEvolution.summarizer.model||'',endpoint:cfg.skillEvolution.summarizer.endpoint||'',apiKey:cfg.skillEvolution.summarizer.apiKey||''})});
@@ -5390,7 +5559,6 @@ async function saveConfig(){
5390
5559
  }catch(e){done();toast(t('settings.save.skill.fail')+': '+e.message,'error');return;}
5391
5560
  }
5392
5561
 
5393
- // 5) If summarizer or skill model not configured, check OpenClaw fallback and confirm
5394
5562
  if(!hasSumConfig||!hasSkillConfig){
5395
5563
  try{
5396
5564
  var fr=await fetch('/api/fallback-model');
@@ -5404,22 +5572,86 @@ async function saveConfig(){
5404
5572
  }catch(e){}
5405
5573
  }
5406
5574
 
5407
- // 6) All tests passed, save
5408
- try{
5409
- const r=await fetch('/api/config',{method:'PUT',headers:{'Content-Type':'application/json'},body:JSON.stringify(cfg)});
5410
- if(!r.ok) throw new Error(await r.text());
5411
- const el=document.getElementById('settingsSaved');
5412
- el.classList.add('show');
5413
- setTimeout(()=>el.classList.remove('show'),2500);
5414
- toast(t('settings.saved'),'success');
5575
+ await doSaveConfig(cfg, saveBtn, 'modelsSaved');
5576
+ }
5577
+
5578
+ async function saveHubConfig(){
5579
+ var card=document.getElementById('settingsSharingConfig');
5580
+ var saveBtn=card.querySelector('.settings-actions .btn-primary');
5581
+ saveBtn.disabled=true;saveBtn.textContent=t('settings.test.loading');
5582
+ function done(){saveBtn.disabled=false;saveBtn.textContent=t('settings.save');}
5583
+
5584
+ const cfg={};
5585
+ var sharingEnabled=document.getElementById('cfgSharingEnabled').checked;
5586
+ cfg.sharing={
5587
+ enabled:sharingEnabled,
5588
+ role:_sharingRole,
5589
+ capabilities:{}
5590
+ };
5591
+ if(sharingEnabled&&_sharingRole==='hub'){
5592
+ var hubPort=document.getElementById('cfgHubPort').value.trim();
5593
+ var hubTeamName=document.getElementById('cfgHubTeamName').value.trim();
5594
+ var hubTeamToken=document.getElementById('cfgHubTeamToken').value.trim();
5595
+ cfg.sharing.hub={port:hubPort?Number(hubPort):18800};
5596
+ if(hubTeamName) cfg.sharing.hub.teamName=hubTeamName;
5597
+ if(hubTeamToken) cfg.sharing.hub.teamToken=hubTeamToken;
5598
+ cfg.sharing.client={hubAddress:'',userToken:'',teamToken:''};
5599
+ }
5600
+ if(sharingEnabled&&_sharingRole==='client'){
5601
+ var clientAddr=document.getElementById('cfgClientHubAddress').value.trim();
5602
+ var clientTeamToken=document.getElementById('cfgClientTeamToken').value.trim();
5603
+ var clientUserToken=document.getElementById('cfgClientUserToken').value.trim();
5604
+ cfg.sharing.client={};
5605
+ if(clientAddr) cfg.sharing.client.hubAddress=clientAddr;
5606
+ if(clientTeamToken) cfg.sharing.client.teamToken=clientTeamToken;
5607
+ if(clientUserToken) cfg.sharing.client.userToken=clientUserToken;
5608
+ cfg.sharing.hub={port:18800,teamName:'',teamToken:''};
5609
+ if(clientAddr){
5610
+ try{
5611
+ var ips=await fetch('/api/local-ips').then(function(r){return r.json();});
5612
+ var localAddrs=['127.0.0.1','localhost','0.0.0.0'].concat(ips.ips||[]);
5613
+ var parsed=new URL(clientAddr.indexOf('://')>-1?clientAddr:'http://'+clientAddr);
5614
+ if(localAddrs.indexOf(parsed.hostname)>=0){
5615
+ done();toast(t('sharing.cannotJoinSelf'),'error');return;
5616
+ }
5617
+ }catch(e){}
5618
+ }
5619
+ }
5620
+
5621
+ var prevSharingEnabled=sharingStatusCache&&sharingStatusCache.enabled;
5622
+ var prevRole=sharingStatusCache&&sharingStatusCache.role;
5623
+ if(prevSharingEnabled&&!sharingEnabled){
5624
+ var confirmMsg=prevRole==='hub'?t('sharing.disable.confirm.hub'):t('sharing.disable.confirm.client');
5625
+ if(!confirm(confirmMsg)){done();return;}
5626
+ }
5627
+
5628
+ var ok=await doSaveConfig(cfg, saveBtn, 'hubSaved');
5629
+ if(ok){
5415
5630
  loadSharingStatus(false);
5416
5631
  if(sharingEnabled){
5417
5632
  updateHubShareInfo();
5418
5633
  setTimeout(function(){alert(t('settings.hub.restartAlert'));},300);
5419
5634
  }
5420
- }catch(e){
5421
- toast(t('settings.save.fail')+': '+e.message,'error');
5422
- }finally{done();}
5635
+ if(prevSharingEnabled&&!sharingEnabled){
5636
+ setTimeout(function(){alert(t('sharing.disable.restartAlert'));},300);
5637
+ }
5638
+ }
5639
+ }
5640
+
5641
+ async function saveGeneralConfig(){
5642
+ var card=document.querySelector('.card-general');
5643
+ var saveBtn=card.querySelector('.settings-actions .btn-primary');
5644
+
5645
+ const cfg={};
5646
+ const vp=document.getElementById('cfgViewerPort').value.trim();
5647
+ if(vp) cfg.viewerPort=Number(vp);
5648
+ cfg.telemetry={enabled:document.getElementById('cfgTelemetryEnabled').checked};
5649
+
5650
+ await doSaveConfig(cfg, saveBtn, 'generalSaved');
5651
+ }
5652
+
5653
+ async function saveConfig(){
5654
+ await saveModelsConfig();
5423
5655
  }
5424
5656
 
5425
5657
  async function testModel(type){
@@ -5781,12 +6013,25 @@ function renderToolAgg(data){
5781
6013
  '</tbody></table>';
5782
6014
  }
5783
6015
 
6016
+ /* ─── Sharing status polling ─── */
6017
+ var _sharingPollTimer=null;
6018
+ function startSharingPoll(){
6019
+ stopSharingPoll();
6020
+ _sharingPollTimer=setInterval(function(){
6021
+ if(sharingStatusCache&&sharingStatusCache.enabled) loadSharingStatus(false);
6022
+ },30000);
6023
+ }
6024
+ function stopSharingPoll(){
6025
+ if(_sharingPollTimer){clearInterval(_sharingPollTimer);_sharingPollTimer=null;}
6026
+ }
6027
+
5784
6028
  /* ─── Data loading ─── */
5785
6029
  async function loadAll(){
5786
6030
  await Promise.all([loadStats(),loadMemories(),loadSharingStatus(false)]);
5787
6031
  checkMigrateStatus();
5788
6032
  connectPPSSE();
5789
6033
  checkForUpdate();
6034
+ startSharingPoll();
5790
6035
  }
5791
6036
 
5792
6037
  async function loadStats(ownerFilter){