kingkont 0.18.10 → 0.18.12

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "kingkont",
3
- "version": "0.18.10",
3
+ "version": "0.18.12",
4
4
  "description": "KingKont · Chatium — нод-редактор сцен с AI-генерацией (картинки/видео/голос/SFX/музыка/текст)",
5
5
  "main": "main.js",
6
6
  "bin": {
@@ -2185,10 +2185,21 @@ async function mutateNode(bKey, boardHandle, nodeId, mutator) {
2185
2185
  }
2186
2186
 
2187
2187
  function updateJobsBadge() {
2188
+ // КРИТИЧНО: #jobsInfo был удалён из sidebar-footer в v0.18.6 (юзер
2189
+ // попросил убрать «В фоне: N» из sidebar — счётчик переехал на 🔔).
2190
+ // Без этого guard'а функция бросала TypeError на null.style → ломала
2191
+ // ВСЕ функции которые её зовут после state.jobs.set/delete (т.е.
2192
+ // startGenerationJob, runTextJob, runMusicJob, runSfxJob, runTTSJob).
2193
+ // Симптом: generation-задачи не уходили на сервер, /api/generate
2194
+ // никогда не вызывался. Resume-job'ы умирали в .catch().
2188
2195
  const el = $('jobsInfo');
2189
- const n = state.jobs.size;
2190
- if (n === 0) { el.style.display = 'none'; return; }
2191
- el.style.display = '';
2192
- el.innerHTML = `<span class="spinner"></span>В фоне: ${n}`;
2196
+ if (el) {
2197
+ const n = state.jobs.size;
2198
+ if (n === 0) { el.style.display = 'none'; }
2199
+ else { el.style.display = ''; el.innerHTML = `<span class="spinner"></span>В фоне: ${n}`; }
2200
+ }
2201
+ // Bell-state показывает то же самое (число + описание справа).
2202
+ // Триггерим перерисовку — чтобы не ждать 1.5s poll.
2203
+ if (window.notifyPanel?.refreshBellState) window.notifyPanel.refreshBellState();
2193
2204
  }
2194
2205
 
@@ -200,14 +200,22 @@
200
200
  // Public API.
201
201
  // Дедуп: gen-completion идёт двумя путями (local pollJob → showToast,
202
202
  // и server _startPoller → WS jobs:all). Без дедупа юзер видит ДВА
203
- // одинаковых события с разницей в секунды. Ключ дедупа — kind+nodeId
204
- // (если есть target.nodeId) или kind+text (fallback). Окно — 30 сек.
203
+ // одинаковых события с разницей в секунды.
204
+ //
205
+ // Ключ дедупа — ТОЛЬКО nodeId (без kind). Раньше был «kind:nodeId»,
206
+ // но local pollJob использует kind='info' когда генерация завершилась
207
+ // на НЕ-текущей доске, а WS-path всегда 'ok' → разные ключи → дубль
208
+ // не ловился (юзер видел два одинаковых «✓ image «name» готов»,
209
+ // один синий, второй зелёный). Сейчас одинаковый nodeId-completion
210
+ // в окне 30s — один event.
205
211
  const DEDUP_WINDOW_MS = 30 * 1000;
212
+ // События с этими kind считаются «completion» — для них дедупим по nodeId.
213
+ const _COMPLETION_KINDS = new Set(['ok', 'error', 'info', 'warn']);
206
214
  function _dedupKey(kind, text, target) {
207
- if (target?.nodeId && (kind === 'ok' || kind === 'error')) {
208
- return `${kind}:${target.nodeId}`;
215
+ if (target?.nodeId && _COMPLETION_KINDS.has(kind)) {
216
+ return `node:${target.nodeId}`; // одинаковый nodeId = тот же gen
209
217
  }
210
- return null; // не дедупим chat / info-сообщения, у них нет nodeId
218
+ return null; // chat-события (target.chat) не дедупим нет nodeId
211
219
  }
212
220
  function addEvent({ kind, text, raw, target }) {
213
221
  if (!text) return;
@@ -427,6 +435,9 @@
427
435
  close: () => setOpen(false),
428
436
  toggle, addEvent, render,
429
437
  isManuallyOpen,
438
+ // Для generate.js:updateJobsBadge — синхронный rerender bell-state
439
+ // когда state.jobs изменился (без ожидания 1.5s poll).
440
+ refreshBellState: _renderBellState,
430
441
  };
431
442
 
432
443
  // Init: ensure UI exists ASAP, drain pending toast queue, subscribe WS.