|
|
| |
| |
| |
|
|
|
|
| import uiModule from './ui.js';
|
| import sessionModule from './sessions.js';
|
| import spinnerModule from './spinner.js';
|
| import markdownModule from './markdown.js';
|
| import { makeWindowDraggable } from './windowDrag.js';
|
| import { langIcon } from './langIcons.js';
|
| import { registerMenuDismiss, dismissOrRemove } from './escMenuStack.js';
|
|
|
|
|
| let API_BASE = '';
|
| let _esc;
|
| let _getDocs;
|
| let _isOpenFn;
|
| let _createDocument;
|
| let _loadDocument;
|
| let _switchToDoc;
|
| let _openPanel;
|
| let _addDocToTabs;
|
| let _syncDocIndicator;
|
|
|
| export function initLibrary(config) {
|
| API_BASE = config.apiBase;
|
| _esc = config.esc;
|
| _getDocs = config.getDocs;
|
| _isOpenFn = config.isOpen;
|
| _createDocument = config.createDocument;
|
| _loadDocument = config.loadDocument;
|
| _switchToDoc = config.switchToDoc;
|
| _openPanel = config.openPanel;
|
| _addDocToTabs = config.addDocToTabs;
|
| _syncDocIndicator = config.syncDocIndicator;
|
| }
|
|
|
|
|
| let _libraryOpen = false;
|
|
|
|
|
|
|
| const _libraryCascadedTabs = new Set();
|
| function _maybeCascadeGrid(grid, tabKey) {
|
| if (!grid || !tabKey || _libraryCascadedTabs.has(tabKey)) return;
|
| _libraryCascadedTabs.add(tabKey);
|
| grid.classList.add('doclib-just-opened');
|
| setTimeout(() => grid.classList.remove('doclib-just-opened'), 900);
|
| }
|
| let _libraryDocs = [];
|
| let _libraryTotal = 0;
|
| let _libraryOffset = 0;
|
| let _docsVisibleLimit = 20;
|
| let _libraryLanguages = {};
|
| let _librarySessionCount = 0;
|
| let _libraryActiveLanguage = null;
|
| let _librarySort = 'recent';
|
| let _librarySearch = '';
|
| let _librarySearchDebounce = null;
|
|
|
|
|
|
|
|
|
| function _hlSearch(text) {
|
| const esc = _esc(text || '');
|
| const q = (_librarySearch || '').trim();
|
| if (!q) return esc;
|
| const toks = [...new Set(q.split(/\s+/).filter(Boolean))]
|
| .sort((a, b) => b.length - a.length)
|
| .map(t => t.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'));
|
| if (!toks.length) return esc;
|
| try {
|
| return esc.replace(new RegExp(`(${toks.join('|')})`, 'gi'),
|
| '<mark class="doclib-search-hl">$1</mark>');
|
| } catch { return esc; }
|
| }
|
|
|
| function _safeResearchHref(raw) {
|
| try {
|
| const parsed = new URL(String(raw || '').trim(), window.location.origin);
|
| if (parsed.protocol === 'http:' || parsed.protocol === 'https:') return _esc(parsed.href);
|
| } catch {}
|
| return '';
|
| }
|
|
|
| let _libraryEscHandler = null;
|
| let _librarySelectMode = false;
|
| let _librarySelectedIds = new Set();
|
| let _libraryImportMode = false;
|
| let _libScrollBound = false;
|
| let _libraryArchivedView = false;
|
|
|
|
|
|
|
|
|
| function _collapseExpandedCard(card) {
|
| const grid = card.closest('.doclib-grid');
|
| const instant = card?.dataset?.spaceToggle === '1';
|
| card.classList.remove('doclib-card-expanded');
|
|
|
| if (grid) {
|
| grid.style.minHeight = '';
|
| grid.style.maxHeight = '';
|
| }
|
| const reader = card.querySelector('.doclib-card-reader');
|
| if (reader) reader.remove();
|
|
|
|
|
| if (grid && !instant) {
|
| const siblings = [...grid.querySelectorAll('.doclib-card')].filter(c => c !== card);
|
| siblings.forEach(s => { s.style.opacity = '0'; });
|
| requestAnimationFrame(() => {
|
| siblings.forEach(s => {
|
| s.style.transition = 'opacity 0.15s ease';
|
| s.style.opacity = '1';
|
| });
|
| setTimeout(() => { siblings.forEach(s => { s.style.transition = ''; s.style.opacity = ''; }); }, 200);
|
| });
|
| }
|
| }
|
|
|
|
|
|
|
|
|
|
|
| async function _copyChatById(sessionId) {
|
| try {
|
| const res = await fetch(`${API_BASE}/api/history/${sessionId}`, { credentials: 'same-origin' });
|
| if (!res.ok) throw new Error(res.statusText);
|
| const data = await res.json();
|
| const history = Array.isArray(data) ? data : (data.history || []);
|
| const lines = [];
|
| for (const m of history) {
|
| if (m.role !== 'user' && m.role !== 'assistant') continue;
|
| const label = m.role === 'user' ? 'User' : 'Assistant';
|
| const body = (m.content || '')
|
| .replace(/<think>[\s\S]*?<\/think>/g, '')
|
| .replace(/<think>[\s\S]*$/, '')
|
| .trim();
|
| if (body) lines.push(`${label}: ${body}`);
|
| }
|
| const text = lines.join('\n\n');
|
| if (uiModule && uiModule.copyToClipboard) {
|
| await uiModule.copyToClipboard(text);
|
| } else {
|
| await navigator.clipboard.writeText(text);
|
| }
|
| } catch (err) {
|
| if (uiModule && uiModule.showError) uiModule.showError('Failed to copy chat');
|
| }
|
| }
|
|
|
|
|
|
|
|
|
|
|
| function _attachLongPressMenu(card, menuSelector) {
|
| let hold = null;
|
| let start = null;
|
| const cancel = () => { if (hold) { clearTimeout(hold); hold = null; } start = null; };
|
| card.addEventListener('pointerdown', (e) => {
|
| if (e.target.closest(menuSelector + ', .memory-select-cb, button')) return;
|
| start = { x: e.clientX, y: e.clientY };
|
| hold = setTimeout(() => {
|
| hold = null;
|
| card._suppressNextClick = true;
|
| setTimeout(() => { card._suppressNextClick = false; }, 400);
|
| if (navigator.vibrate) try { navigator.vibrate(15); } catch {}
|
| const btn = card.querySelector(menuSelector);
|
| if (btn) btn.click();
|
| }, 500);
|
| });
|
| card.addEventListener('pointermove', (e) => {
|
| if (!start) return;
|
| if (Math.hypot(e.clientX - start.x, e.clientY - start.y) > 10) cancel();
|
| });
|
| card.addEventListener('pointerup', cancel);
|
| card.addEventListener('pointercancel', cancel);
|
| }
|
|
|
|
|
|
|
|
|
| const _LIB_DD_ICONS = {
|
| open: '<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6"/><polyline points="15 3 21 3 21 9"/><line x1="10" y1="14" x2="21" y2="3"/></svg>',
|
| archive: '<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="21 8 21 21 3 21 3 8"/><rect x="1" y="3" width="22" height="5"/><line x1="10" y1="12" x2="14" y2="12"/></svg>',
|
| restore: '<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M3 12a9 9 0 1 0 9-9"/><polyline points="3 4 3 9 8 9"/></svg>',
|
| delete: '<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="3 6 5 6 21 6"/><path d="M19 6l-1 14a2 2 0 0 1-2 2H8a2 2 0 0 1-2-2L5 6"/><path d="M10 11v6"/><path d="M14 11v6"/></svg>',
|
| clone: '<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="9" y="9" width="13" height="13" rx="2"/><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"/></svg>',
|
| copy: '<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="9" y="9" width="13" height="13" rx="2"/><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"/></svg>',
|
| };
|
|
|
| function _showLibDropdown(anchor, items, opts) {
|
| opts = opts || {};
|
| document.querySelectorAll('._lib-dd').forEach(dismissOrRemove);
|
| const dd = document.createElement('div');
|
| dd.className = 'dropdown session-dropdown-menu _lib-dd';
|
| for (const item of items) {
|
| const row = document.createElement('div');
|
| row.className = 'dropdown-item-compact' + (item.danger ? ' dropdown-item-danger' : '');
|
| const iconKey = item.icon || item.label.toLowerCase();
|
| const iconSvg = _LIB_DD_ICONS[iconKey] || '';
|
| row.innerHTML = (iconSvg ? '<span class="dropdown-icon">' + iconSvg + '</span>' : '') + '<span>' + item.label + '</span>';
|
| row.addEventListener('click', (e) => { e.stopPropagation(); teardown(); item.action(); });
|
| dd.appendChild(row);
|
| }
|
| if (typeof opts.onSelect === 'function') {
|
| const sel = document.createElement('div');
|
| sel.className = 'dropdown-item-compact';
|
| sel.innerHTML =
|
| '<span class="dropdown-icon"><span style="font-size:16px;line-height:1;position:relative;top:-2px;">●</span></span>'
|
| + '<span>Select</span>';
|
| sel.addEventListener('click', (e) => { e.stopPropagation(); teardown(); opts.onSelect(); });
|
| dd.appendChild(sel);
|
| }
|
| const cancel = document.createElement('div');
|
| cancel.className = 'dropdown-item-compact dropdown-cancel-mobile';
|
| cancel.innerHTML =
|
| '<span class="dropdown-icon"><svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round"><line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/></svg></span>'
|
| + '<span>Cancel</span>';
|
| cancel.addEventListener('click', (e) => { e.stopPropagation(); teardown(); if (typeof opts.onCancel === 'function') opts.onCancel(); });
|
| dd.appendChild(cancel);
|
| document.body.appendChild(dd);
|
| const rect = anchor.getBoundingClientRect();
|
| dd.style.right = (window.innerWidth - rect.right) + 'px';
|
| dd.style.top = (rect.bottom + 2) + 'px';
|
| dd.style.display = 'block';
|
| dd.style.zIndex = '100000';
|
| requestAnimationFrame(() => {
|
| const mr = dd.getBoundingClientRect();
|
| if (mr.bottom > window.innerHeight - 8) {
|
| dd.style.top = (rect.top - mr.height - 2) + 'px';
|
| }
|
| if (mr.left < 8) { dd.style.left = '8px'; dd.style.right = 'auto'; }
|
| });
|
|
|
|
|
| let _unreg = () => {};
|
| const teardown = () => {
|
| _unreg(); _unreg = () => {};
|
| document.removeEventListener('click', close);
|
| dd.remove();
|
| };
|
| const close = (e) => { if (!dd.contains(e.target)) teardown(); };
|
| setTimeout(() => document.addEventListener('click', close), 0);
|
| _unreg = registerMenuDismiss(teardown);
|
| dd._dismiss = teardown;
|
|
|
|
|
|
|
|
|
| let _swipeStart = null;
|
| let _swipeDy = 0;
|
| dd.addEventListener('touchstart', (e) => {
|
| if (e.touches.length !== 1) return;
|
| _swipeStart = { x: e.touches[0].clientX, y: e.touches[0].clientY };
|
| _swipeDy = 0;
|
| dd.style.transition = '';
|
| }, { passive: true });
|
| dd.addEventListener('touchmove', (e) => {
|
| if (!_swipeStart || e.touches.length !== 1) return;
|
| const dx = e.touches[0].clientX - _swipeStart.x;
|
| const dy = e.touches[0].clientY - _swipeStart.y;
|
| if (Math.abs(dy) < Math.abs(dx)) { _swipeStart = null; return; }
|
| if (dy > 0) {
|
| _swipeDy = dy;
|
| dd.style.transform = 'translateY(' + dy + 'px)';
|
| dd.style.opacity = String(Math.max(0.3, 1 - dy / 240));
|
| }
|
| }, { passive: true });
|
| dd.addEventListener('touchend', () => {
|
| if (!_swipeStart) return;
|
| _swipeStart = null;
|
| if (_swipeDy > 60) {
|
| dd.style.transition = 'transform 0.15s ease, opacity 0.15s ease';
|
| dd.style.transform = 'translateY(120px)';
|
| dd.style.opacity = '0';
|
|
|
|
|
| _unreg(); _unreg = () => {};
|
| document.removeEventListener('click', close);
|
| setTimeout(() => dd.remove(), 160);
|
| } else {
|
| dd.style.transition = 'transform 0.18s ease, opacity 0.18s ease';
|
| dd.style.transform = '';
|
| dd.style.opacity = '';
|
| }
|
| });
|
| }
|
|
|
|
|
|
|
| function libraryRelativeTime(isoString) {
|
| if (!isoString) return '';
|
| const now = Date.now();
|
| const then = new Date(isoString).getTime();
|
| const diffS = Math.floor((now - then) / 1000);
|
| if (diffS < 60) return 'just now';
|
| const diffM = Math.floor(diffS / 60);
|
| if (diffM < 60) return diffM + 'm ago';
|
| const diffH = Math.floor(diffM / 60);
|
| if (diffH < 24) return diffH + 'h ago';
|
| const diffD = Math.floor(diffH / 24);
|
| if (diffD === 1) return 'yesterday';
|
| if (diffD < 14) return diffD + 'd ago';
|
| const diffW = Math.floor(diffD / 7);
|
| if (diffW < 8) return diffW + 'w ago';
|
| return new Date(isoString).toLocaleDateString();
|
| }
|
|
|
| async function libraryFetch(append) {
|
| if (!append) _libraryOffset = 0;
|
|
|
|
|
|
|
|
|
| const params = new URLSearchParams({
|
| sort: _librarySort,
|
| offset: String(_libraryOffset),
|
| limit: '50',
|
| });
|
| if (_librarySearch) params.set('search', _librarySearch);
|
| if (_libraryActiveLanguage) params.set('language', _libraryActiveLanguage);
|
| if (_libraryArchivedView) params.set('archived', 'true');
|
|
|
| try {
|
| const res = await fetch(`${API_BASE}/api/documents/library?${params}`);
|
| if (!res.ok) throw new Error(res.statusText);
|
| const data = await res.json();
|
|
|
| if (append) {
|
| _libraryDocs = _libraryDocs.concat(data.documents);
|
| } else {
|
| _libraryDocs = data.documents;
|
| _docsVisibleLimit = 20;
|
| }
|
| _libraryTotal = data.total;
|
| _libraryLanguages = data.languages;
|
| _librarySessionCount = data.session_count;
|
|
|
| libraryRenderStats();
|
| libraryRenderLangChips();
|
| libraryRenderGrid();
|
| libraryRenderLoadMore();
|
| } catch (e) {
|
| console.error('Library fetch error:', e);
|
| }
|
| }
|
|
|
| function libraryRenderStats() {
|
| const el = document.getElementById('doclib-stats');
|
| if (!el) return;
|
| const totalAll = Object.values(_libraryLanguages).reduce((a, b) => a + b, 0);
|
| if (_librarySearch || _libraryActiveLanguage) {
|
| el.textContent = `${_libraryTotal} of ${totalAll} document${totalAll !== 1 ? 's' : ''}`;
|
| } else {
|
| el.textContent = `${totalAll} document${totalAll !== 1 ? 's' : ''}`;
|
| }
|
| }
|
|
|
| function libraryRenderLangChips() {
|
| const wrap = document.getElementById('doclib-chips');
|
| if (!wrap) return;
|
|
|
| wrap.querySelectorAll('.memory-cat-chip').forEach(c => c.remove());
|
| const totalAll = Object.values(_libraryLanguages).reduce((a, b) => a + b, 0);
|
|
|
|
|
| if (totalAll === 0) return;
|
|
|
| const allChip = document.createElement('button');
|
| allChip.className = 'memory-cat-chip' + (!_libraryActiveLanguage ? ' active' : '');
|
| allChip.textContent = `all (${totalAll})`;
|
| allChip.addEventListener('click', () => {
|
| if (_librarySelectMode) {
|
| _libraryDocs.forEach(d => _librarySelectedIds.add(d.id));
|
| libraryUpdateBulkCount();
|
| const selectAllEl = document.getElementById('doclib-select-all');
|
| if (selectAllEl) selectAllEl.checked = true;
|
| libraryRenderGrid();
|
| return;
|
| }
|
| _libraryActiveLanguage = null;
|
| libraryFetch(false);
|
| });
|
| wrap.appendChild(allChip);
|
|
|
| const sorted = Object.entries(_libraryLanguages).sort((a, b) => b[1] - a[1]);
|
| for (const [lang, count] of sorted) {
|
| const chip = document.createElement('button');
|
| chip.className = 'memory-cat-chip' + (_libraryActiveLanguage === lang ? ' active' : '');
|
| chip.textContent = `${lang} (${count})`;
|
| chip.addEventListener('click', () => {
|
| _libraryActiveLanguage = lang;
|
| libraryFetch(false);
|
| });
|
| wrap.appendChild(chip);
|
| }
|
| }
|
|
|
| function libraryRemoveDocumentFromState(docId) {
|
| const removed = _libraryDocs.find(d => String(d.id) === String(docId));
|
| _libraryDocs = _libraryDocs.filter(d => String(d.id) !== String(docId));
|
| _librarySelectedIds.delete(docId);
|
| _libraryTotal = Math.max(0, _libraryTotal - 1);
|
|
|
| const lang = removed && (removed.language || 'text');
|
| if (lang && Object.prototype.hasOwnProperty.call(_libraryLanguages, lang)) {
|
| const next = Math.max(0, Number(_libraryLanguages[lang] || 0) - 1);
|
| if (next > 0) {
|
| _libraryLanguages[lang] = next;
|
| } else {
|
| delete _libraryLanguages[lang];
|
| }
|
| }
|
|
|
| libraryRenderStats();
|
| libraryRenderLangChips();
|
| libraryUpdateBulkCount();
|
| }
|
|
|
| function libraryRenderGrid() {
|
| const grid = document.getElementById('doclib-grid');
|
| if (!grid) return;
|
|
|
|
|
|
|
| document.querySelectorAll('.doclib-card-dropdown').forEach(dismissOrRemove);
|
| grid.innerHTML = '';
|
|
|
| if (grid.parentElement) grid.parentElement.querySelectorAll(':scope > .doclib-inline-load-more').forEach(b => b.remove());
|
|
|
| if (_libraryDocs.length === 0) {
|
| if (_librarySearch || _libraryActiveLanguage) {
|
| grid.innerHTML = '<div class="doclib-empty">No documents match your search.</div>';
|
| } else {
|
| const _impIco = '<svg width="13" height="13" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-2px;margin:0 4px;"><path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"/><polyline points="17 8 12 3 7 8"/><line x1="12" y1="3" x2="12" y2="15"/></svg>';
|
| grid.innerHTML =
|
| '<div class="doclib-empty" style="display:flex;align-items:center;justify-content:center;gap:8px;flex-wrap:wrap;">' +
|
| '<span>No documents yet</span>' +
|
| '<span style="opacity:0.7;font-size:11px;">' +
|
| '<a href="#" data-doclib-import style="color:var(--accent,var(--red));text-decoration:underline;">Import' + _impIco + '</a>' +
|
| ' · or create one in a session' +
|
| '</span>' +
|
| '</div>';
|
| grid.querySelector('[data-doclib-import]')?.addEventListener('click', (e) => {
|
| e.preventDefault();
|
| document.getElementById('doclib-import-file-btn')?.click();
|
| });
|
| }
|
| return;
|
| }
|
| _maybeCascadeGrid(grid, 'documents');
|
|
|
|
|
|
|
|
|
| const shown = _libraryDocs.slice(0, _docsVisibleLimit);
|
| for (const doc of shown) {
|
| grid.appendChild(libraryCreateCard(doc));
|
| }
|
|
|
|
|
| const shownCount = shown.length;
|
| if (shownCount < _libraryTotal) {
|
| const btn = document.createElement('button');
|
| btn.className = 'doclib-load-more doclib-inline-load-more';
|
| btn.id = 'doclib-docs-load-more';
|
| btn.textContent = `Load more (${shownCount} of ${_libraryTotal})`;
|
| btn.addEventListener('click', async () => {
|
| _docsVisibleLimit += 20;
|
|
|
| if (_docsVisibleLimit > _libraryDocs.length && _libraryDocs.length < _libraryTotal) {
|
| _libraryOffset = _libraryDocs.length;
|
| await libraryFetch(true);
|
| } else {
|
| libraryRenderGrid();
|
| }
|
| });
|
| grid.parentElement.appendChild(btn);
|
| }
|
| }
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| if (!_libScrollBound) {
|
| _libScrollBound = true;
|
| let _tick = false;
|
| const _maybeAutoLoad = () => {
|
| _tick = false;
|
| if (!_libraryOpen) return;
|
| for (const btn of document.querySelectorAll('.doclib-inline-load-more')) {
|
| if (btn.dataset.autoLoaded) continue;
|
| if (!btn.offsetParent) continue;
|
| if (btn.getBoundingClientRect().top > window.innerHeight + 600) continue;
|
| btn.dataset.autoLoaded = '1';
|
| btn.click();
|
| break;
|
| }
|
| };
|
| document.addEventListener('scroll', () => {
|
| if (_tick) return;
|
| _tick = true;
|
| requestAnimationFrame(_maybeAutoLoad);
|
| }, true);
|
| }
|
|
|
| function libraryCreateCard(doc) {
|
| const card = document.createElement('div');
|
| card.className = 'doclib-card memory-item';
|
| card.dataset.docId = doc.id;
|
| if (_librarySelectMode && _librarySelectedIds.has(doc.id)) {
|
| card.classList.add('selected');
|
| }
|
|
|
|
|
| if (_librarySelectMode) {
|
| const cb = document.createElement('input');
|
| cb.type = 'checkbox';
|
| cb.className = 'memory-select-cb';
|
| cb.checked = _librarySelectedIds.has(doc.id);
|
| cb.addEventListener('click', (e) => e.stopPropagation());
|
| cb.addEventListener('change', () => {
|
| libraryToggleSelectItem(doc.id);
|
| card.classList.toggle('selected', _librarySelectedIds.has(doc.id));
|
| const selectAllEl = document.getElementById('doclib-select-all');
|
| if (selectAllEl) selectAllEl.checked = _libraryDocs.every(d => _librarySelectedIds.has(d.id));
|
| });
|
| card.appendChild(cb);
|
| }
|
|
|
|
|
| const content = document.createElement('div');
|
| content.style.cssText = 'flex:1;min-width:0;padding-top:4px;';
|
|
|
|
|
| const titleRow = document.createElement('div');
|
| titleRow.style.cssText = 'display:flex;align-items:center;gap:6px;width:100%;';
|
| const titleEl = document.createElement('span');
|
| titleEl.className = 'memory-item-title';
|
| titleEl.style.cssText = 'flex:0 1 auto;min-width:0;';
|
|
|
|
|
|
|
| const _GEN_DOC_ICON = '<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-2px;margin-right:4px;opacity:0.4;flex-shrink:0;"><path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/><polyline points="14 2 14 8 20 8"/><line x1="16" y1="13" x2="8" y2="13"/><line x1="16" y1="17" x2="8" y2="17"/></svg>';
|
| const _langSvg = doc.language && doc.language !== 'text'
|
| ? langIcon(doc.language, 12, { style: 'vertical-align:-2px;margin-right:4px;opacity:0.55;flex-shrink:0;color:currentColor;' })
|
| : '';
|
| titleEl.innerHTML = (_langSvg || _GEN_DOC_ICON) + _hlSearch(doc.title || 'Untitled');
|
| titleRow.appendChild(titleEl);
|
| const verBadge = document.createElement('span');
|
| verBadge.style.cssText = 'font-size:9px;padding:1px 6px;border-radius:8px;background:color-mix(in srgb, var(--red) 15%, transparent);border:1px solid color-mix(in srgb, var(--red) 40%, transparent);color:var(--red);flex-shrink:0;';
|
| verBadge.textContent = 'v' + (doc.version_count || 1);
|
| titleRow.appendChild(verBadge);
|
|
|
|
|
|
|
| const chevron = document.createElement('span');
|
| chevron.className = 'doclib-card-chevron';
|
| chevron.style.marginLeft = 'auto';
|
| chevron.innerHTML = '<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="6 9 12 15 18 9"/></svg>';
|
| titleRow.appendChild(chevron);
|
| content.appendChild(titleRow);
|
|
|
|
|
| const meta = document.createElement('div');
|
| meta.className = 'memory-item-meta';
|
| meta.style.cssText = 'font-size:10px;opacity:0.55;margin-top:2px;display:flex;align-items:center;gap:6px;flex-wrap:wrap;';
|
| const _esc = (s) => uiModule.esc(String(s || ''));
|
| const pieces = [];
|
| if (doc.session_name) pieces.push(`<span>${_esc(doc.session_name)}</span>`);
|
| if (doc.language && doc.language !== 'text') {
|
|
|
|
|
| pieces.push(`<span>${_esc(doc.language)}</span>`);
|
| }
|
| pieces.push(`<span>${_esc(libraryRelativeTime(doc.updated_at))}</span>`);
|
| meta.innerHTML = pieces.join('<span style="opacity:0.5;">\u00b7</span>');
|
| content.appendChild(meta);
|
| card.appendChild(content);
|
|
|
|
|
| const header = document.createElement('div');
|
| header.className = 'doclib-card-header';
|
| header.style.display = 'none';
|
|
|
|
|
| const actionsWrap = document.createElement('div');
|
| actionsWrap.className = 'memory-item-actions';
|
| const menuWrap = document.createElement('span');
|
| menuWrap.className = 'doclib-card-menu-wrap';
|
| menuWrap.style.position = 'relative';
|
| const menuBtn = document.createElement('button');
|
| menuBtn.className = 'memory-item-btn';
|
| menuBtn.title = 'Actions';
|
| menuBtn.innerHTML = '<svg width="14" height="14" viewBox="0 0 24 24" fill="currentColor"><circle cx="12" cy="5" r="2"/><circle cx="12" cy="12" r="2"/><circle cx="12" cy="19" r="2"/></svg>';
|
| menuBtn.addEventListener('click', (e) => {
|
| e.stopPropagation();
|
|
|
|
|
|
|
| if (window.innerWidth <= 768) {
|
| const items = [];
|
| if (doc.session_id) items.push({ label: 'Open', action: () => libraryOpenInSession(doc) });
|
| items.push({ label: 'Clone', action: () => libraryImportDocument(doc) });
|
| _showLibDropdown(menuBtn, items, { onSelect: () => {
|
| libraryEnterSelectMode();
|
| _librarySelectedIds.add(doc.id);
|
| libraryUpdateBulkCount();
|
| libraryRenderGrid();
|
| } });
|
| return;
|
| }
|
| const dropdown = menuWrap.querySelector('.doclib-card-dropdown') || document.body.querySelector('.doclib-card-dropdown[data-owner="' + CSS.escape(doc.id) + '"]');
|
| if (dropdown) {
|
| const isOpen = dropdown.style.display !== 'none' && dropdown.parentElement === document.body;
|
| if (isOpen) {
|
| hideCardDropdown();
|
| } else {
|
|
|
| const rect = menuBtn.getBoundingClientRect();
|
| document.body.appendChild(dropdown);
|
| dropdown.dataset.owner = doc.id;
|
| dropdown.style.cssText = 'position:fixed;z-index:10000;min-width:0;width:max-content;padding:4px;background:var(--panel);border:1px solid var(--border);border-radius:8px;box-shadow:0 8px 24px rgba(0,0,0,0.3);backdrop-filter:blur(12px);font-size:12px;display:block;';
|
| dropdown.style.top = (rect.bottom + 4) + 'px';
|
| dropdown.style.left = 'auto';
|
| dropdown.style.right = (window.innerWidth - rect.right) + 'px';
|
|
|
| requestAnimationFrame(() => {
|
| const mr = dropdown.getBoundingClientRect();
|
| if (mr.bottom > window.innerHeight - 8) dropdown.style.top = (rect.top - mr.height - 4) + 'px';
|
| if (mr.left < 8) { dropdown.style.left = '8px'; dropdown.style.right = 'auto'; }
|
| });
|
|
|
| _cardDocClick = (ev) => {
|
| if (!dropdown.contains(ev.target) && !menuWrap.contains(ev.target)) hideCardDropdown();
|
| };
|
| setTimeout(() => document.addEventListener('click', _cardDocClick, true), 0);
|
| _cardUnreg = registerMenuDismiss(hideCardDropdown);
|
| }
|
| }
|
| });
|
| menuWrap.appendChild(menuBtn);
|
|
|
|
|
| const dropdown = document.createElement('div');
|
| dropdown.className = 'doclib-card-dropdown';
|
| dropdown.style.cssText = 'display:none;position:absolute;top:100%;right:0;z-index:1000;min-width:0;width:max-content;padding:4px;background:var(--panel);border:1px solid var(--border);border-radius:8px;box-shadow:0 8px 24px rgba(0,0,0,0.3);backdrop-filter:blur(12px);font-size:12px;';
|
|
|
|
|
|
|
|
|
|
|
|
|
| let _cardUnreg = () => {};
|
| let _cardDocClick = null;
|
| function hideCardDropdown() {
|
| _cardUnreg(); _cardUnreg = () => {};
|
| if (_cardDocClick) { document.removeEventListener('click', _cardDocClick, true); _cardDocClick = null; }
|
| dropdown.style.display = 'none';
|
| if (dropdown.parentElement === document.body) menuWrap.appendChild(dropdown);
|
| }
|
| dropdown._dismiss = hideCardDropdown;
|
|
|
| const _di = (svg) => `<span class="dropdown-icon">${svg}</span>`;
|
| const _openIco = '<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M18 13v6a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2V8a2 2 0 0 1 2-2h6"/><polyline points="15 3 21 3 21 9"/><line x1="10" y1="14" x2="21" y2="3"/></svg>';
|
|
|
|
|
| const openItem = document.createElement('button');
|
| openItem.className = 'dropdown-item-compact';
|
| openItem.style.cssText = 'background:none;border:none;width:100%;';
|
| openItem.innerHTML = _di(_openIco) + '<span>Open</span>';
|
| if (doc.session_id) {
|
| openItem.addEventListener('click', (e) => { e.stopPropagation(); hideCardDropdown(); libraryOpenInSession(doc); });
|
| } else {
|
|
|
|
|
| openItem.title = 'Open in the editor';
|
| openItem.addEventListener('click', (e) => { e.stopPropagation(); hideCardDropdown(); libraryOpenDocument(doc); });
|
| }
|
| dropdown.appendChild(openItem);
|
|
|
|
|
| const _cloneIco = '<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="9" y="9" width="13" height="13" rx="2"/><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"/></svg>';
|
| const cloneItem = document.createElement('button');
|
| cloneItem.className = 'dropdown-item-compact';
|
| cloneItem.style.cssText = 'background:none;border:none;width:100%;';
|
| cloneItem.innerHTML = _di(_cloneIco) + '<span>Clone</span>';
|
| cloneItem.title = 'Clone to active session';
|
| cloneItem.addEventListener('click', (e) => { e.stopPropagation(); hideCardDropdown(); libraryImportDocument(doc); });
|
| dropdown.appendChild(cloneItem);
|
|
|
|
|
| const _exportIco = '<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"/><polyline points="7 10 12 15 17 10"/><line x1="12" y1="15" x2="12" y2="3"/></svg>';
|
| const exportItem = document.createElement('button');
|
| exportItem.className = 'dropdown-item-compact';
|
| exportItem.style.cssText = 'background:none;border:none;width:100%;';
|
| exportItem.innerHTML = _di(_exportIco) + '<span>Export</span>';
|
| exportItem.addEventListener('click', async (e) => {
|
| e.stopPropagation();
|
| hideCardDropdown();
|
| try {
|
| const res = await fetch(`${API_BASE}/api/document/${doc.id}`);
|
| if (!res.ok) throw new Error('Failed');
|
| const full = await res.json();
|
| const extMap = { javascript: '.js', python: '.py', html: '.html', css: '.css', markdown: '.md', json: '.json', yaml: '.yml', bash: '.sh', sql: '.sql', rust: '.rs', go: '.go', java: '.java', c: '.c', cpp: '.cpp', typescript: '.ts', ruby: '.rb', php: '.php', xml: '.xml', toml: '.toml', ini: '.ini' };
|
| const ext = extMap[full.language] || '.txt';
|
| const blob = new Blob([full.current_content || ''], { type: 'text/plain' });
|
| const a = document.createElement('a');
|
| a.href = URL.createObjectURL(blob);
|
| a.download = (full.title || 'document') + ext;
|
| a.click();
|
| URL.revokeObjectURL(a.href);
|
| } catch { if (uiModule) uiModule.showError('Failed to export document'); }
|
| });
|
| dropdown.appendChild(exportItem);
|
|
|
|
|
| const _archiveIco = '<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="21 8 21 21 3 21 3 8"/><rect x="1" y="3" width="22" height="5"/><line x1="10" y1="12" x2="14" y2="12"/></svg>';
|
| const archiveItem = document.createElement('button');
|
| archiveItem.className = 'dropdown-item-compact';
|
| archiveItem.style.cssText = 'background:none;border:none;width:100%;';
|
| archiveItem.innerHTML = _di(_archiveIco) + `<span>${_libraryArchivedView ? 'Restore' : 'Archive'}</span>`;
|
| archiveItem.title = _libraryArchivedView ? 'Restore to active documents' : 'Archive (hide from the main list)';
|
| archiveItem.addEventListener('click', async (e) => {
|
| e.stopPropagation();
|
| hideCardDropdown();
|
| const toArchived = !_libraryArchivedView;
|
| try {
|
| const res = await fetch(`${API_BASE}/api/document/${doc.id}/archive?archived=${toArchived}`, { method: 'POST', credentials: 'same-origin' });
|
| if (!res.ok) throw new Error('failed');
|
|
|
| libraryRemoveDocumentFromState(doc.id);
|
| libraryRenderGrid();
|
| if (uiModule) uiModule.showToast(toArchived ? 'Archived' : 'Restored');
|
| } catch { if (uiModule) uiModule.showError('Failed to ' + (toArchived ? 'archive' : 'restore')); }
|
| });
|
| dropdown.appendChild(archiveItem);
|
|
|
|
|
| const _deleteIco = '<svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="3 6 5 6 21 6"/><path d="M19 6l-1 14a2 2 0 0 1-2 2H8a2 2 0 0 1-2-2L5 6"/><path d="M10 11v6"/><path d="M14 11v6"/><path d="M9 6V4a1 1 0 0 1 1-1h4a1 1 0 0 1 1 1v2"/></svg>';
|
| const deleteItem = document.createElement('button');
|
| deleteItem.className = 'dropdown-item-compact dropdown-item-danger';
|
| deleteItem.style.cssText = 'background:none;border:none;width:100%;';
|
| deleteItem.innerHTML = _di(_deleteIco) + '<span>Delete</span>';
|
| deleteItem.addEventListener('click', (e) => { e.stopPropagation(); hideCardDropdown(); libraryDeleteSingle(doc.id, card); });
|
| dropdown.appendChild(deleteItem);
|
|
|
| menuWrap.appendChild(dropdown);
|
| actionsWrap.appendChild(menuWrap);
|
| card.appendChild(actionsWrap);
|
|
|
|
|
| card.appendChild(header);
|
|
|
|
|
| if (!document.getElementById('doclib-card-styles')) {
|
| const s = document.createElement('style');
|
| s.id = 'doclib-card-styles';
|
| s.textContent = `.doclib-card:hover .doclib-card-icon-btn{opacity:.4}.doclib-card-icon-btn:hover{opacity:1!important}.doclib-card-text-btn{background:none;border:1px solid var(--border);color:var(--fg-muted);font-size:10px;padding:3px 8px;border-radius:4px;cursor:pointer;transition:border-color .15s,color .15s}.doclib-card-text-btn:hover{border-color:var(--accent,var(--red));color:var(--accent,var(--red))}.doclib-card-text-btn-danger{border-color:var(--color-danger,#e06c75)!important;color:var(--color-danger,#e06c75)!important}.doclib-card-text-btn-danger:hover{border-color:#ff4d4d!important;color:#ff4d4d!important}.doclib-card-chevron{display:none;align-items:center;justify-content:center;align-self:center;opacity:0.6;transition:transform .15s ease;flex-shrink:0;height:14px;line-height:0}.doclib-card-expanded .doclib-card-chevron{display:inline-flex;transform:rotate(180deg)}.doclib-card-chevron svg{display:block}`;
|
| document.head.appendChild(s);
|
| }
|
|
|
|
|
| const preview = document.createElement('div');
|
| preview.className = 'doclib-card-preview';
|
| const pre = document.createElement('pre');
|
| const code = document.createElement('code');
|
| try {
|
| if (doc.language && doc.language !== 'text' && window.hljs && !_librarySearch) {
|
| code.innerHTML = window.hljs.highlight(doc.preview || '', { language: doc.language }).value;
|
| } else if (_librarySearch) {
|
|
|
|
|
| code.innerHTML = _hlSearch(doc.preview || '');
|
| } else {
|
| code.textContent = doc.preview || '';
|
| }
|
| } catch {
|
| code.textContent = doc.preview || '';
|
| }
|
| pre.appendChild(code);
|
| preview.appendChild(pre);
|
|
|
|
|
| const expandedActions = document.createElement('div');
|
| expandedActions.className = 'doclib-card-expanded-actions';
|
|
|
| const openBtn = document.createElement('button');
|
| openBtn.className = 'doclib-card-text-btn doclib-card-action-btn';
|
| openBtn.innerHTML = '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><path d="M5 12h14M13 5l7 7-7 7"/></svg>Open';
|
| if (doc.session_id) {
|
| openBtn.title = 'Open in original session';
|
| openBtn.addEventListener('click', (e) => { e.stopPropagation(); libraryOpenInSession(doc); });
|
| } else {
|
|
|
|
|
| openBtn.title = 'Open in the editor';
|
| openBtn.addEventListener('click', (e) => { e.stopPropagation(); libraryOpenDocument(doc); });
|
| }
|
|
|
| const cloneBtn = document.createElement('button');
|
| cloneBtn.className = 'doclib-card-text-btn doclib-card-action-btn';
|
| cloneBtn.innerHTML = '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><rect x="9" y="9" width="13" height="13" rx="2"/><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"/></svg>Clone';
|
| cloneBtn.title = 'Clone — copy to active session';
|
| cloneBtn.addEventListener('click', (e) => { e.stopPropagation(); libraryImportDocument(doc); });
|
|
|
| const deleteBtn = document.createElement('button');
|
| deleteBtn.className = 'doclib-card-text-btn doclib-card-action-btn doclib-card-text-btn-danger';
|
| deleteBtn.innerHTML = '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><polyline points="3 6 5 6 21 6"/><path d="M19 6l-1 14a2 2 0 0 1-2 2H8a2 2 0 0 1-2-2L5 6"/><path d="M10 11v6"/><path d="M14 11v6"/><path d="M9 6V4a1 1 0 0 1 1-1h4a1 1 0 0 1 1 1v2"/></svg>Delete';
|
| deleteBtn.addEventListener('click', (e) => { e.stopPropagation(); libraryDeleteSingle(doc.id, card); });
|
|
|
|
|
|
|
| const archiveBtn = document.createElement('button');
|
| archiveBtn.className = 'doclib-card-text-btn doclib-card-action-btn';
|
| archiveBtn.innerHTML = '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><polyline points="21 8 21 21 3 21 3 8"/><rect x="1" y="3" width="22" height="5"/><line x1="10" y1="12" x2="14" y2="12"/></svg>' + (_libraryArchivedView ? 'Restore' : 'Archive');
|
| archiveBtn.title = _libraryArchivedView ? 'Restore to active documents' : 'Archive (hide from the main list)';
|
| archiveBtn.addEventListener('click', async (e) => {
|
| e.stopPropagation();
|
| const toArchived = !_libraryArchivedView;
|
| try {
|
| const res = await fetch(`${API_BASE}/api/document/${doc.id}/archive?archived=${toArchived}`, { method: 'POST', credentials: 'same-origin' });
|
| if (!res.ok) throw new Error('failed');
|
| libraryRemoveDocumentFromState(doc.id);
|
| libraryRenderGrid();
|
| if (uiModule) uiModule.showToast(toArchived ? 'Archived' : 'Restored');
|
| } catch { if (uiModule) uiModule.showError('Failed to ' + (toArchived ? 'archive' : 'restore')); }
|
| });
|
|
|
| const leftGroup = document.createElement('div');
|
| leftGroup.className = 'doclib-action-group';
|
| const btnRow = document.createElement('div');
|
| btnRow.className = 'doclib-action-btn-row';
|
|
|
| btnRow.appendChild(cloneBtn);
|
| btnRow.appendChild(openBtn);
|
| leftGroup.appendChild(btnRow);
|
|
|
|
|
| deleteBtn.style.cssText += ';position:relative;left:-8px;';
|
| archiveBtn.style.cssText += ';position:relative;left:-8px;';
|
| expandedActions.appendChild(deleteBtn);
|
| expandedActions.appendChild(archiveBtn);
|
| expandedActions.appendChild(leftGroup);
|
|
|
| preview.appendChild(expandedActions);
|
| card.appendChild(preview);
|
|
|
| card.addEventListener('click', () => {
|
| if (card._suppressNextClick) { card._suppressNextClick = false; return; }
|
| if (_librarySelectMode) {
|
| const cb = card.querySelector('.memory-select-cb');
|
| if (cb) { cb.checked = !cb.checked; cb.dispatchEvent(new Event('change')); }
|
| } else {
|
| libraryExpandCard(card, doc);
|
| }
|
| });
|
| _attachLongPressMenu(card, '.memory-item-btn');
|
| return card;
|
| }
|
|
|
| async function libraryExpandCard(card, doc) {
|
| const grid = card.closest('.doclib-grid');
|
| const instant = card?.dataset?.spaceToggle === '1';
|
|
|
|
|
| if (card.classList.contains('doclib-card-expanded')) {
|
| _collapseExpandedCard(card);
|
| return;
|
| }
|
|
|
|
|
| if (grid) {
|
| grid.querySelectorAll('.doclib-card-expanded').forEach(c => _collapseExpandedCard(c));
|
| }
|
|
|
|
|
| const siblings = grid ? [...grid.querySelectorAll('.doclib-card')].filter(c => c !== card) : [];
|
|
|
| siblings.forEach(s => { s.style.opacity = '1'; });
|
|
|
| if (!instant) {
|
| if (siblings.length) siblings[0].offsetHeight;
|
| siblings.forEach(s => { s.style.transition = 'opacity 0.12s ease'; s.style.opacity = '0'; });
|
| }
|
|
|
|
|
|
|
|
|
| const isMobile = window.innerWidth <= 768;
|
| const toolbar = grid ? grid.closest('.admin-card')?.querySelector('.memory-toolbar') : null;
|
| const toolbarH = toolbar ? toolbar.offsetHeight : 0;
|
| if (grid && !isMobile) {
|
| grid.style.minHeight = (grid.offsetHeight + toolbarH) + 'px';
|
| grid.style.maxHeight = (grid.offsetHeight + toolbarH) + 'px';
|
| }
|
|
|
|
|
| if (!instant) await new Promise(r => setTimeout(r, 120));
|
|
|
| card.classList.add('doclib-card-expanded');
|
| if (grid) grid.scrollTop = 0;
|
|
|
|
|
| siblings.forEach(s => { s.style.transition = ''; s.style.opacity = ''; });
|
|
|
|
|
| const preview = card.querySelector('.doclib-card-preview');
|
| if (!preview) return;
|
|
|
| const actionsBar = preview.querySelector('.doclib-card-expanded-actions');
|
| const existingPre = preview.querySelector('pre');
|
|
|
| try {
|
| const res = await fetch(`${API_BASE}/api/document/${doc.id}`);
|
| if (!res.ok) throw new Error('Failed');
|
| const full = await res.json();
|
| const content = full.current_content || '';
|
| const lang = full.language || doc.language || 'text';
|
|
|
|
|
|
|
| const isPdfDoc = /<!--\s*pdf_(?:form_)?source\s+upload_id="[^"]+"/.test(content);
|
| const existingFrame = preview.querySelector('.doclib-card-pdf-frame');
|
|
|
| if (isPdfDoc) {
|
| const frame = document.createElement('iframe');
|
| frame.className = 'doclib-card-pdf-frame';
|
| frame.src = `${API_BASE}/api/document/${doc.id}/render-pdf?t=${Date.now()}`;
|
| frame.style.cssText = 'width:100%;height:60vh;border:1px solid var(--border);border-radius:6px;background:var(--bg);opacity:0;transition:opacity 0.15s ease;';
|
| if (existingPre) existingPre.remove();
|
| if (existingFrame) existingFrame.remove();
|
| preview.insertBefore(frame, preview.firstChild);
|
| if (actionsBar && !preview.contains(actionsBar)) preview.appendChild(actionsBar);
|
| requestAnimationFrame(() => { frame.style.opacity = '1'; });
|
| return;
|
| }
|
|
|
| const pre = document.createElement('pre');
|
| const code = document.createElement('code');
|
|
|
|
|
|
|
|
|
|
|
| const HL_CAP = 20000;
|
| try {
|
| if (lang && lang !== 'text' && lang !== 'markdown' && window.hljs && content.length <= HL_CAP) {
|
| code.innerHTML = window.hljs.highlight(content, { language: lang }).value;
|
| } else {
|
| code.textContent = content;
|
| }
|
| } catch {
|
| code.textContent = content;
|
| }
|
| pre.appendChild(code);
|
|
|
|
|
| if (existingPre) existingPre.remove();
|
| if (existingFrame) existingFrame.remove();
|
| pre.style.opacity = '0';
|
| preview.insertBefore(pre, preview.firstChild);
|
| if (actionsBar && !preview.contains(actionsBar)) preview.appendChild(actionsBar);
|
| requestAnimationFrame(() => {
|
| pre.style.transition = 'opacity 0.15s ease';
|
| pre.style.opacity = '1';
|
| });
|
| } catch (e) {
|
|
|
| if (!existingPre) {
|
| preview.innerHTML = '<div style="padding:8px;color:var(--color-error);font-size:10px;">Failed to load</div>';
|
| }
|
| if (actionsBar && !preview.contains(actionsBar)) preview.appendChild(actionsBar);
|
| }
|
| }
|
|
|
| function libraryRenderLoadMore() {
|
|
|
|
|
|
|
|
|
| const legacy = document.getElementById('doclib-load-more');
|
| if (legacy) legacy.style.display = 'none';
|
| }
|
|
|
| async function libraryOpenDocument(doc) {
|
| closeLibrary();
|
|
|
| if (!doc.session_id) {
|
| _loadDocument(doc.id);
|
| return;
|
| }
|
| const currentSessionId = sessionModule && sessionModule.getCurrentSessionId();
|
| if (doc.session_id !== currentSessionId) {
|
| await sessionModule.selectSession(doc.session_id);
|
| }
|
| _loadDocument(doc.id);
|
| }
|
|
|
|
|
| async function libraryOpenInSession(doc) {
|
| if (!doc.session_id) return;
|
| closeLibrary();
|
|
|
|
|
| const currentSessionId = sessionModule && sessionModule.getCurrentSessionId();
|
| if (doc.session_id !== currentSessionId) {
|
| await sessionModule.selectSession(doc.session_id);
|
|
|
| await new Promise(r => setTimeout(r, 150));
|
| }
|
|
|
|
|
| const docs = _getDocs();
|
| if (!docs.has(doc.id)) {
|
| const res = await fetch(`${API_BASE}/api/document/${doc.id}`);
|
| if (res.ok) {
|
| const full = await res.json();
|
| _addDocToTabs(full, doc.session_id);
|
| }
|
| }
|
|
|
|
|
| if (!_isOpenFn()) _openPanel();
|
|
|
| _switchToDoc(doc.id);
|
| _syncDocIndicator();
|
| }
|
|
|
|
|
| async function libraryImportDocument(doc) {
|
| let sessionId = sessionModule && sessionModule.getCurrentSessionId();
|
| if (!sessionId) {
|
|
|
| if (sessionModule && sessionModule.hasPendingChat && sessionModule.hasPendingChat()) {
|
| const ok = await sessionModule.materializePendingSession();
|
| if (ok) sessionId = sessionModule.getCurrentSessionId();
|
| }
|
| if (!sessionId) {
|
|
|
| const curModel = sessionModule.getCurrentModel ? sessionModule.getCurrentModel() : null;
|
| const sessions = sessionModule ? sessionModule.getSessions() : [];
|
|
|
| const withModel = sessions.filter(s => s.endpoint_url && s.model);
|
| const match = (curModel && withModel.find(s => s.model === curModel)) || withModel[0];
|
| if (match) {
|
| sessionModule.createDirectChat(match.endpoint_url, match.model, match.endpoint_id);
|
| const ok = await sessionModule.materializePendingSession();
|
| if (ok) sessionId = sessionModule.getCurrentSessionId();
|
| }
|
| }
|
| if (!sessionId) {
|
| if (uiModule) uiModule.showError('Could not create a session');
|
| return;
|
| }
|
| }
|
| try {
|
|
|
| const srcRes = await fetch(`${API_BASE}/api/document/${doc.id}`);
|
| if (!srcRes.ok) throw new Error('Failed to fetch document');
|
| const src = await srcRes.json();
|
|
|
|
|
| let baseTitle = src.title || doc.title || 'Untitled';
|
| const existingTitles = new Set();
|
| const docs = _getDocs();
|
| for (const [, d] of docs) {
|
| if (d.sessionId === sessionId && d.title) existingTitles.add(d.title);
|
| }
|
| if (existingTitles.has(baseTitle)) {
|
|
|
| const root = baseTitle.replace(/\s*\(\d+\)$/, '');
|
| let n = 2;
|
| while (existingTitles.has(root + ' (' + n + ')')) n++;
|
| baseTitle = root + ' (' + n + ')';
|
| }
|
|
|
|
|
| const res = await fetch(`${API_BASE}/api/document`, {
|
| method: 'POST',
|
| credentials: 'same-origin',
|
| headers: { 'Content-Type': 'application/json' },
|
| body: JSON.stringify({
|
| session_id: sessionId,
|
|
|
|
|
| language: src.language || doc.language || 'markdown',
|
| content: src.current_content || '',
|
| }),
|
| });
|
| if (!res.ok) throw new Error('Failed to create document');
|
| const created = await res.json();
|
| closeLibrary();
|
| _addDocToTabs(created, sessionId);
|
| if (!_isOpenFn()) _openPanel();
|
|
|
| _switchToDoc(created.id);
|
| _syncDocIndicator();
|
| if (uiModule) uiModule.showToast('Document cloned to session');
|
| } catch (e) {
|
| console.error('Failed to import document:', e);
|
| if (uiModule) uiModule.showError('Failed to import document');
|
| }
|
| }
|
|
|
|
|
|
|
| function libraryEnterSelectMode() {
|
| _librarySelectMode = true;
|
| _librarySelectedIds.clear();
|
| const bulkBar = document.getElementById('doclib-bulk-bar');
|
| const selectBtn = document.getElementById('doclib-select-btn');
|
| if (bulkBar) bulkBar.classList.remove('hidden');
|
| if (selectBtn) { selectBtn.classList.add('active'); selectBtn.textContent = 'Cancel'; }
|
| libraryUpdateBulkCount();
|
| libraryRenderGrid();
|
| }
|
|
|
| function libraryExitSelectMode() {
|
| _librarySelectMode = false;
|
| _librarySelectedIds.clear();
|
| const bulkBar = document.getElementById('doclib-bulk-bar');
|
| const selectBtn = document.getElementById('doclib-select-btn');
|
| const selectAll = document.getElementById('doclib-select-all');
|
| if (bulkBar) bulkBar.classList.add('hidden');
|
| if (selectBtn) { selectBtn.classList.remove('active'); selectBtn.textContent = 'Select'; }
|
| if (selectAll) selectAll.checked = false;
|
| libraryRenderGrid();
|
| }
|
|
|
| function libraryToggleSelectItem(id) {
|
| if (_librarySelectedIds.has(id)) {
|
| _librarySelectedIds.delete(id);
|
| } else {
|
| _librarySelectedIds.add(id);
|
| }
|
| libraryUpdateBulkCount();
|
| }
|
|
|
| function libraryToggleSelectAll() {
|
| const selectAllEl = document.getElementById('doclib-select-all');
|
| if (!selectAllEl) return;
|
| if (selectAllEl.checked) {
|
| _libraryDocs.forEach(d => _librarySelectedIds.add(d.id));
|
| } else {
|
| _librarySelectedIds.clear();
|
| }
|
| libraryUpdateBulkCount();
|
| libraryRenderGrid();
|
| }
|
|
|
| function libraryUpdateBulkCount() {
|
| const countEl = document.getElementById('doclib-selected-count');
|
| const actionsBtn = document.getElementById('doclib-bulk-actions');
|
| if (countEl) countEl.textContent = `${_librarySelectedIds.size} Selected`;
|
| if (actionsBtn) actionsBtn.style.color = _librarySelectedIds.size > 0 ? 'var(--fg)' : '';
|
|
|
|
|
| const deleteBtn = document.getElementById('doclib-bulk-delete');
|
| const exportBtn = document.getElementById('doclib-bulk-export');
|
| const archiveBtn = document.getElementById('doclib-bulk-archive');
|
| const cloneBtn = document.getElementById('doclib-bulk-clone');
|
| if (deleteBtn) deleteBtn.disabled = _librarySelectedIds.size === 0;
|
| if (exportBtn) exportBtn.disabled = _librarySelectedIds.size === 0;
|
| if (cloneBtn) cloneBtn.disabled = _librarySelectedIds.size === 0;
|
| if (archiveBtn) {
|
| archiveBtn.disabled = _librarySelectedIds.size === 0;
|
| archiveBtn.textContent = _libraryArchivedView ? 'Restore' : 'Archive';
|
| }
|
| }
|
|
|
| async function libraryDeleteSingle(docId, card) {
|
| if (uiModule && uiModule.styledConfirm) {
|
| const ok = await uiModule.styledConfirm('Delete this document?', { confirmText: 'Delete', danger: true });
|
| if (!ok) return;
|
| } else if (!confirm('Delete this document?')) {
|
| return;
|
| }
|
| try {
|
| const res = await fetch(`${API_BASE}/api/document/${docId}`, { method: 'DELETE', credentials: 'same-origin' });
|
| if (!res.ok) {
|
| let detail = `HTTP ${res.status}`;
|
| try { const j = await res.json(); if (j?.detail) detail = j.detail; } catch {}
|
| throw new Error(detail);
|
| }
|
| if (card) {
|
| card.classList.add('doclib-card-deleting');
|
| card.addEventListener('transitionend', () => card.remove(), { once: true });
|
| setTimeout(() => { if (card.parentElement) card.remove(); }, 400);
|
| }
|
| libraryRemoveDocumentFromState(docId);
|
| if (uiModule) uiModule.showToast('Document deleted');
|
| } catch (e) {
|
| if (uiModule) uiModule.showError(`Failed to delete document: ${e.message || e}`);
|
| }
|
| }
|
|
|
| async function libraryBulkDelete() {
|
| if (_librarySelectedIds.size === 0) return;
|
| const count = _librarySelectedIds.size;
|
| if (uiModule && uiModule.styledConfirm) {
|
| const ok = await uiModule.styledConfirm(
|
| `Delete ${count} document${count !== 1 ? 's' : ''}?`,
|
| { confirmText: 'Delete', danger: true }
|
| );
|
| if (!ok) return;
|
| } else if (!confirm(`Delete ${count} document${count !== 1 ? 's' : ''}?`)) {
|
| return;
|
| }
|
|
|
| let deleted = 0;
|
| let failed = 0;
|
| const deletedIds = [];
|
| for (const id of _librarySelectedIds) {
|
| try {
|
| const res = await fetch(`${API_BASE}/api/document/${id}`, { method: 'DELETE', credentials: 'same-origin' });
|
| if (res.ok) {
|
| deleted++;
|
| deletedIds.push(id);
|
| }
|
| else { failed++; console.warn('Delete failed for', id, 'status', res.status); }
|
| } catch (e) {
|
| failed++;
|
| console.error('Failed to delete document:', id, e);
|
| }
|
| }
|
|
|
| for (const id of deletedIds) {
|
| const card = document.querySelector(`.doclib-card[data-doc-id="${CSS.escape(String(id))}"]`);
|
| if (card) card.classList.add('doclib-card-deleting');
|
| }
|
| if (deletedIds.length) await new Promise(r => setTimeout(r, 320));
|
| libraryExitSelectMode();
|
| await libraryFetch(false);
|
| if (uiModule) {
|
| const msg = failed > 0
|
| ? `Deleted ${deleted} · ${failed} failed`
|
| : `Deleted ${deleted} document${deleted !== 1 ? 's' : ''}`;
|
| (failed > 0 ? uiModule.showError : uiModule.showToast)(msg);
|
| }
|
| }
|
|
|
| async function libraryBulkArchive() {
|
| if (_librarySelectedIds.size === 0) return;
|
| const toArchived = !_libraryArchivedView;
|
| const ids = [..._librarySelectedIds];
|
| let done = 0, failed = 0;
|
| for (const id of ids) {
|
| try {
|
| const res = await fetch(`${API_BASE}/api/document/${id}/archive?archived=${toArchived}`, { method: 'POST', credentials: 'same-origin' });
|
| if (res.ok) done++; else failed++;
|
| } catch { failed++; }
|
| }
|
| libraryExitSelectMode();
|
| await libraryFetch(false);
|
| if (uiModule) {
|
| const verb = toArchived ? 'Archived' : 'Restored';
|
| const msg = failed > 0 ? `${verb} ${done} · ${failed} failed` : `${verb} ${done} document${done !== 1 ? 's' : ''}`;
|
| (failed > 0 ? uiModule.showError : uiModule.showToast)(msg);
|
| }
|
| }
|
|
|
|
|
|
|
|
|
| async function libraryBulkClone() {
|
| if (_librarySelectedIds.size === 0) return;
|
| const ids = [..._librarySelectedIds];
|
| let done = 0, failed = 0;
|
| for (const id of ids) {
|
| const doc = _libraryDocs.find(d => d.id === id);
|
| if (!doc) { failed++; continue; }
|
| try {
|
| const ok = await libraryImportDocument(doc);
|
| if (ok === false) failed++; else done++;
|
| } catch { failed++; }
|
| }
|
| libraryExitSelectMode();
|
| if (uiModule) {
|
| const msg = failed > 0
|
| ? `Cloned ${done} · ${failed} failed`
|
| : `Cloned ${done} document${done !== 1 ? 's' : ''}`;
|
| (failed > 0 ? uiModule.showError : uiModule.showToast)(msg);
|
| }
|
| }
|
|
|
| async function libraryBulkExport() {
|
| if (_librarySelectedIds.size === 0) return;
|
|
|
|
|
| if (_librarySelectedIds.size > 5) {
|
| const ids = [..._librarySelectedIds];
|
| try {
|
| if (uiModule) uiModule.showToast(`Zipping ${ids.length} documents…`);
|
| const res = await fetch(`${API_BASE}/api/documents/export-zip`, {
|
| method: 'POST', credentials: 'same-origin',
|
| headers: { 'Content-Type': 'application/json' },
|
| body: JSON.stringify({ ids }),
|
| });
|
| if (!res.ok) throw new Error('zip failed');
|
| const blob = await res.blob();
|
| const url = URL.createObjectURL(blob);
|
| const a = document.createElement('a');
|
| a.href = url;
|
| a.download = 'documents.zip';
|
| document.body.appendChild(a);
|
| a.click();
|
| a.remove();
|
| setTimeout(() => URL.revokeObjectURL(url), 2000);
|
| if (uiModule) uiModule.showToast(`Exported ${ids.length} documents (zip)`);
|
| } catch (e) {
|
| if (uiModule) uiModule.showError('Failed to create zip');
|
| }
|
| return;
|
| }
|
| const extMap = {
|
| javascript: '.js', python: '.py', html: '.html', css: '.css',
|
| markdown: '.md', json: '.json', yaml: '.yml', bash: '.sh',
|
| sql: '.sql', rust: '.rs', go: '.go', java: '.java', c: '.c', cpp: '.cpp',
|
| typescript: '.ts', ruby: '.rb', php: '.php', text: '.txt',
|
| xml: '.xml', toml: '.toml', ini: '.ini',
|
| };
|
|
|
| const docs = await Promise.all([..._librarySelectedIds].map(async id => {
|
| try {
|
| const res = await fetch(`${API_BASE}/api/document/${id}`);
|
| if (!res.ok) return null;
|
| return await res.json();
|
| } catch (e) {
|
| console.error('Failed to export document:', id, e);
|
| return null;
|
| }
|
| }));
|
| for (const doc of docs) {
|
| if (!doc) continue;
|
| const ext = extMap[doc.language] || '.txt';
|
| const filename = (doc.title || 'document') + (doc.title && doc.title.includes('.') ? '' : ext);
|
| const blob = new Blob([doc.current_content || ''], { type: 'text/plain' });
|
| const url = URL.createObjectURL(blob);
|
| const a = document.createElement('a');
|
| a.href = url;
|
| a.download = filename;
|
| a.click();
|
| URL.revokeObjectURL(url);
|
| }
|
| if (uiModule) uiModule.showToast(`Exported ${_librarySelectedIds.size} document${_librarySelectedIds.size !== 1 ? 's' : ''}`);
|
| }
|
|
|
|
|
| let _xlsxReady = null;
|
| function ensureXLSX() {
|
| if (_xlsxReady) return _xlsxReady;
|
| if (window.XLSX) return (_xlsxReady = Promise.resolve());
|
| _xlsxReady = new Promise((resolve, reject) => {
|
| const s = document.createElement('script');
|
| s.src = '/static/lib/xlsx.full.min.js';
|
| s.onload = resolve;
|
| s.onerror = () => reject(new Error('Failed to load XLSX library'));
|
| document.head.appendChild(s);
|
| });
|
| return _xlsxReady;
|
| }
|
|
|
| let _mammothReady = null;
|
| function ensureMammoth() {
|
| if (_mammothReady) return _mammothReady;
|
| if (window.mammoth) return (_mammothReady = Promise.resolve());
|
| _mammothReady = new Promise((resolve, reject) => {
|
| const s = document.createElement('script');
|
| s.src = '/static/lib/mammoth.browser.min.js';
|
| s.onload = resolve;
|
| s.onerror = () => reject(new Error('Failed to load DOCX library'));
|
| document.head.appendChild(s);
|
| });
|
| return _mammothReady;
|
| }
|
|
|
|
|
| function htmlToMarkdown(html) {
|
| const doc = new DOMParser().parseFromString(html, 'text/html');
|
| let md = '';
|
| function walk(node) {
|
| if (node.nodeType === 3) { md += node.textContent; return; }
|
| if (node.nodeType !== 1) return;
|
| const tag = node.tagName.toLowerCase();
|
| if (tag === 'h1') { md += '\n# '; walkChildren(node); md += '\n'; }
|
| else if (tag === 'h2') { md += '\n## '; walkChildren(node); md += '\n'; }
|
| else if (tag === 'h3') { md += '\n### '; walkChildren(node); md += '\n'; }
|
| else if (tag === 'h4') { md += '\n#### '; walkChildren(node); md += '\n'; }
|
| else if (tag === 'strong' || tag === 'b') { md += '**'; walkChildren(node); md += '**'; }
|
| else if (tag === 'em' || tag === 'i') { md += '*'; walkChildren(node); md += '*'; }
|
| else if (tag === 'a') { md += '['; walkChildren(node); md += `](${node.href || ''})`; }
|
| else if (tag === 'br') { md += '\n'; }
|
| else if (tag === 'p') { md += '\n'; walkChildren(node); md += '\n'; }
|
| else if (tag === 'ul' || tag === 'ol') { md += '\n'; walkChildren(node); }
|
| else if (tag === 'li') {
|
| const parent = node.parentElement?.tagName?.toLowerCase();
|
| if (parent === 'ol') {
|
| const idx = Array.from(node.parentElement.children).indexOf(node) + 1;
|
| md += `${idx}. `;
|
| } else { md += '- '; }
|
| walkChildren(node);
|
| md += '\n';
|
| }
|
| else if (tag === 'table') { md += '\n'; convertTable(node); md += '\n'; }
|
| else if (tag === 'img') {
|
|
|
| const src = node.src || '';
|
| if (!src.startsWith('data:')) {
|
| md += ``;
|
| } else if (node.alt) {
|
| md += `*[image: ${node.alt}]*`;
|
| }
|
| }
|
| else { walkChildren(node); }
|
| }
|
| function walkChildren(node) { for (const child of node.childNodes) walk(child); }
|
| function convertTable(table) {
|
| const rows = table.querySelectorAll('tr');
|
| rows.forEach((tr, i) => {
|
| const cells = tr.querySelectorAll('th, td');
|
| md += '| ' + Array.from(cells).map(c => c.textContent.trim()).join(' | ') + ' |\n';
|
| if (i === 0) md += '| ' + Array.from(cells).map(() => '---').join(' | ') + ' |\n';
|
| });
|
| }
|
| walkChildren(doc.body);
|
| return md.replace(/\n{3,}/g, '\n\n').trim();
|
| }
|
|
|
|
|
| async function readFileContent(file) {
|
| const name = file.name.toLowerCase();
|
| const isSpreadsheet = name.endsWith('.xlsx') || name.endsWith('.xls') || name.endsWith('.ods');
|
| const isDocx = name.endsWith('.docx');
|
|
|
| if (isSpreadsheet) {
|
| await ensureXLSX();
|
| const buf = await file.arrayBuffer();
|
| const wb = window.XLSX.read(buf, { type: 'array' });
|
|
|
| const parts = [];
|
| for (const sheetName of wb.SheetNames) {
|
| if (wb.SheetNames.length > 1) parts.push(`# Sheet: ${sheetName}`);
|
| parts.push(window.XLSX.utils.sheet_to_csv(wb.Sheets[sheetName]));
|
| }
|
| return parts.join('\n\n');
|
| }
|
|
|
| if (isDocx) {
|
| await ensureMammoth();
|
| const buf = await file.arrayBuffer();
|
| const result = await window.mammoth.convertToHtml({ arrayBuffer: buf });
|
| return htmlToMarkdown(result.value);
|
| }
|
|
|
|
|
| return new Promise((resolve, reject) => {
|
| const reader = new FileReader();
|
| reader.onload = () => resolve(reader.result);
|
| reader.onerror = () => reject(reader.error);
|
| reader.readAsText(file);
|
| });
|
| }
|
|
|
|
|
| async function libraryImportFiles(fileList) {
|
| const EXT_TO_LANG = {
|
| '.py': 'python', '.js': 'javascript', '.ts': 'typescript',
|
| '.html': 'html', '.htm': 'html', '.css': 'css', '.md': 'markdown',
|
| '.json': 'json', '.yml': 'yaml', '.yaml': 'yaml', '.sh': 'bash',
|
| '.bash': 'bash', '.sql': 'sql', '.rs': 'rust', '.go': 'go',
|
| '.java': 'java', '.c': 'c', '.cpp': 'cpp', '.h': 'c', '.hpp': 'cpp',
|
| '.rb': 'ruby', '.php': 'php', '.xml': 'xml',
|
| '.toml': 'toml', '.ini': 'ini', '.txt': '', '.log': '',
|
| '.cfg': 'ini', '.conf': 'ini', '.env': '', '.jsx': 'javascript',
|
| '.tsx': 'typescript', '.vue': 'html', '.svelte': 'html',
|
| '.scss': 'css', '.sass': 'css', '.less': 'css',
|
| '.csv': 'csv', '.tsv': 'csv',
|
| '.xlsx': 'csv', '.xls': 'csv', '.ods': 'csv',
|
| '.docx': 'markdown', '.doc': 'markdown',
|
| };
|
|
|
| let imported = 0;
|
| let failed = 0;
|
| let _firstErr = '';
|
|
|
|
|
|
|
| for (const file of fileList) {
|
| try {
|
| const name = file.name;
|
| const dotIdx = name.lastIndexOf('.');
|
| const ext = dotIdx >= 0 ? name.slice(dotIdx).toLowerCase() : '';
|
| const baseTitle = dotIdx > 0 ? name.slice(0, dotIdx) : name;
|
| const language = EXT_TO_LANG[ext] !== undefined ? EXT_TO_LANG[ext] : null;
|
|
|
| const isSpreadsheet = ['.xlsx', '.xls', '.ods'].includes(ext);
|
| const isPdf = ext === '.pdf';
|
|
|
| if (isPdf) {
|
|
|
|
|
|
|
| const fd = new FormData();
|
| fd.append('file', file);
|
| const res = await fetch(`${API_BASE}/api/documents/import-pdf`, {
|
| method: 'POST',
|
| body: fd,
|
| });
|
| if (!res.ok) {
|
| let _e = `HTTP ${res.status}`;
|
| try { const _j = await res.json(); _e = _j.detail || _j.error || _e; } catch {}
|
| throw new Error('PDF import failed: ' + _e);
|
| }
|
| imported++;
|
| continue;
|
| }
|
|
|
| if (isSpreadsheet) {
|
|
|
| await ensureXLSX();
|
| const buf = await file.arrayBuffer();
|
| const wb = window.XLSX.read(buf, { type: 'array' });
|
| for (const sheetName of wb.SheetNames) {
|
| const csv = window.XLSX.utils.sheet_to_csv(wb.Sheets[sheetName]);
|
| if (!csv.trim()) continue;
|
| const sheetTitle = wb.SheetNames.length > 1
|
| ? `${baseTitle} - ${sheetName}` : baseTitle;
|
| const res = await fetch(`${API_BASE}/api/document`, {
|
| method: 'POST',
|
| headers: { 'Content-Type': 'application/json' },
|
| body: JSON.stringify({ title: sheetTitle, language: 'csv', content: csv }),
|
| });
|
| if (!res.ok) throw new Error('Server error');
|
| }
|
| imported++;
|
| } else {
|
| const content = await readFileContent(file);
|
| const res = await fetch(`${API_BASE}/api/document`, {
|
| method: 'POST',
|
| headers: { 'Content-Type': 'application/json' },
|
| body: JSON.stringify({ title: baseTitle, language, content }),
|
| });
|
| if (!res.ok) throw new Error('Server error');
|
| imported++;
|
| }
|
| } catch (e) {
|
| console.error('Failed to import file:', file.name, e);
|
| if (!_firstErr) _firstErr = (e && e.message) || String(e);
|
| failed++;
|
| }
|
| }
|
|
|
| const msg = `Imported ${imported} file${imported !== 1 ? 's' : ''}` +
|
| (failed ? `, ${failed} failed${_firstErr ? ' — ' + _firstErr : ''}` : '');
|
| if (failed && uiModule) uiModule.showError(msg);
|
| else if (uiModule) uiModule.showToast(msg);
|
| await libraryFetch(false);
|
| }
|
|
|
| export function openLibrary(opts) {
|
| if (_libraryOpen) {
|
|
|
|
|
|
|
| const existing = document.getElementById('doclib-modal');
|
| if (!existing || existing.classList.contains('hidden')) {
|
| if (existing) existing.remove();
|
| _libraryOpen = false;
|
| } else {
|
| return;
|
| }
|
| }
|
| _libraryOpen = true;
|
| _libraryImportMode = !!(opts && opts.import);
|
| _librarySelectMode = false;
|
| _librarySelectedIds.clear();
|
| _librarySearch = '';
|
| _libraryActiveLanguage = null;
|
| _librarySort = 'recent';
|
| _libraryOffset = 0;
|
| _libraryDocs = [];
|
|
|
|
|
| const modal = document.createElement('div');
|
| modal.className = 'modal';
|
| modal.id = 'doclib-modal';
|
| modal.innerHTML = `
|
| <div class="modal-content doclib-modal-content" style="width:min(640px, 92vw);max-height:85vh;background:var(--bg);">
|
| <div class="modal-header">
|
| <!-- Header title + icon mirror the currently-active sub-tab (Chats /
|
| Documents / Research / Archive) so the user sees ONE icon at
|
| the top representing the section they're in, with the tab
|
| strip below as sub-navigation. _switchLibTab() updates this. -->
|
| <h4 id="doclib-header-title"><span id="doclib-header-icon" style="vertical-align:-2px;margin-right:4px;display:inline-flex;"></span><span id="doclib-header-text">Library</span></h4>
|
| <button class="close-btn" id="doclib-close">\u2716</button>
|
| </div>
|
| <div class="lib-tabs" id="doclib-lib-tabs" style="padding:0 10px;">
|
| <button class="lib-tab" data-doclib-tab="chats"><svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"/></svg>Chats</button>
|
| <button class="lib-tab active" data-doclib-tab="documents"><svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/><polyline points="14 2 14 8 20 8"/><line x1="8" y1="13" x2="16" y2="13"/><line x1="8" y1="17" x2="13" y2="17"/></svg>Documents</button>
|
| <button class="lib-tab" data-doclib-tab="research"><svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" style="vertical-align:-1px;margin-right:3px;"><circle cx="11" cy="11" r="8"/><path d="M21 21l-4.35-4.35"/><line x1="11" y1="8" x2="11" y2="14"/><line x1="8" y1="11" x2="14" y2="11"/></svg>Research</button>
|
| <button class="lib-tab" data-doclib-tab="archive"><svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><polyline points="21 8 21 21 3 21 3 8"/><rect x="1" y="3" width="22" height="5"/><line x1="10" y1="12" x2="14" y2="12"/></svg>Archive</button>
|
| </div>
|
| <div class="modal-body" style="display:flex;flex-direction:column;gap:10px;overflow:hidden;">
|
| <div id="doclib-panel-chats" data-doclib-panel="chats" class="admin-card" style="display:none;flex:1;flex-direction:column;overflow:hidden;">
|
| <div style="display:flex;align-items:baseline;gap:8px;margin-bottom:2px;">
|
| <h2 style="margin:0;padding:0;line-height:1;">Chats <span id="doclib-chats-stats" class="memory-count" style="font-size:0.6em;opacity:0.6;font-weight:normal"></span></h2>
|
| </div>
|
| <p class="memory-desc doclib-desc">All active chat sessions. Click to open.</p>
|
| <div class="memory-toolbar">
|
| <div class="memory-category-filters">
|
| <select class="memory-sort-select" id="doclib-chats-sort">
|
| <option value="recent">Recent</option>
|
| <option value="oldest">Oldest</option>
|
| <option value="most-messages">Most messages</option>
|
| <option value="alpha">A\u2013Z</option>
|
| </select>
|
| <button class="memory-toolbar-btn" id="doclib-chats-select-btn">Select</button>
|
| <button class="memory-toolbar-btn" id="doclib-chats-tidy-btn" title="AI tidy: delete junk sessions and organize into folders"><svg width="11" height="11" viewBox="0 0 24 24" fill="currentColor" style="vertical-align:-1px;margin-right:2px;"><path d="M12 0L14.59 8.41L23 12L14.59 15.59L12 24L9.41 15.59L1 12L9.41 8.41Z"/></svg> Tidy</button>
|
| </div>
|
| <input type="text" id="doclib-chats-search" placeholder="Search chats\u2026" class="memory-search-input" />
|
| <div id="doclib-chats-chips" class="doclib-lang-chips"></div>
|
| </div>
|
| <div id="doclib-chats-bulk" class="memory-bulk-bar hidden" style="margin-bottom:5px;">
|
| <label class="memory-bulk-check-all" style="position:relative;top:0px;left:-1px;"><input type="checkbox" id="doclib-chats-select-all" style="position:relative;top:0px;"> All</label>
|
| <span id="doclib-chats-selected-count">0 Selected</span>
|
| <button class="memory-toolbar-btn" id="doclib-chats-bulk-archive" style="position:relative;top:-3px;left:2px;"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><polyline points="21 8 21 21 3 21 3 8"/><rect x="1" y="3" width="22" height="5"/><line x1="10" y1="12" x2="14" y2="12"/></svg>Archive</button>
|
| <button class="memory-toolbar-btn danger" id="doclib-chats-bulk-delete" style="position:relative;left:2px;"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><polyline points="3 6 5 6 21 6"/><path d="M19 6l-1 14a2 2 0 0 1-2 2H8a2 2 0 0 1-2-2L5 6"/><path d="M10 11v6"/><path d="M14 11v6"/></svg>Delete</button>
|
| <button class="memory-toolbar-btn" id="doclib-chats-bulk-cancel" title="Cancel (Esc)" style="margin-left:4px;padding:3px 6px;position:relative;left:2px;"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round"><line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/></svg></button>
|
| </div>
|
| <div id="doclib-chats-grid" class="doclib-grid"></div>
|
| </div>
|
| <div id="doclib-panel-archive" data-doclib-panel="archive" class="admin-card" style="display:none;flex:1;flex-direction:column;overflow:hidden;">
|
| <div style="display:flex;align-items:baseline;gap:8px;margin-bottom:2px;">
|
| <h2 style="margin:0;padding:0;line-height:1;position:relative;top:2px;">Archive <span id="doclib-arc-stats" class="memory-count" style="font-size:0.6em;opacity:0.6;font-weight:normal"></span></h2>
|
| </div>
|
| <p class="memory-desc doclib-desc" style="position:relative;top:0.5px;">Archived sessions. Restore to make active again.</p>
|
| <div class="memory-toolbar">
|
| <div class="memory-category-filters">
|
| <select class="memory-sort-select" id="doclib-arc-sort">
|
| <option value="recent">Recent</option>
|
| <option value="oldest">Oldest</option>
|
| <option value="most-messages">Most messages</option>
|
| <option value="alpha">A\u2013Z</option>
|
| </select>
|
| <button class="memory-toolbar-btn" id="doclib-arc-select-btn">Select</button>
|
| </div>
|
| <input type="text" id="doclib-arc-search" placeholder="Search archive\u2026" class="memory-search-input" />
|
| <div id="doclib-arc-chips" class="doclib-lang-chips"></div>
|
| </div>
|
| <div id="doclib-arc-bulk" class="memory-bulk-bar hidden" style="margin-bottom:5px;">
|
| <label class="memory-bulk-check-all" style="position:relative;top:0px;left:1px;"><input type="checkbox" id="doclib-arc-select-all"> All</label>
|
| <span id="doclib-arc-selected-count">0 Selected</span>
|
| <button class="memory-toolbar-btn" id="doclib-arc-bulk-restore" style="position:relative;top:-3px;"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><polyline points="1 4 1 10 7 10"/><path d="M3.51 15a9 9 0 1 0 2.13-9.36L1 10"/></svg>Restore</button>
|
| <button class="memory-toolbar-btn danger" id="doclib-arc-bulk-delete"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><polyline points="3 6 5 6 21 6"/><path d="M19 6l-1 14a2 2 0 0 1-2 2H8a2 2 0 0 1-2-2L5 6"/><path d="M10 11v6"/><path d="M14 11v6"/></svg>Delete</button>
|
| <button class="memory-toolbar-btn" id="doclib-arc-bulk-cancel" title="Cancel (Esc)" style="margin-left:4px;padding:3px 6px;"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round"><line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/></svg></button>
|
| </div>
|
| <div id="doclib-arc-grid" class="doclib-grid"></div>
|
| </div>
|
| <div id="doclib-panel-research" data-doclib-panel="research" class="admin-card" style="display:none;flex:1;flex-direction:column;overflow:hidden;">
|
| <div style="display:flex;align-items:baseline;gap:8px;margin-bottom:2px;margin-top:10px;">
|
| <h2 style="margin:0;padding:0;line-height:1;">Research <span id="doclib-research-stats" class="memory-count" style="font-size:0.6em;opacity:0.6;font-weight:normal"></span></h2>
|
| </div>
|
| <p class="memory-desc doclib-desc" style="position:relative;top:-1px;">Completed deep research reports. Click to view.</p>
|
| <div class="memory-toolbar">
|
| <div class="memory-category-filters">
|
| <select class="memory-sort-select" id="doclib-research-sort">
|
| <option value="recent">Recent</option>
|
| <option value="oldest">Oldest</option>
|
| <option value="most-sources">Most sources</option>
|
| <option value="alpha">A\u2013Z</option>
|
| </select>
|
| <button class="memory-toolbar-btn" id="doclib-research-select-btn">Select</button>
|
| <button class="memory-toolbar-btn" id="doclib-research-tidy-btn" title="Tidy: delete research with no sources or empty reports">Tidy</button>
|
| </div>
|
| <input type="text" id="doclib-research-search" placeholder="Search research\u2026" class="memory-search-input" />
|
| </div>
|
| <div id="doclib-research-bulk" class="memory-bulk-bar hidden" style="margin-bottom:5px;">
|
| <label class="memory-bulk-check-all" style="position:relative;top:0px;left:1px;"><input type="checkbox" id="doclib-research-select-all"> All</label>
|
| <span id="doclib-research-selected-count">0 Selected</span>
|
| <button class="memory-toolbar-btn" id="doclib-research-bulk-archive" style="position:relative;top:-2px;"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><polyline points="21 8 21 21 3 21 3 8"/><rect x="1" y="3" width="22" height="5"/><line x1="10" y1="12" x2="14" y2="12"/></svg>Archive</button>
|
| <button class="memory-toolbar-btn danger" id="doclib-research-bulk-delete" style="position:relative;top:-2px;"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><polyline points="3 6 5 6 21 6"/><path d="M19 6l-1 14a2 2 0 0 1-2 2H8a2 2 0 0 1-2-2L5 6"/><path d="M10 11v6"/><path d="M14 11v6"/></svg>Delete</button>
|
| <button class="memory-toolbar-btn" id="doclib-research-bulk-cancel" title="Cancel (Esc)" style="margin-left:4px;padding:3px 6px;position:relative;top:-2px;"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round"><line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/></svg></button>
|
| </div>
|
| <div id="doclib-research-grid" class="doclib-grid"></div>
|
| </div>
|
| <div data-doclib-panel="documents" class="admin-card" style="flex:1;display:flex;flex-direction:column;overflow:hidden;">
|
| <div style="display:flex;align-items:baseline;gap:8px;margin-bottom:2px;">
|
| <h2 style="margin:0;padding:0;line-height:1;">Documents <span id="doclib-stats" class="memory-count" style="font-size:0.6em;opacity:0.6;font-weight:normal"></span></h2>
|
| <button class="memory-toolbar-btn" id="doclib-import-file-btn" title="Import files from disk" style="margin-left:auto;"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:2px;"><polyline points="7 10 12 5 17 10"/><line x1="12" y1="5" x2="12" y2="19"/><line x1="5" y1="21" x2="19" y2="21"/></svg> Import</button>
|
| <button class="memory-toolbar-btn" id="doclib-create-btn" title="Create new blank document"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/><polyline points="14 2 14 8 20 8"/><line x1="16" y1="13" x2="8" y2="13"/><line x1="16" y1="17" x2="8" y2="17"/><polyline points="10 9 9 9 8 9"/></svg> Create</button>
|
| </div>
|
| <p class="memory-desc doclib-desc">Open documents in a session, clone to a new or import new files.</p>
|
| <div class="memory-toolbar">
|
| <div class="memory-category-filters">
|
| <select class="memory-sort-select" id="doclib-sort">
|
| <option value="recent">Recent</option>
|
| <option value="oldest">Oldest</option>
|
| <option value="edits">Most edits</option>
|
| <option value="alpha">A\u2013Z</option>
|
| </select>
|
| <button class="memory-toolbar-btn" id="doclib-select-btn" title="Select documents">Select</button>
|
| <button class="memory-toolbar-btn" id="doclib-tidy-btn" title="Tidy: remove empty / junk / duplicate documents">Tidy</button>
|
| </div>
|
| <input type="text" id="doclib-search" placeholder="Search titles & content\u2026" class="memory-search-input" />
|
| <div id="doclib-chips" class="doclib-lang-chips"></div>
|
| </div>
|
| <input type="file" id="doclib-file-input" multiple style="display:none" />
|
| <div id="doclib-bulk-bar" class="memory-bulk-bar hidden" style="margin-bottom:5px;">
|
| <label class="memory-bulk-check-all" style="position:relative;top:0px;left:1px;"><input type="checkbox" id="doclib-select-all" /> All</label>
|
| <span id="doclib-selected-count">0 Selected</span>
|
| <button id="doclib-bulk-actions" class="memory-toolbar-btn" style="position:relative;top:-2px;margin-left:auto;"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-1px;margin-right:3px;"><line x1="8" y1="6" x2="21" y2="6"/><line x1="8" y1="12" x2="21" y2="12"/><line x1="8" y1="18" x2="21" y2="18"/><line x1="3" y1="6" x2="3.01" y2="6"/><line x1="3" y1="12" x2="3.01" y2="12"/><line x1="3" y1="18" x2="3.01" y2="18"/></svg>Actions <span style="opacity:0.55;font-size:9px;">▼</span></button>
|
| <button id="doclib-bulk-cancel" class="memory-toolbar-btn" title="Cancel (Esc)" style="margin-left:4px;margin-right:4px;padding:3px 6px;position:relative;top:-2px;"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2.5" stroke-linecap="round"><line x1="18" y1="6" x2="6" y2="18"/><line x1="6" y1="6" x2="18" y2="18"/></svg></button>
|
| </div>
|
| <div class="doclib-grid" id="doclib-grid"></div>
|
| <button class="doclib-load-more" id="doclib-load-more" style="display:none">Load more</button>
|
| </div>
|
| </div>
|
| </div>
|
| `;
|
| document.body.appendChild(modal);
|
|
|
|
|
| {
|
| const content = modal.querySelector('.modal-content');
|
| const header = modal.querySelector('.modal-header');
|
| if (content && header) {
|
|
|
| try {
|
| const saved = JSON.parse(localStorage.getItem('doclib-pos'));
|
| if (saved && saved.fullscreen) {
|
| localStorage.removeItem('doclib-pos');
|
| } else if (saved && saved.left && saved.top) {
|
| content.style.position = 'fixed';
|
| content.style.left = saved.left;
|
| content.style.top = saved.top;
|
| content.style.margin = '0';
|
|
|
| requestAnimationFrame(() => {
|
| const r = content.getBoundingClientRect();
|
| if (r.right > window.innerWidth) content.style.left = Math.max(0, window.innerWidth - r.width - 8) + 'px';
|
| if (r.bottom > window.innerHeight) content.style.top = Math.max(0, window.innerHeight - r.height - 8) + 'px';
|
| if (r.left < 0) content.style.left = '8px';
|
| if (r.top < 0) content.style.top = '8px';
|
| });
|
| }
|
| } catch {}
|
|
|
|
|
|
|
| const FS_CLASS = 'doclib-fullscreen';
|
| const enterFullscreen = () => {
|
| if (modal.classList.contains(FS_CLASS)) return;
|
| modal.classList.add(FS_CLASS);
|
| content.style.position = 'fixed';
|
| content.style.left = '0';
|
| content.style.top = '0';
|
| content.style.right = '0';
|
| content.style.bottom = '0';
|
| content.style.width = '100vw';
|
| content.style.maxWidth = '100vw';
|
| content.style.height = '100vh';
|
| content.style.maxHeight = '100vh';
|
| content.style.borderRadius = '0';
|
| content.style.margin = '0';
|
| content.style.transform = 'none';
|
| try { localStorage.setItem('doclib-pos', JSON.stringify({ fullscreen: true })); } catch {}
|
| };
|
| const exitFullscreen = (cx, cy) => {
|
| if (!modal.classList.contains(FS_CLASS)) return;
|
| modal.classList.remove(FS_CLASS);
|
| content.style.width = '';
|
| content.style.maxWidth = '';
|
| content.style.height = '';
|
| content.style.maxHeight = '';
|
| content.style.borderRadius = '';
|
| content.style.right = '';
|
| content.style.bottom = '';
|
| const r0 = content.getBoundingClientRect();
|
| const w = r0.width || Math.min(900, window.innerWidth * 0.92);
|
| content.style.left = Math.max(8, cx - w / 2) + 'px';
|
| content.style.top = Math.max(8, cy - 20) + 'px';
|
| };
|
| makeWindowDraggable(modal, {
|
| content,
|
| header,
|
| fsClass: FS_CLASS,
|
| skipSelector: '.modal-close',
|
| onEnterFullscreen: enterFullscreen,
|
| onExitFullscreen: exitFullscreen,
|
| enableFullscreen: false,
|
| onDragEnd: () => {
|
| try { localStorage.setItem('doclib-pos', JSON.stringify({ left: content.style.left, top: content.style.top })); } catch {}
|
| },
|
| });
|
| }
|
| }
|
|
|
|
|
| document.getElementById('doclib-close').addEventListener('click', closeLibrary);
|
|
|
|
|
| let _activeLibTab = (opts && opts.tab) || 'documents';
|
| const _tabBtns = modal.querySelectorAll('[data-doclib-tab]');
|
| const _tabPanels = modal.querySelectorAll('[data-doclib-panel]');
|
|
|
|
|
|
|
|
|
| const _LIB_PAGE_SIZE = 20;
|
| let _chatsVisibleLimit = _LIB_PAGE_SIZE;
|
| let _arcVisibleLimit = _LIB_PAGE_SIZE;
|
| let _researchVisibleLimit = _LIB_PAGE_SIZE;
|
|
|
| function _appendInlineLoadMore(grid, totalCount, currentLimit, onClick) {
|
| if (!grid || !grid.parentElement) return;
|
|
|
|
|
| grid.parentElement.querySelectorAll(':scope > .doclib-inline-load-more').forEach(b => b.remove());
|
| if (totalCount <= currentLimit) return;
|
| const btn = document.createElement('button');
|
| btn.className = 'doclib-load-more doclib-inline-load-more';
|
| btn.textContent = `Load more (${currentLimit} of ${totalCount})`;
|
| btn.addEventListener('click', onClick);
|
| grid.parentElement.appendChild(btn);
|
| }
|
|
|
|
|
|
|
| const _TAB_HEADERS = {
|
| chats: {
|
| label: 'Chats',
|
| svg: '<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"/></svg>',
|
| },
|
| documents: {
|
| label: 'Documents',
|
| svg: '<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/><polyline points="14 2 14 8 20 8"/><line x1="8" y1="13" x2="16" y2="13"/><line x1="8" y1="17" x2="13" y2="17"/></svg>',
|
| },
|
| research: {
|
| label: 'Research',
|
| svg: '<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2"><circle cx="11" cy="11" r="8"/><path d="M21 21l-4.35-4.35"/><line x1="11" y1="8" x2="11" y2="14"/><line x1="8" y1="11" x2="14" y2="11"/></svg>',
|
| },
|
| archive: {
|
| label: 'Archive',
|
| svg: '<svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="21 8 21 21 3 21 3 8"/><rect x="1" y="3" width="22" height="5"/><line x1="10" y1="12" x2="14" y2="12"/></svg>',
|
| },
|
| };
|
|
|
| function _switchLibTab(tab) {
|
| _activeLibTab = tab;
|
| _tabBtns.forEach(b => b.classList.toggle('active', b.dataset.doclibTab === tab));
|
| _tabPanels.forEach(p => {
|
| if (p.dataset.doclibPanel === tab) {
|
| p.style.display = 'flex';
|
| } else {
|
| p.style.display = 'none';
|
| }
|
| });
|
|
|
| const hdr = _TAB_HEADERS[tab];
|
| if (hdr) {
|
| const ico = document.getElementById('doclib-header-icon');
|
| const txt = document.getElementById('doclib-header-text');
|
| if (ico) ico.innerHTML = hdr.svg;
|
| if (txt) txt.textContent = hdr.label;
|
| }
|
| if (tab === 'chats') _renderLibChats();
|
| else if (tab === 'archive') _renderLibArchive();
|
| else if (tab === 'research') _renderLibResearch();
|
| }
|
|
|
| _tabBtns.forEach(btn => {
|
| btn.addEventListener('click', () => _switchLibTab(btn.dataset.doclibTab));
|
| });
|
|
|
|
|
| let _chatsSessions = [];
|
| let _chatsSearch = '';
|
| let _chatsSort = 'recent';
|
| let _chatsSelectMode = false;
|
| const _chatsSelected = new Set();
|
| let _chatsModelFilter = '';
|
|
|
| function _renderLibChats() {
|
| const grid = document.getElementById('doclib-chats-grid');
|
| if (!grid) return;
|
| grid.innerHTML = '';
|
| grid.appendChild(spinnerModule.createLoadingRow('Loading…'));
|
| fetch(API_BASE + '/api/sessions', { credentials: 'same-origin' }).then(r => r.json()).then(data => {
|
| const raw = Array.isArray(data) ? data : (data.sessions || []);
|
| _chatsSessions = raw.filter(s => !s.archived);
|
| _renderChatsGrid();
|
| _renderChatsChips();
|
| }).catch(() => { grid.innerHTML = '<div class="doclib-empty">Failed to load</div>'; });
|
| }
|
|
|
|
|
|
|
|
|
| async function _toggleChatPreview(card, session) {
|
| const preview = card.querySelector('.doclib-chat-preview');
|
| if (!preview) return;
|
| const isOpen = card.classList.contains('doclib-card-expanded');
|
|
|
| const grid = card.closest('.doclib-grid');
|
| if (grid) {
|
| grid.querySelectorAll('.doclib-card-expanded').forEach(c => {
|
| if (c !== card) {
|
| c.classList.remove('doclib-card-expanded');
|
| const p = c.querySelector('.doclib-chat-preview');
|
| if (p) { p.style.display = 'none'; p.innerHTML = ''; }
|
| }
|
| });
|
| }
|
| if (isOpen) {
|
| card.classList.remove('doclib-card-expanded');
|
| preview.style.display = 'none';
|
| preview.innerHTML = '';
|
| return;
|
| }
|
| card.classList.add('doclib-card-expanded');
|
| preview.style.display = 'block';
|
| preview.innerHTML = '<div style="opacity:0.4;font-size:11px;padding:8px 4px;">Loading…</div>';
|
| try {
|
| const res = await fetch(`${API_BASE}/api/history/${session.id}`, { credentials: 'same-origin' });
|
| if (!res.ok) throw new Error('Failed');
|
| const data = await res.json();
|
| const history = Array.isArray(data) ? data : (data.history || []);
|
| const recent = history.filter(m => m.role === 'user' || m.role === 'assistant').slice(-5);
|
| const sessionModel = (session.model || '').split('/').pop();
|
| const msgsHtml = recent.length
|
| ? recent.map(m => {
|
| const isUser = m.role === 'user';
|
| const raw = m.content || '';
|
| const truncated = raw.length > 600 ? raw.slice(0, 600) + '…' : raw;
|
|
|
|
|
| const cleaned = truncated
|
| .replace(/<think>[\s\S]*?<\/think>/g, '')
|
| .replace(/<think>[\s\S]*$/, '')
|
| .trim();
|
| let body;
|
| try {
|
| body = markdownModule.mdToHtml(cleaned);
|
| } catch { body = _esc(cleaned); }
|
|
|
|
|
| const msgModel = (m.metadata && (m.metadata.model || m.metadata.model_name)) || '';
|
| const modelTag = !isUser && (msgModel || sessionModel)
|
| ? `<span class="doclib-chat-msg-model">${_esc(msgModel || sessionModel)}</span>`
|
| : '';
|
| return `<div class="doclib-chat-bubble-row ${isUser ? 'user' : 'assistant'}">
|
| <div class="doclib-chat-bubble">
|
| ${modelTag}
|
| <div class="doclib-chat-bubble-body">${body}</div>
|
| </div>
|
| </div>`;
|
| }).join('')
|
| : '<div style="opacity:0.4;font-size:11px;padding:6px 4px;">No messages yet</div>';
|
| const isArchive = !!session.archived;
|
|
|
|
|
| const archiveHtml = isArchive
|
| ? '<button class="doclib-chat-restore-btn">' +
|
| '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="9 14 4 9 9 4"/><path d="M4 9h11a5 5 0 0 1 5 5v0a5 5 0 0 1-5 5H9"/></svg>' +
|
| 'Restore' +
|
| '</button>'
|
| : '<button class="doclib-chat-archive-btn">' +
|
| '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="21 8 21 21 3 21 3 8"/><rect x="1" y="3" width="22" height="5"/><line x1="10" y1="12" x2="14" y2="12"/></svg>' +
|
| 'Archive' +
|
| '</button>';
|
|
|
|
|
|
|
|
|
|
|
| const copyHtml = isArchive ? '' : '<button class="doclib-chat-copy-btn">' +
|
| '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><rect x="9" y="9" width="13" height="13" rx="2"/><path d="M5 15H4a2 2 0 0 1-2-2V4a2 2 0 0 1 2-2h9a2 2 0 0 1 2 2v1"/></svg>' +
|
| 'Copy' +
|
| '</button>';
|
| const deleteHtml = '<button class="doclib-chat-delete-btn">' +
|
| '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="3 6 5 6 21 6"/><path d="M19 6l-1 14a2 2 0 0 1-2 2H8a2 2 0 0 1-2-2L5 6"/><path d="M10 11v6"/><path d="M14 11v6"/></svg>' +
|
| 'Delete' +
|
| '</button>';
|
| preview.innerHTML =
|
| '<div class="doclib-chat-preview-messages">' + msgsHtml + '</div>' +
|
| '<div class="doclib-chat-preview-actions">' +
|
| deleteHtml +
|
| archiveHtml +
|
| copyHtml +
|
| '<button class="doclib-chat-open-btn">' +
|
| '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M5 12h14M13 5l7 7-7 7"/></svg>' +
|
| 'Open' +
|
| '</button>' +
|
| '</div>';
|
| const openBtn = preview.querySelector('.doclib-chat-open-btn');
|
| if (openBtn) openBtn.addEventListener('click', (e) => {
|
| e.stopPropagation();
|
| if (window.sessionModule) window.sessionModule.selectSession(session.id);
|
| closeLibrary();
|
|
|
|
|
|
|
|
|
| if (window.innerWidth <= 768) {
|
| const sb = document.getElementById('sidebar');
|
| if (sb) {
|
| sb.classList.add('hidden');
|
| try { window.syncRailSide && window.syncRailSide(); } catch (_) {}
|
| }
|
| }
|
| });
|
| const archiveBtn = preview.querySelector('.doclib-chat-archive-btn');
|
| if (archiveBtn) archiveBtn.addEventListener('click', async (e) => {
|
| e.stopPropagation();
|
| await fetch(API_BASE + '/api/session/' + session.id + '/archive', {
|
| method: 'POST',
|
| headers: { 'Content-Type': 'application/json' },
|
| });
|
| _renderLibChats();
|
| });
|
| const restoreBtn = preview.querySelector('.doclib-chat-restore-btn');
|
| if (restoreBtn) restoreBtn.addEventListener('click', async (e) => {
|
| e.stopPropagation();
|
| await fetch(API_BASE + '/api/session/' + session.id + '/unarchive', { method: 'POST' });
|
| _renderLibArchive();
|
| });
|
| const copyBtn = preview.querySelector('.doclib-chat-copy-btn');
|
| if (copyBtn) copyBtn.addEventListener('click', (e) => {
|
| e.stopPropagation();
|
| _copyChatById(session.id);
|
| });
|
| const deleteBtn = preview.querySelector('.doclib-chat-delete-btn');
|
| if (deleteBtn) deleteBtn.addEventListener('click', async (e) => {
|
| e.stopPropagation();
|
| if (!await window.styledConfirm('Delete this chat?', { confirmText: 'Delete', danger: true })) return;
|
| await fetch(API_BASE + '/api/session/' + session.id, { method: 'DELETE' });
|
| card.style.maxHeight = `${Math.max(card.getBoundingClientRect().height, card.scrollHeight)}px`;
|
| card.classList.add('memory-tidy-removing');
|
| await new Promise(r => setTimeout(r, 520));
|
| if (isArchive) _renderLibArchive(); else _renderLibChats();
|
| });
|
| } catch (e) {
|
| preview.innerHTML = '<div style="opacity:0.5;font-size:11px;padding:6px 4px;color:var(--color-error);">Failed to load preview</div>';
|
| }
|
| }
|
|
|
| function _renderChatsGrid() {
|
| const grid = document.getElementById('doclib-chats-grid');
|
| if (!grid) return;
|
| const _csb = document.getElementById('doclib-chats-select-btn');
|
| if (_csb) { _csb.classList.toggle('active', _chatsSelectMode); _csb.textContent = _chatsSelectMode ? 'Cancel' : 'Select'; }
|
| let filtered = _chatsSessions.slice();
|
| if (_chatsSearch) {
|
| const q = _chatsSearch.toLowerCase();
|
| filtered = filtered.filter(s => (s.name || '').toLowerCase().includes(q) || (s.model || '').toLowerCase().includes(q));
|
| }
|
| if (_chatsModelFilter) filtered = filtered.filter(s => s.folder === _chatsModelFilter);
|
| if (_chatsSort === 'oldest') filtered.sort((a, b) => (a.updated_at || '') > (b.updated_at || '') ? 1 : -1);
|
| else if (_chatsSort === 'most-messages') filtered.sort((a, b) => (b.message_count || 0) - (a.message_count || 0));
|
| else if (_chatsSort === 'alpha') filtered.sort((a, b) => (a.name || '').localeCompare(b.name || ''));
|
| else filtered.sort((a, b) => (b.updated_at || '') > (a.updated_at || '') ? 1 : -1);
|
|
|
| const stats = document.getElementById('doclib-chats-stats');
|
| if (stats) stats.textContent = filtered.length + ' chat' + (filtered.length !== 1 ? 's' : '');
|
|
|
| if (!filtered.length) {
|
|
|
| const _sadIco = '<span style="vertical-align:-3px;margin-left:6px;">' + uiModule.emptyStateIcon('sad') + '</span>';
|
| grid.innerHTML = '<div class="doclib-empty">No chats' + _sadIco + '</div>';
|
| _appendInlineLoadMore(grid, 0, _chatsVisibleLimit, () => {});
|
| return;
|
| }
|
| const total = filtered.length;
|
| const visible = filtered.slice(0, _chatsVisibleLimit);
|
| grid.innerHTML = '';
|
| _maybeCascadeGrid(grid, 'chats');
|
| for (const s of visible) {
|
| const card = document.createElement('div');
|
| card.className = 'memory-item doclib-chat-row';
|
| card.style.cursor = 'pointer';
|
| card.dataset.sid = s.id;
|
| const model = (s.model || '').split('/').pop();
|
| const cbHtml = _chatsSelectMode ? '<input type="checkbox" class="memory-select-cb"' + (_chatsSelected.has(s.id) ? ' checked' : '') + '>' : '';
|
| const chatIconSvg = '<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-2px;margin-right:4px;opacity:0.4;flex-shrink:0;"><path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"/></svg>';
|
| const chevronSvg = '<span class="doclib-card-chevron"><svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="6 9 12 15 18 9"/></svg></span>';
|
|
|
|
|
|
|
| const _chatMsgs = s.message_count || 0;
|
| const msgCountHtml = _chatMsgs > 0
|
| ? '<span style="opacity:0.45;font-weight:normal;font-size:0.9em;margin-left:6px;">\u00b7 ' + _chatMsgs + ' msg' + (_chatMsgs === 1 ? '' : 's') + '</span>'
|
| : '';
|
| card.innerHTML =
|
| '<div class="doclib-chat-header" style="display:flex;align-items:center;width:100%;gap:6px;">' +
|
| cbHtml +
|
| '<div style="flex:1;min-width:0;">' +
|
| '<div class="memory-item-title">' + chatIconSvg + _esc(s.name || 'Untitled') + msgCountHtml + '</div>' +
|
| '<div class="memory-item-meta" style="font-size:10px;opacity:0.4;margin-top:2px;">' + [model, _relTime(s.updated_at)].filter(Boolean).join(' \u00b7 ') + '</div>' +
|
| '</div>' +
|
| chevronSvg +
|
| '<div class="memory-item-actions"><button class="memory-item-btn _chat-menu" title="Actions"><svg width="14" height="14" viewBox="0 0 24 24" fill="currentColor"><circle cx="12" cy="5" r="2"/><circle cx="12" cy="12" r="2"/><circle cx="12" cy="19" r="2"/></svg></button></div>' +
|
| '</div>' +
|
| '<div class="doclib-chat-preview" style="display:none;"></div>';
|
| const cb = card.querySelector('.memory-select-cb');
|
| if (cb) { cb.addEventListener('click', e => e.stopPropagation()); cb.addEventListener('change', () => { if (cb.checked) _chatsSelected.add(s.id); else _chatsSelected.delete(s.id); _updateChatsCount(); }); }
|
| card.querySelector('._chat-menu').addEventListener('click', (e) => { e.stopPropagation(); _showLibDropdown(e.currentTarget, [
|
| { label: 'Open', action: () => { if (window.sessionModule) window.sessionModule.selectSession(s.id); } },
|
| { label: 'Copy', action: () => _copyChatById(s.id) },
|
| { label: 'Archive', action: async () => { await fetch(API_BASE + '/api/session/' + s.id + '/archive', { method: 'POST', headers: {'Content-Type':'application/json'} }); _renderLibChats(); } },
|
| { label: 'Delete', action: async () => {
|
| if (!await window.styledConfirm('Delete this chat?', { confirmText: 'Delete', danger: true })) return;
|
| await fetch(API_BASE + '/api/session/' + s.id, { method: 'DELETE' });
|
| card.style.maxHeight = `${Math.max(card.getBoundingClientRect().height, card.scrollHeight)}px`;
|
| card.classList.add('memory-tidy-removing');
|
| await new Promise(r => setTimeout(r, 520));
|
| _renderLibChats();
|
| }, danger: true },
|
| ], { onSelect: () => {
|
| _chatsSelectMode = true;
|
| _chatsSelected.add(s.id);
|
| document.getElementById('doclib-chats-bulk')?.classList.remove('hidden');
|
| _renderChatsGrid();
|
| } }); });
|
| card.addEventListener('click', (e) => {
|
| if (card._suppressNextClick) { card._suppressNextClick = false; return; }
|
| if (_chatsSelectMode) { const c = card.querySelector('.memory-select-cb'); if (c) { c.checked = !c.checked; if (c.checked) _chatsSelected.add(s.id); else _chatsSelected.delete(s.id); _updateChatsCount(); } return; }
|
| if (e.target.closest('._chat-menu') || e.target.closest('.memory-select-cb') || e.target.closest('.doclib-chat-open-btn')) return;
|
| _toggleChatPreview(card, s);
|
| });
|
| _attachLongPressMenu(card, '._chat-menu');
|
| grid.appendChild(card);
|
| }
|
| _appendInlineLoadMore(grid, total, _chatsVisibleLimit, () => {
|
| _chatsVisibleLimit += _LIB_PAGE_SIZE;
|
| _renderChatsGrid();
|
| });
|
| }
|
|
|
| function _renderChatsChips() {
|
| const el = document.getElementById('doclib-chats-chips');
|
| if (!el) return;
|
| const counts = {};
|
| _chatsSessions.forEach(s => { const f = s.folder; if (f) counts[f] = (counts[f] || 0) + 1; });
|
| const folders = Object.keys(counts).sort();
|
| if (folders.length < 1) { el.innerHTML = ''; return; }
|
| el.innerHTML = '';
|
| const mk = (label, val, count) => { const c = document.createElement('button'); c.className = 'memory-cat-chip' + (_chatsModelFilter === val ? ' active' : ''); c.textContent = label + ' (' + count + ')'; c.addEventListener('click', () => { _chatsModelFilter = _chatsModelFilter === val ? '' : val; _renderChatsGrid(); _renderChatsChips(); }); el.appendChild(c); };
|
| mk('all', '', _chatsSessions.length);
|
| folders.forEach(f => mk(f, f, counts[f]));
|
| }
|
|
|
| function _updateChatsCount() { const el = document.getElementById('doclib-chats-selected-count'); if (el) el.textContent = _chatsSelected.size + ' Selected'; }
|
|
|
|
|
| document.getElementById('doclib-chats-sort').addEventListener('change', (e) => { _chatsSort = e.target.value; _renderChatsGrid(); });
|
| document.getElementById('doclib-chats-search').addEventListener('input', (e) => { _chatsSearch = e.target.value.trim(); _renderChatsGrid(); });
|
| document.getElementById('doclib-chats-select-btn').addEventListener('click', () => { _chatsSelectMode = !_chatsSelectMode; _chatsSelected.clear(); document.getElementById('doclib-chats-bulk').classList.toggle('hidden', !_chatsSelectMode); _renderChatsGrid(); });
|
| document.getElementById('doclib-chats-bulk-cancel')?.addEventListener('click', () => {
|
| _chatsSelectMode = false; _chatsSelected.clear();
|
| document.getElementById('doclib-chats-bulk').classList.add('hidden');
|
| _renderChatsGrid();
|
| });
|
| function _chatsToggleAll() {
|
| const allCb = document.getElementById('doclib-chats-select-all');
|
| const newState = _chatsSelected.size < _chatsSessions.length;
|
| if (allCb) allCb.checked = newState;
|
| document.querySelectorAll('#doclib-chats-grid .memory-select-cb').forEach(cb => { cb.checked = newState; });
|
| _chatsSessions.forEach(s => { if (newState) _chatsSelected.add(s.id); else _chatsSelected.delete(s.id); });
|
| _updateChatsCount();
|
| }
|
| document.getElementById('doclib-chats-select-all').addEventListener('change', _chatsToggleAll);
|
| document.getElementById('doclib-chats-bulk').addEventListener('click', (e) => {
|
| if (e.target.closest('button') || e.target.closest('input')) return;
|
| _chatsToggleAll();
|
| });
|
| document.getElementById('doclib-chats-bulk-archive').addEventListener('click', async () => {
|
| const count = _chatsSelected.size;
|
| if (!count) return;
|
| const grid = document.getElementById('doclib-chats-grid');
|
| if (grid) {
|
| grid.querySelectorAll('.doclib-card').forEach(card => {
|
| const sid = card.dataset.sid || card.dataset.sessionId;
|
| if (sid && _chatsSelected.has(sid)) {
|
| card.style.transition = 'opacity 0.25s, transform 0.25s';
|
| card.style.opacity = '0';
|
| card.style.transform = 'scale(0.95)';
|
| }
|
| });
|
| }
|
| await new Promise(r => setTimeout(r, 250));
|
| const ids = [..._chatsSelected];
|
| const results = await Promise.all(
|
| ids.map(sid => fetch(API_BASE + '/api/session/' + sid + '/archive', { method: 'POST', headers: {'Content-Type':'application/json'} })
|
| .then(r => ({ sid, ok: r.ok }))
|
| .catch(() => ({ sid, ok: false }))
|
| )
|
| );
|
| const failed = results.filter(r => !r.ok).map(r => r.sid);
|
| if (failed.length && grid) {
|
| grid.querySelectorAll('.doclib-card').forEach(card => {
|
| const sid = card.dataset.sid || card.dataset.sessionId;
|
| if (sid && failed.includes(sid)) {
|
| card.style.opacity = '';
|
| card.style.transform = '';
|
| }
|
| });
|
| if (window.uiModule) window.uiModule.showError(`Failed to archive ${failed.length} of ${ids.length} chat${ids.length > 1 ? 's' : ''}`);
|
| }
|
| _chatsSelected.clear();
|
| _chatsSelectMode = false;
|
| document.getElementById('doclib-chats-bulk').classList.add('hidden');
|
| _renderLibChats();
|
| });
|
| document.getElementById('doclib-chats-bulk-delete').addEventListener('click', async () => {
|
| const count = _chatsSelected.size;
|
| if (!count) return;
|
| if (!await window.styledConfirm(`Delete ${count} chat${count > 1 ? 's' : ''}? This cannot be undone.`, { confirmText: 'Delete', danger: true })) return;
|
|
|
| const grid = document.getElementById('doclib-chats-grid');
|
| if (grid) {
|
| grid.querySelectorAll('.doclib-card').forEach(card => {
|
| const sid = card.dataset.sid || card.dataset.sessionId;
|
| if (sid && _chatsSelected.has(sid)) {
|
| card.style.transition = 'opacity 0.25s, transform 0.25s';
|
| card.style.opacity = '0';
|
| card.style.transform = 'scale(0.95)';
|
| }
|
| });
|
| }
|
|
|
|
|
|
|
|
|
| await new Promise(r => setTimeout(r, 250));
|
| const ids = [..._chatsSelected];
|
| const results = await Promise.all(
|
| ids.map(sid => fetch(API_BASE + '/api/session/' + sid, { method: 'DELETE' })
|
| .then(r => ({ sid, ok: r.ok }))
|
| .catch(() => ({ sid, ok: false }))
|
| )
|
| );
|
| const failed = results.filter(r => !r.ok).map(r => r.sid);
|
| if (failed.length && grid) {
|
|
|
| grid.querySelectorAll('.doclib-card').forEach(card => {
|
| const sid = card.dataset.sid || card.dataset.sessionId;
|
| if (sid && failed.includes(sid)) {
|
| card.style.opacity = '';
|
| card.style.transform = '';
|
| }
|
| });
|
| if (window.uiModule) window.uiModule.showError(`Failed to delete ${failed.length} of ${ids.length} chat${ids.length > 1 ? 's' : ''}`);
|
| }
|
| _chatsSelected.clear();
|
| _chatsSelectMode = false;
|
| document.getElementById('doclib-chats-bulk').classList.add('hidden');
|
| _renderLibChats();
|
| });
|
|
|
|
|
| document.getElementById('doclib-chats-tidy-btn').addEventListener('click', async () => {
|
| const tidyBtn = document.getElementById('doclib-chats-tidy-btn');
|
| const origHTML = tidyBtn.innerHTML;
|
| tidyBtn.disabled = true;
|
| tidyBtn.classList.add('spinning');
|
| tidyBtn.textContent = '';
|
|
|
|
|
|
|
|
|
| const sp = spinnerModule.create('', 'clean', 'whirlpool');
|
| const el = sp.createElement();
|
| el.style.position = 'relative';
|
| el.style.top = '1px';
|
| tidyBtn.appendChild(el);
|
| sp.start();
|
| try {
|
| const res = await fetch(API_BASE + '/api/sessions/auto-sort', { method: 'POST', credentials: 'same-origin' });
|
| const data = await res.json();
|
| if (!res.ok) throw new Error(data.detail || 'Tidy failed');
|
| if (data.status === 'ok') {
|
| if (window.uiModule) window.uiModule.showToast('Sorted ' + data.updated + ' sessions into ' + data.folders.length + ' folders');
|
| if (window.sessionModule) await window.sessionModule.loadSessions();
|
| _renderLibChats();
|
| } else {
|
| if (window.uiModule) window.uiModule.showToast(data.reason || 'Nothing to tidy');
|
| }
|
| } catch (e) {
|
| if (window.uiModule) window.uiModule.showError('Tidy: ' + e.message);
|
| } finally {
|
| tidyBtn.disabled = false;
|
| tidyBtn.classList.remove('spinning');
|
| tidyBtn.innerHTML = origHTML;
|
| }
|
| });
|
|
|
|
|
| let _arcSessions = [];
|
| let _arcDocs = [];
|
| let _arcResearch = [];
|
| let _arcSearch = '';
|
| let _arcSort = 'recent';
|
| let _arcSelectMode = false;
|
| const _arcSelected = new Set();
|
| let _arcModelFilter = '';
|
| let _arcTypeFilter = '';
|
|
|
| function _renderLibArchive() {
|
| const grid = document.getElementById('doclib-arc-grid');
|
| if (!grid) return;
|
| grid.innerHTML = '';
|
| grid.appendChild(spinnerModule.createLoadingRow('Loading…'));
|
|
|
|
|
| Promise.all([
|
| fetch(API_BASE + '/api/sessions/archived?limit=100&sort=recent', { credentials: 'same-origin' }).then(r => r.json()).catch(() => ({})),
|
| fetch(API_BASE + '/api/documents/library?archived=true&limit=50', { credentials: 'same-origin' }).then(r => r.json()).catch(() => ({})),
|
| fetch('/api/research/library?archived=true', { credentials: 'same-origin' }).then(r => r.json()).catch(() => ({})),
|
| ]).then(([s, d, r]) => {
|
|
|
|
|
| _arcSessions = (s.sessions || []).map(x => ({ ...x, archived: true }));
|
| _arcDocs = d.documents || [];
|
| _arcResearch = (r.research || []).map(x => ({ ...x, archived: true }));
|
| _renderArcGrid();
|
| _renderArcChips();
|
| }).catch(() => { grid.innerHTML = '<div class="doclib-empty">Failed to load</div>'; });
|
| }
|
|
|
|
|
|
|
|
|
| async function _toggleArcDocPreview(card, d) {
|
| const preview = card.querySelector('.doclib-chat-preview');
|
| if (!preview) return;
|
| const grid = card.closest('.doclib-grid');
|
| if (grid) {
|
| grid.querySelectorAll('.doclib-card-expanded').forEach(c => {
|
| if (c !== card) {
|
| c.classList.remove('doclib-card-expanded');
|
| const p = c.querySelector('.doclib-chat-preview');
|
| if (p) { p.style.display = 'none'; p.innerHTML = ''; }
|
| }
|
| });
|
| }
|
| if (card.classList.contains('doclib-card-expanded')) {
|
| card.classList.remove('doclib-card-expanded');
|
| preview.style.display = 'none'; preview.innerHTML = '';
|
| return;
|
| }
|
| card.classList.add('doclib-card-expanded');
|
| preview.style.display = 'block';
|
| preview.innerHTML = '<div style="opacity:0.4;font-size:11px;padding:8px 4px;">Loading…</div>';
|
| try {
|
| const res = await fetch(`${API_BASE}/api/document/${d.id}`, { credentials: 'same-origin' });
|
| if (!res.ok) throw new Error('failed');
|
| const full = await res.json();
|
| const content = (full.current_content || '').slice(0, 20000);
|
| const pre = document.createElement('pre');
|
| pre.style.cssText = 'white-space:pre-wrap;word-break:break-word;font-size:11px;margin:6px 4px;max-height:50vh;overflow:auto;';
|
| pre.textContent = content || '(empty document)';
|
| preview.innerHTML = '';
|
| preview.appendChild(pre);
|
|
|
|
|
|
|
|
|
|
|
| const actions = document.createElement('div');
|
| actions.className = 'doclib-chat-preview-actions';
|
| actions.innerHTML =
|
| '<button class="doclib-chat-delete-btn"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="3 6 5 6 21 6"/><path d="M19 6l-1 14a2 2 0 0 1-2 2H8a2 2 0 0 1-2-2L5 6"/><path d="M10 11v6"/><path d="M14 11v6"/></svg>Delete</button>' +
|
| '<button class="doclib-chat-restore-btn"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="9 14 4 9 9 4"/><path d="M4 9h11a5 5 0 0 1 5 5v0a5 5 0 0 1-5 5H9"/></svg>Restore</button>' +
|
| '<button class="doclib-chat-open-btn"><svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M5 12h14M13 5l7 7-7 7"/></svg>Open</button>';
|
| actions.querySelector('.doclib-chat-delete-btn').addEventListener('click', async (ev) => {
|
| ev.stopPropagation();
|
| if (!await window.styledConfirm('Delete this document?', { confirmText: 'Delete', danger: true })) return;
|
| await fetch(`${API_BASE}/api/document/${d.id}`, { method: 'DELETE', credentials: 'same-origin' });
|
| _renderLibArchive();
|
| });
|
| actions.querySelector('.doclib-chat-restore-btn').addEventListener('click', async (ev) => {
|
| ev.stopPropagation();
|
| await fetch(`${API_BASE}/api/document/${d.id}/archive?archived=false`, { method: 'POST', credentials: 'same-origin' });
|
| _renderLibArchive();
|
| });
|
|
|
| actions.querySelector('.doclib-chat-open-btn').addEventListener('click', (ev) => {
|
| ev.stopPropagation();
|
| libraryImportDocument(d);
|
| });
|
| preview.appendChild(actions);
|
| } catch {
|
| preview.innerHTML = '<div style="opacity:0.4;font-size:11px;padding:8px 4px;">Failed to load preview</div>';
|
| }
|
| }
|
|
|
| function _renderArcGrid() {
|
| const grid = document.getElementById('doclib-arc-grid');
|
| if (!grid) return;
|
| const _asb = document.getElementById('doclib-arc-select-btn');
|
| if (_asb) { _asb.classList.toggle('active', _arcSelectMode); _asb.textContent = _arcSelectMode ? 'Cancel' : 'Select'; }
|
| let filtered = _arcSessions.slice();
|
| if (_arcSearch) {
|
| const q = _arcSearch.toLowerCase();
|
| filtered = filtered.filter(s => (s.name || '').toLowerCase().includes(q) || (s.model || '').toLowerCase().includes(q));
|
| }
|
| if (_arcModelFilter) filtered = filtered.filter(s => (s.model || '').split('/').pop() === _arcModelFilter);
|
| if (_arcSort === 'oldest') filtered.sort((a, b) => (a.updated_at || '') > (b.updated_at || '') ? 1 : -1);
|
| else if (_arcSort === 'most-messages') filtered.sort((a, b) => (b.message_count || 0) - (a.message_count || 0));
|
| else if (_arcSort === 'alpha') filtered.sort((a, b) => (a.name || '').localeCompare(b.name || ''));
|
| else filtered.sort((a, b) => (b.updated_at || '') > (a.updated_at || '') ? 1 : -1);
|
|
|
|
|
| const _aq = (_arcSearch || '').toLowerCase();
|
| let filtDocs = _aq ? _arcDocs.filter(d => (d.title || '').toLowerCase().includes(_aq)) : _arcDocs;
|
| let filtResearch = _aq ? _arcResearch.filter(r => (r.query || '').toLowerCase().includes(_aq)) : _arcResearch;
|
|
|
|
|
| const _showChats = !_arcTypeFilter || _arcTypeFilter === 'chats';
|
| const _showDocs = !_arcTypeFilter || _arcTypeFilter === 'documents';
|
| const _showResearch = !_arcTypeFilter || _arcTypeFilter === 'research';
|
| if (!_showChats) filtered = [];
|
| if (!_showDocs) filtDocs = [];
|
| if (!_showResearch) filtResearch = [];
|
|
|
| const stats = document.getElementById('doclib-arc-stats');
|
| if (stats) stats.textContent = (filtered.length + filtDocs.length + filtResearch.length) + ' archived';
|
|
|
| if (!filtered.length && !filtDocs.length && !filtResearch.length) {
|
|
|
| const _neutralIco = '<span style="vertical-align:-3px;margin-left:6px;">' + uiModule.emptyStateIcon('neutral') + '</span>';
|
| grid.innerHTML = '<div class="doclib-empty">No archived items' + _neutralIco + '</div>';
|
| _appendInlineLoadMore(grid, 0, _arcVisibleLimit, () => {});
|
| return;
|
| }
|
| const total = filtered.length;
|
| const visible = filtered.slice(0, _arcVisibleLimit);
|
| grid.innerHTML = '';
|
| _maybeCascadeGrid(grid, 'archive');
|
| for (const s of visible) {
|
| const card = document.createElement('div');
|
| card.className = 'memory-item doclib-chat-row';
|
| card.style.cursor = 'pointer';
|
| card.dataset.sid = s.id;
|
| card.dataset.arckey = 'chats:' + s.id;
|
| const model = (s.model || '').split('/').pop();
|
| const cbHtml = _arcSelectMode ? '<input type="checkbox" class="memory-select-cb" data-arckey="chats:' + s.id + '"' + (_arcSelected.has('chats:' + s.id) ? ' checked' : '') + '>' : '';
|
| const arcIconSvg = '<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-2px;margin-right:4px;opacity:0.5;flex-shrink:0;"><path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"/></svg>';
|
| card.innerHTML =
|
| '<div class="doclib-chat-header" style="display:flex;align-items:center;width:100%;gap:6px;">' +
|
| cbHtml +
|
| '<div style="flex:1;min-width:0;">' +
|
| '<div class="memory-item-title">' + arcIconSvg + _esc(s.name || 'Untitled') + '</div>' +
|
| '<div class="memory-item-meta" style="font-size:10px;opacity:0.4;margin-top:2px;">' + [model, _relTime(s.updated_at)].filter(Boolean).join(' \u00b7 ') + '</div>' +
|
| '</div>' +
|
| '<div class="memory-item-actions"><button class="memory-item-btn _arc-menu" title="Actions"><svg width="14" height="14" viewBox="0 0 24 24" fill="currentColor"><circle cx="12" cy="5" r="2"/><circle cx="12" cy="12" r="2"/><circle cx="12" cy="19" r="2"/></svg></button></div>' +
|
| '</div>' +
|
| '<div class="doclib-chat-preview" style="display:none;"></div>';
|
| const cb = card.querySelector('.memory-select-cb');
|
| if (cb) { cb.addEventListener('click', e => e.stopPropagation()); cb.addEventListener('change', () => { if (cb.checked) _arcSelected.add('chats:' + s.id); else _arcSelected.delete('chats:' + s.id); _updateArcCount(); }); }
|
| card.querySelector('._arc-menu').addEventListener('click', (e) => { e.stopPropagation(); _showLibDropdown(e.currentTarget, [
|
| { label: 'Open', action: () => { if (window.sessionModule) window.sessionModule.selectSession(s.id); } },
|
| { label: 'Copy', action: () => _copyChatById(s.id) },
|
| { label: 'Restore', action: async () => { await fetch(API_BASE + '/api/session/' + s.id + '/unarchive', { method: 'POST' }); _renderLibArchive(); } },
|
| { label: 'Delete', action: async () => {
|
| if (!await window.styledConfirm('Delete this chat permanently?', { confirmText: 'Delete', danger: true })) return;
|
| await fetch(API_BASE + '/api/session/' + s.id, { method: 'DELETE' });
|
| _renderLibArchive();
|
| }, danger: true },
|
| ], { onSelect: () => {
|
| _arcSelectMode = true;
|
| _arcSelected.add('chats:' + s.id);
|
| document.getElementById('doclib-arc-bulk')?.classList.remove('hidden');
|
| _renderArcGrid();
|
| } }); });
|
| card.addEventListener('click', (e) => {
|
| if (card._suppressNextClick) { card._suppressNextClick = false; return; }
|
| if (_arcSelectMode) { const c = card.querySelector('.memory-select-cb'); if (c) { c.checked = !c.checked; if (c.checked) _arcSelected.add('chats:' + s.id); else _arcSelected.delete('chats:' + s.id); _updateArcCount(); } return; }
|
| if (e.target.closest('._arc-menu') || e.target.closest('.memory-select-cb') || e.target.closest('.doclib-chat-open-btn')) return;
|
| _toggleChatPreview(card, s);
|
| });
|
| _attachLongPressMenu(card, '._arc-menu');
|
| grid.appendChild(card);
|
| }
|
|
|
| const _arcDocIco = '<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-2px;margin-right:4px;opacity:0.5;flex-shrink:0;"><path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/><polyline points="14 2 14 8 20 8"/><line x1="16" y1="13" x2="8" y2="13"/><line x1="16" y1="17" x2="8" y2="17"/><polyline points="10 9 9 9 8 9"/></svg>';
|
| for (const d of filtDocs) {
|
| const card = document.createElement('div');
|
| card.className = 'memory-item doclib-chat-row';
|
| card.style.cursor = 'pointer';
|
| card.dataset.arckey = 'documents:' + d.id;
|
| const _dcb = _arcSelectMode ? '<input type="checkbox" class="memory-select-cb" data-arckey="documents:' + d.id + '"' + (_arcSelected.has('documents:' + d.id) ? ' checked' : '') + '>' : '';
|
| card.innerHTML =
|
| '<div class="doclib-chat-header" style="display:flex;align-items:center;width:100%;gap:6px;">' +
|
| _dcb +
|
| '<div style="flex:1;min-width:0;">' +
|
| '<div class="memory-item-title">' + _arcDocIco + _esc(d.title || 'Untitled') + '</div>' +
|
| '<div class="memory-item-meta" style="font-size:10px;opacity:0.4;margin-top:2px;">' + ['Document', (d.language || 'text'), _relTime(d.updated_at)].filter(Boolean).join(' · ') + '</div>' +
|
| '</div>' +
|
| '<div class="memory-item-actions"><button class="memory-item-btn _arc-doc-menu" title="Actions"><svg width="14" height="14" viewBox="0 0 24 24" fill="currentColor"><circle cx="12" cy="5" r="2"/><circle cx="12" cy="12" r="2"/><circle cx="12" cy="19" r="2"/></svg></button></div>' +
|
| '</div>' +
|
| '<div class="doclib-chat-preview" style="display:none;"></div>';
|
| const _dcbEl = card.querySelector('.memory-select-cb');
|
| if (_dcbEl) { _dcbEl.addEventListener('click', e => e.stopPropagation()); _dcbEl.addEventListener('change', () => { if (_dcbEl.checked) _arcSelected.add('documents:' + d.id); else _arcSelected.delete('documents:' + d.id); _updateArcCount(); }); }
|
| card.addEventListener('click', (e) => {
|
| if (e.target.closest('._arc-doc-menu') || e.target.closest('.memory-select-cb')) return;
|
| if (_arcSelectMode) { const c = card.querySelector('.memory-select-cb'); if (c) { c.checked = !c.checked; if (c.checked) _arcSelected.add('documents:' + d.id); else _arcSelected.delete('documents:' + d.id); _updateArcCount(); } return; }
|
| _toggleArcDocPreview(card, d);
|
| });
|
| card.querySelector('._arc-doc-menu').addEventListener('click', (e) => { e.stopPropagation(); _showLibDropdown(e.currentTarget, [
|
| { label: 'Restore', action: async () => { await fetch(API_BASE + '/api/document/' + d.id + '/archive?archived=false', { method: 'POST', credentials: 'same-origin' }); _renderLibArchive(); } },
|
| { label: 'Delete', danger: true, action: async () => { if (!await window.styledConfirm('Delete this document?', { confirmText: 'Delete', danger: true })) return; await fetch(API_BASE + '/api/document/' + d.id, { method: 'DELETE', credentials: 'same-origin' }); _renderLibArchive(); } },
|
| ], { onSelect: () => {
|
| _arcSelectMode = true;
|
| _arcSelected.add('documents:' + d.id);
|
| document.getElementById('doclib-arc-bulk')?.classList.remove('hidden');
|
| _renderArcGrid();
|
| } }); });
|
| _attachLongPressMenu(card, '._arc-doc-menu');
|
| grid.appendChild(card);
|
| }
|
|
|
| const _arcResIco = '<svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-2px;margin-right:4px;opacity:0.5;flex-shrink:0;"><circle cx="11" cy="11" r="8"/><path d="M21 21l-4.35-4.35"/><line x1="11" y1="8" x2="11" y2="14"/><line x1="8" y1="11" x2="14" y2="11"/></svg>';
|
| for (const r of filtResearch) {
|
| const card = document.createElement('div');
|
| card.className = 'memory-item doclib-chat-row';
|
| card.style.cursor = 'pointer';
|
| card.dataset.arckey = 'research:' + r.id;
|
| const _rcb = _arcSelectMode ? '<input type="checkbox" class="memory-select-cb" data-arckey="research:' + r.id + '"' + (_arcSelected.has('research:' + r.id) ? ' checked' : '') + '>' : '';
|
| card.innerHTML =
|
| '<div class="doclib-chat-header" style="display:flex;align-items:center;width:100%;gap:6px;">' +
|
| _rcb +
|
| '<div style="flex:1;min-width:0;">' +
|
| '<div class="memory-item-title">' + _arcResIco + _esc(r.query || 'Research') + '</div>' +
|
| '<div class="memory-item-meta" style="font-size:10px;opacity:0.4;margin-top:2px;">' + ['Research', (r.source_count ? r.source_count + ' sources' : ''), _relTime(r.completed_at ? new Date(r.completed_at * 1000).toISOString() : '')].filter(Boolean).join(' · ') + '</div>' +
|
| '</div>' +
|
| '<div class="memory-item-actions"><button class="memory-item-btn _arc-res-menu" title="Actions"><svg width="14" height="14" viewBox="0 0 24 24" fill="currentColor"><circle cx="12" cy="5" r="2"/><circle cx="12" cy="12" r="2"/><circle cx="12" cy="19" r="2"/></svg></button></div>' +
|
| '</div>' +
|
| '<div class="doclib-chat-preview" style="display:none;"></div>';
|
| const _rcbEl = card.querySelector('.memory-select-cb');
|
| if (_rcbEl) { _rcbEl.addEventListener('click', e => e.stopPropagation()); _rcbEl.addEventListener('change', () => { if (_rcbEl.checked) _arcSelected.add('research:' + r.id); else _arcSelected.delete('research:' + r.id); _updateArcCount(); }); }
|
| card.addEventListener('click', (e) => {
|
| if (e.target.closest('._arc-res-menu') || e.target.closest('.memory-select-cb')) return;
|
| if (_arcSelectMode) { const c = card.querySelector('.memory-select-cb'); if (c) { c.checked = !c.checked; if (c.checked) _arcSelected.add('research:' + r.id); else _arcSelected.delete('research:' + r.id); _updateArcCount(); } return; }
|
| _toggleResearchPreview(card, r);
|
| });
|
| card.querySelector('._arc-res-menu').addEventListener('click', (e) => { e.stopPropagation(); _showLibDropdown(e.currentTarget, [
|
| { label: 'Open', action: () => { const a = document.createElement('a'); a.href = '/api/research/report/' + r.id; a.target = '_blank'; a.rel = 'noopener'; document.body.appendChild(a); a.click(); a.remove(); } },
|
| { label: 'Restore', action: async () => { await fetch('/api/research/' + r.id + '/archive?archived=false', { method: 'POST', credentials: 'same-origin' }); _renderLibArchive(); } },
|
| { label: 'Delete', danger: true, action: async () => { if (!await window.styledConfirm('Delete this research?', { confirmText: 'Delete', danger: true })) return; await fetch('/api/research/' + r.id, { method: 'DELETE', credentials: 'same-origin' }); _renderLibArchive(); } },
|
| ], { onSelect: () => {
|
| _arcSelectMode = true;
|
| _arcSelected.add('research:' + r.id);
|
| document.getElementById('doclib-arc-bulk')?.classList.remove('hidden');
|
| _renderArcGrid();
|
| } }); });
|
| _attachLongPressMenu(card, '._arc-res-menu');
|
| grid.appendChild(card);
|
| }
|
| _appendInlineLoadMore(grid, total, _arcVisibleLimit, () => {
|
| _arcVisibleLimit += _LIB_PAGE_SIZE;
|
| _renderArcGrid();
|
| });
|
| }
|
|
|
| function _renderArcChips() {
|
| const el = document.getElementById('doclib-arc-chips');
|
| if (!el) return;
|
|
|
| el.innerHTML = '';
|
| const mk = (label, val, count) => {
|
| const c = document.createElement('button');
|
| c.className = 'memory-cat-chip' + (_arcTypeFilter === val ? ' active' : '');
|
| c.textContent = label + ' (' + count + ')';
|
| c.addEventListener('click', () => { _arcTypeFilter = _arcTypeFilter === val ? '' : val; _renderArcGrid(); _renderArcChips(); });
|
| el.appendChild(c);
|
| };
|
| const total = _arcSessions.length + _arcDocs.length + _arcResearch.length;
|
| if (!total) return;
|
| mk('All', '', total);
|
| if (_arcSessions.length) mk('Chats', 'chats', _arcSessions.length);
|
| if (_arcDocs.length) mk('Documents', 'documents', _arcDocs.length);
|
| if (_arcResearch.length) mk('Research', 'research', _arcResearch.length);
|
| }
|
|
|
| function _updateArcCount() { const el = document.getElementById('doclib-arc-selected-count'); if (el) el.textContent = _arcSelected.size + ' Selected'; }
|
|
|
|
|
| document.getElementById('doclib-arc-sort').addEventListener('change', (e) => { _arcSort = e.target.value; _renderArcGrid(); });
|
| document.getElementById('doclib-arc-search').addEventListener('input', (e) => { _arcSearch = e.target.value.trim(); _renderArcGrid(); });
|
| document.getElementById('doclib-arc-select-btn').addEventListener('click', () => { _arcSelectMode = !_arcSelectMode; _arcSelected.clear(); document.getElementById('doclib-arc-bulk').classList.toggle('hidden', !_arcSelectMode); _renderArcGrid(); });
|
| document.getElementById('doclib-arc-bulk-cancel')?.addEventListener('click', () => {
|
| _arcSelectMode = false; _arcSelected.clear();
|
| document.getElementById('doclib-arc-bulk').classList.add('hidden');
|
| _renderArcGrid();
|
| });
|
|
|
|
|
| function _arcToggleAll() {
|
| const cbs = document.querySelectorAll('#doclib-arc-grid .memory-select-cb');
|
| const newState = _arcSelected.size < cbs.length;
|
| const allCb = document.getElementById('doclib-arc-select-all');
|
| if (allCb) allCb.checked = newState;
|
| cbs.forEach(cb => {
|
| cb.checked = newState;
|
| const k = cb.dataset.arckey;
|
| if (k) { if (newState) _arcSelected.add(k); else _arcSelected.delete(k); }
|
| });
|
| _updateArcCount();
|
| }
|
| document.getElementById('doclib-arc-select-all').addEventListener('change', _arcToggleAll);
|
| document.getElementById('doclib-arc-bulk').addEventListener('click', (e) => {
|
| if (e.target.closest('button') || e.target.closest('input')) return;
|
| _arcToggleAll();
|
| });
|
|
|
| function _arcRestoreOne(key) {
|
| const i = key.indexOf(':'), type = key.slice(0, i), id = key.slice(i + 1);
|
| if (type === 'documents') return fetch(API_BASE + '/api/document/' + id + '/archive?archived=false', { method: 'POST', credentials: 'same-origin' });
|
| if (type === 'research') return fetch('/api/research/' + id + '/archive?archived=false', { method: 'POST', credentials: 'same-origin' });
|
| return fetch(API_BASE + '/api/session/' + id + '/unarchive', { method: 'POST', credentials: 'same-origin' });
|
| }
|
| function _arcDeleteOne(key) {
|
| const i = key.indexOf(':'), type = key.slice(0, i), id = key.slice(i + 1);
|
| if (type === 'documents') return fetch(API_BASE + '/api/document/' + id, { method: 'DELETE', credentials: 'same-origin' });
|
| if (type === 'research') return fetch('/api/research/' + id, { method: 'DELETE', credentials: 'same-origin' });
|
| return fetch(API_BASE + '/api/session/' + id, { method: 'DELETE', credentials: 'same-origin' });
|
| }
|
| document.getElementById('doclib-arc-bulk-restore').addEventListener('click', async () => {
|
| if (!_arcSelected.size) return;
|
| await Promise.all([..._arcSelected].map(_arcRestoreOne));
|
| _arcSelected.clear(); _arcSelectMode = false;
|
| document.getElementById('doclib-arc-bulk').classList.add('hidden');
|
| _renderLibArchive();
|
| });
|
| document.getElementById('doclib-arc-bulk-delete').addEventListener('click', async () => {
|
| const count = _arcSelected.size;
|
| if (!count) return;
|
| if (!await window.styledConfirm(`Delete ${count} archived item${count > 1 ? 's' : ''} permanently?`, { confirmText: 'Delete', danger: true })) return;
|
| const grid = document.getElementById('doclib-arc-grid');
|
| if (grid) {
|
| grid.querySelectorAll('.memory-item[data-arckey]').forEach(card => {
|
| if (_arcSelected.has(card.dataset.arckey)) {
|
| card.style.transition = 'opacity 0.25s, transform 0.25s';
|
| card.style.opacity = '0';
|
| card.style.transform = 'scale(0.95)';
|
| }
|
| });
|
| }
|
| await new Promise(r => setTimeout(r, 250));
|
| await Promise.all([..._arcSelected].map(_arcDeleteOne));
|
| _arcSelected.clear();
|
| _arcSelectMode = false;
|
| document.getElementById('doclib-arc-bulk').classList.add('hidden');
|
| _renderLibArchive();
|
| });
|
|
|
|
|
| let _researchItems = [];
|
| let _researchSearch = '';
|
| let _researchSelectMode = false;
|
| let _researchArchivedView = false;
|
| const _researchSelected = new Set();
|
|
|
| async function _renderLibResearch() {
|
| const grid = document.getElementById('doclib-research-grid');
|
| const stats = document.getElementById('doclib-research-stats');
|
| if (!grid) return;
|
|
|
| grid.innerHTML = '';
|
| try {
|
| const _spm = (await import('./spinner.js')).default;
|
| const _sp = _spm.createWhirlpool(22);
|
| _sp.element.style.cssText = 'margin:18px auto;display:block;';
|
| grid.appendChild(_sp.element);
|
| } catch { grid.innerHTML = '<div class="hwfit-loading">Loading…</div>'; }
|
| try {
|
| const res = await fetch('/api/research/library' + (_researchArchivedView ? '?archived=true' : ''), { credentials: 'same-origin' });
|
| if (!res.ok) throw new Error(res.statusText);
|
| const data = await res.json();
|
| _researchItems = data.research || data || [];
|
| } catch (e) {
|
| grid.innerHTML = `<div class="hwfit-loading">Failed to load: ${_esc(e.message)}</div>`;
|
| return;
|
| }
|
| _renderResearchGrid();
|
| }
|
|
|
|
|
|
|
|
|
| async function _toggleResearchPreview(card, item) {
|
| const preview = card.querySelector('.doclib-chat-preview');
|
| if (!preview) return;
|
| const isOpen = card.classList.contains('doclib-card-expanded');
|
| const grid = card.closest('.doclib-grid');
|
| if (grid) {
|
| grid.querySelectorAll('.doclib-card-expanded').forEach(c => {
|
| if (c !== card) {
|
| c.classList.remove('doclib-card-expanded');
|
| const p = c.querySelector('.doclib-chat-preview');
|
| if (p) { p.style.display = 'none'; p.innerHTML = ''; }
|
| }
|
| });
|
| }
|
| if (isOpen) {
|
| card.classList.remove('doclib-card-expanded');
|
| preview.style.display = 'none';
|
| preview.innerHTML = '';
|
| return;
|
| }
|
| card.classList.add('doclib-card-expanded');
|
| preview.style.display = 'block';
|
| preview.innerHTML = '<div style="opacity:0.4;font-size:11px;padding:8px 4px;">Loading…</div>';
|
| let detail = item;
|
| try {
|
|
|
|
|
| const res = await fetch(`${API_BASE}/api/research/detail/${item.id}`, { credentials: 'same-origin' });
|
| if (res.ok) detail = await res.json();
|
| } catch {}
|
| const sources = Array.isArray(detail.sources) ? detail.sources : [];
|
| const sourcesList = sources.slice(0, 12).map((src, i) => {
|
| const title = _esc(src.title || src.url || `Source ${i + 1}`);
|
| const url = _safeResearchHref(src.url);
|
| return url
|
| ? `<li><a href="${url}" target="_blank" rel="noopener">${title}</a></li>`
|
| : `<li>${title}</li>`;
|
| }).join('');
|
| const sourcesHtml = sources.length
|
| ? `<div class="doclib-research-sources"><div class="doclib-research-section-label">Sources (${sources.length})</div><ol>${sourcesList}${sources.length > 12 ? `<li style="opacity:0.5;">…and ${sources.length - 12} more</li>` : ''}</ol></div>`
|
| : '';
|
|
|
|
|
| const summary = (detail.summary || detail.report_summary || detail.result || detail.raw_report || '').toString().trim();
|
| const summaryHtml = summary
|
| ? `<div class="doclib-research-summary"><div class="doclib-research-section-label">Report</div><div>${markdownModule.mdToHtml ? markdownModule.mdToHtml(summary) : _esc(summary)}</div></div>`
|
| : '';
|
| preview.innerHTML =
|
| '<div class="doclib-chat-preview-messages">' +
|
| (summaryHtml || sourcesHtml || '<div style="opacity:0.4;font-size:11px;padding:6px 4px;">No preview available</div>') +
|
| (summaryHtml && sourcesHtml ? sourcesHtml : '') +
|
| '</div>' +
|
| '<div class="doclib-chat-preview-actions">' +
|
| '<button class="doclib-chat-delete-btn">' +
|
| '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="3 6 5 6 21 6"/><path d="M19 6l-1 14a2 2 0 0 1-2 2H8a2 2 0 0 1-2-2L5 6"/><path d="M10 11v6"/><path d="M14 11v6"/></svg>' +
|
| 'Delete' +
|
| '</button>' +
|
| '<button class="doclib-chat-archive-btn">' +
|
| '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polyline points="21 8 21 21 3 21 3 8"/><rect x="1" y="3" width="22" height="5"/><line x1="10" y1="12" x2="14" y2="12"/></svg>' +
|
| ((_researchArchivedView || item.archived) ? 'Restore' : 'Archive') +
|
| '</button>' +
|
|
|
|
|
| (item.archived ? '' :
|
| '<button class="doclib-chat-discuss-btn">' +
|
| '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z"/></svg>' +
|
| 'Discuss' +
|
| '</button>') +
|
| '<button class="doclib-chat-open-btn">' +
|
| '<svg width="11" height="11" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><path d="M5 12h14M13 5l7 7-7 7"/></svg>' +
|
| 'Open' +
|
| '</button>' +
|
| '</div>';
|
| const discussBtn = preview.querySelector('.doclib-chat-discuss-btn');
|
| if (discussBtn) discussBtn.addEventListener('click', async (e) => {
|
| e.stopPropagation();
|
| const _orig = discussBtn.innerHTML;
|
| discussBtn.disabled = true;
|
| discussBtn.textContent = 'Creating…';
|
| try {
|
| const _sid = detail.session_id || detail.id || item.id;
|
| const res = await fetch(`${API_BASE}/api/research/spinoff/${_sid}`, { method: 'POST', credentials: 'same-origin' });
|
| if (!res.ok) { let d = ''; try { d = (await res.json()).detail || ''; } catch {} throw new Error(d || ('HTTP ' + res.status)); }
|
| const payload = await res.json();
|
| if (window.sessionModule && payload.session_id) {
|
| await window.sessionModule.loadSessions().catch(() => {});
|
| await window.sessionModule.selectSession(payload.session_id);
|
| }
|
| closeLibrary();
|
| } catch (err) {
|
| discussBtn.disabled = false;
|
| discussBtn.innerHTML = _orig;
|
| if (uiModule) uiModule.showError('Could not start discussion: ' + (err.message || err));
|
| }
|
| });
|
| const openBtn = preview.querySelector('.doclib-chat-open-btn');
|
| if (openBtn) openBtn.addEventListener('click', (e) => {
|
| e.stopPropagation();
|
| const a = document.createElement('a');
|
| a.href = '/api/research/report/' + item.id;
|
| a.target = '_blank';
|
| a.rel = 'noopener';
|
| document.body.appendChild(a);
|
| a.click();
|
| a.remove();
|
| });
|
| const delBtn = preview.querySelector('.doclib-chat-delete-btn');
|
| if (delBtn) delBtn.addEventListener('click', async (e) => {
|
| e.stopPropagation();
|
| const ok = uiModule && uiModule.styledConfirm
|
| ? await uiModule.styledConfirm('Delete this research report?', { confirmText: 'Delete', danger: true })
|
| : window.confirm('Delete this research report?');
|
| if (!ok) return;
|
| try {
|
| const res = await fetch(`${API_BASE}/api/research/${item.id}`, { method: 'DELETE', credentials: 'same-origin' });
|
| if (!res.ok) throw new Error(await res.text());
|
| if (item.archived) {
|
| _renderLibArchive();
|
| } else {
|
| _researchItems = _researchItems.filter(r => r.id !== item.id);
|
| _renderResearchGrid();
|
| }
|
| } catch (err) {
|
| if (uiModule && uiModule.showError) uiModule.showError('Failed to delete: ' + err.message);
|
| }
|
| });
|
| const arcBtn = preview.querySelector('.doclib-chat-archive-btn');
|
| if (arcBtn) arcBtn.addEventListener('click', async (e) => {
|
| e.stopPropagation();
|
|
|
|
|
| const fromArchiveTab = !!item.archived;
|
| const toArchived = fromArchiveTab ? false : !_researchArchivedView;
|
| try {
|
| await fetch(`${API_BASE}/api/research/${item.id}/archive?archived=${toArchived}`, { method: 'POST', credentials: 'same-origin' });
|
| if (fromArchiveTab) {
|
| _renderLibArchive();
|
| } else {
|
| _researchItems = _researchItems.filter(r => r.id !== item.id);
|
| _renderResearchGrid();
|
| }
|
| if (uiModule) uiModule.showToast(toArchived ? 'Archived' : 'Restored');
|
| } catch { if (uiModule) uiModule.showError('Failed to ' + (toArchived ? 'archive' : 'restore')); }
|
| });
|
| }
|
|
|
| function _renderResearchGrid() {
|
| const grid = document.getElementById('doclib-research-grid');
|
| const stats = document.getElementById('doclib-research-stats');
|
| if (!grid) return;
|
| const _rsb = document.getElementById('doclib-research-select-btn');
|
| if (_rsb) { _rsb.classList.toggle('active', _researchSelectMode); _rsb.textContent = _researchSelectMode ? 'Cancel' : 'Select'; }
|
| let items = _researchItems;
|
| if (_researchSearch) {
|
| const s = _researchSearch.toLowerCase();
|
| items = items.filter(r => (r.query || '').toLowerCase().includes(s));
|
| }
|
|
|
| const _rSort = document.getElementById('doclib-research-sort')?.value || 'recent';
|
| if (_rSort === 'recent') items.sort((a, b) => (b.completed_at || 0) - (a.completed_at || 0));
|
| else if (_rSort === 'oldest') items.sort((a, b) => (a.completed_at || 0) - (b.completed_at || 0));
|
| else if (_rSort === 'most-sources') items.sort((a, b) => (b.source_count || 0) - (a.source_count || 0));
|
| else if (_rSort === 'alpha') items.sort((a, b) => (a.query || '').localeCompare(b.query || ''));
|
| if (stats) stats.textContent = items.length + ' research' + (items.length !== 1 ? 'es' : '');
|
| if (!items.length) {
|
| grid.innerHTML =
|
| '<div class="hwfit-loading" style="display:flex;align-items:center;justify-content:center;gap:8px;flex-wrap:wrap;">' +
|
| '<span>No research yet</span>' +
|
| '<span style="opacity:0.7;font-size:11px;">' +
|
| 'create one in the <a href="#" data-doclib-open-research style="color:var(--accent,var(--red));text-decoration:underline;">Deep Research</a> tab' +
|
| '</span>' +
|
| '</div>';
|
| grid.querySelector('[data-doclib-open-research]')?.addEventListener('click', (e) => {
|
| e.preventDefault();
|
| document.getElementById('rail-research')?.click();
|
| });
|
| _appendInlineLoadMore(grid, 0, _researchVisibleLimit, () => {});
|
| return;
|
| }
|
| const total = items.length;
|
| items = items.slice(0, _researchVisibleLimit);
|
| let html = '';
|
| for (const r of items) {
|
| const date = r.completed_at ? new Date(r.completed_at * 1000).toLocaleDateString(undefined, { month: 'short', day: 'numeric', year: 'numeric' }) : '';
|
| const time = r.completed_at ? new Date(r.completed_at * 1000).toLocaleTimeString(undefined, { hour: '2-digit', minute: '2-digit' }) : '';
|
| const sources = r.source_count || 0;
|
| const duration = r.duration || '';
|
| const rounds = r.rounds || '';
|
| const selected = _researchSelected.has(r.id);
|
| const metaBits = [];
|
| if (date) metaBits.push(`${date} ${time}`);
|
| if (sources) metaBits.push(`${sources} sources`);
|
| if (rounds) metaBits.push(`${rounds} rounds`);
|
| if (duration) metaBits.push(`${duration}`);
|
| const metaText = metaBits.join(' \u00B7 ');
|
| html += `<div class="memory-item doclib-chat-row doclib-research-card" data-research-id="${r.id}" style="cursor:pointer;">`;
|
| html += `<div class="doclib-chat-header" style="display:flex;align-items:center;width:100%;gap:6px;">`;
|
| if (_researchSelectMode) html += `<input type="checkbox" class="memory-select-cb _res-cb" data-rid="${r.id}"${selected ? ' checked' : ''}>`;
|
| html += `<div style="flex:1;min-width:0;">`;
|
| html += `<div class="memory-item-title"><svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="vertical-align:-2px;margin-right:4px;opacity:0.4;flex-shrink:0;"><circle cx="11" cy="11" r="8"/><path d="M21 21l-4.35-4.35"/><line x1="11" y1="8" x2="11" y2="14"/><line x1="8" y1="11" x2="14" y2="11"/></svg>${_esc(r.query || 'Untitled Research')}</div>`;
|
| html += `<div class="memory-item-meta" style="font-size:10px;opacity:0.4;margin-top:2px;">${metaText}</div>`;
|
| html += `</div>`;
|
| if (!_researchSelectMode) html += `<div class="memory-item-actions"><button class="memory-item-btn doclib-research-delete" data-rid="${r.id}" title="Delete"><svg width="14" height="14" viewBox="0 0 24 24" fill="currentColor"><circle cx="12" cy="5" r="2"/><circle cx="12" cy="12" r="2"/><circle cx="12" cy="19" r="2"/></svg></button></div>`;
|
| html += `</div>`;
|
| html += `<div class="doclib-chat-preview" style="display:none;"></div>`;
|
| html += `</div>`;
|
| }
|
| grid.innerHTML = html;
|
| _maybeCascadeGrid(grid, 'research');
|
|
|
|
|
| grid.querySelectorAll('._res-cb').forEach(cb => {
|
| cb.addEventListener('click', e => e.stopPropagation());
|
| cb.addEventListener('change', () => {
|
| if (cb.checked) _researchSelected.add(cb.dataset.rid); else _researchSelected.delete(cb.dataset.rid);
|
| _updateResearchCount();
|
| });
|
| });
|
|
|
|
|
|
|
| grid.querySelectorAll('.doclib-research-card').forEach(card => {
|
| card.addEventListener('click', (e) => {
|
| if (card._suppressNextClick) { card._suppressNextClick = false; return; }
|
| if (e.target.closest('.doclib-research-delete') || e.target.closest('._res-cb') || e.target.closest('.doclib-chat-open-btn')) return;
|
| const rid = card.dataset.researchId;
|
| if (_researchSelectMode) {
|
| const cb = card.querySelector('._res-cb');
|
| if (cb) { cb.checked = !cb.checked; cb.dispatchEvent(new Event('change')); }
|
| return;
|
| }
|
| const item = _researchItems.find(r => r.id === rid);
|
| if (item) _toggleResearchPreview(card, item);
|
| });
|
| _attachLongPressMenu(card, '.doclib-research-delete');
|
| });
|
|
|
|
|
|
|
| grid.querySelectorAll('.doclib-research-delete').forEach(btn => {
|
| btn.addEventListener('click', (e) => {
|
| e.stopPropagation();
|
| const rid = btn.dataset.rid;
|
| _showLibDropdown(btn, [
|
| { label: 'Open', action: () => {
|
| const a = document.createElement('a');
|
| a.href = '/api/research/report/' + rid;
|
| a.target = '_blank';
|
| a.rel = 'noopener';
|
| document.body.appendChild(a);
|
| a.click();
|
| a.remove();
|
| } },
|
| { label: _researchArchivedView ? 'Restore' : 'Archive', action: async () => {
|
| const toArchived = !_researchArchivedView;
|
| const card = btn.closest('.doclib-research-card');
|
| if (card) { card.style.transition = 'opacity 0.25s, transform 0.25s'; card.style.opacity = '0'; card.style.transform = 'scale(0.95)'; }
|
| try { await fetch('/api/research/' + rid + '/archive?archived=' + toArchived, { method: 'POST', credentials: 'same-origin' }); } catch {}
|
| await new Promise(r => setTimeout(r, 200));
|
| _researchItems = _researchItems.filter(r => r.id !== rid);
|
| _renderResearchGrid();
|
| if (uiModule) uiModule.showToast(toArchived ? 'Archived' : 'Restored');
|
| } },
|
| { label: 'Delete', danger: true, action: async () => {
|
| if (!await window.styledConfirm('Delete this research?', { confirmText: 'Delete', danger: true })) return;
|
| const card = btn.closest('.doclib-research-card');
|
| if (card) {
|
| card.style.transition = 'opacity 0.25s, transform 0.25s';
|
| card.style.opacity = '0';
|
| card.style.transform = 'scale(0.95)';
|
| }
|
| await new Promise(r => setTimeout(r, 250));
|
| await fetch('/api/research/' + rid, { method: 'DELETE', credentials: 'same-origin' });
|
| _researchItems = _researchItems.filter(r => r.id !== rid);
|
| _renderResearchGrid();
|
| } },
|
| ], { onSelect: () => {
|
| _researchSelectMode = true;
|
| _researchSelected.add(rid);
|
| document.getElementById('doclib-research-bulk')?.classList.remove('hidden');
|
| _renderResearchGrid();
|
| } });
|
| });
|
| });
|
| _appendInlineLoadMore(grid, total, _researchVisibleLimit, () => {
|
| _researchVisibleLimit += _LIB_PAGE_SIZE;
|
| _renderResearchGrid();
|
| });
|
| }
|
|
|
|
|
| const researchSortEl = document.getElementById('doclib-research-sort');
|
| if (researchSortEl) researchSortEl.addEventListener('change', () => _renderResearchGrid());
|
| const researchSearchEl = document.getElementById('doclib-research-search');
|
| if (researchSearchEl) {
|
| researchSearchEl.addEventListener('input', () => {
|
| _researchSearch = researchSearchEl.value.trim();
|
| _renderResearchGrid();
|
| });
|
| }
|
|
|
| function _updateResearchCount() {
|
| const el = document.getElementById('doclib-research-selected-count');
|
| if (el) el.textContent = _researchSelected.size + ' Selected';
|
| const arc = document.getElementById('doclib-research-bulk-archive');
|
| if (arc) arc.textContent = _researchArchivedView ? 'Restore' : 'Archive';
|
| }
|
|
|
|
|
| document.getElementById('doclib-research-select-btn')?.addEventListener('click', () => {
|
| _researchSelectMode = !_researchSelectMode;
|
| _researchSelected.clear();
|
| document.getElementById('doclib-research-bulk').classList.toggle('hidden', !_researchSelectMode);
|
| _renderResearchGrid();
|
| });
|
|
|
|
|
|
|
|
|
| document.getElementById('doclib-research-tidy-btn')?.addEventListener('click', async (e) => {
|
| const btn = e.currentTarget;
|
| const origHTML = btn.innerHTML;
|
| btn.disabled = true;
|
| btn.classList.add('spinning');
|
| btn.textContent = '';
|
| const sp = spinnerModule.create('', 'clean', 'whirlpool');
|
| const el = sp.createElement();
|
| el.style.position = 'relative';
|
| el.style.top = '1px';
|
| btn.appendChild(el);
|
| sp.start();
|
| try {
|
| const candidates = [];
|
| const needFetch = [];
|
| for (const r of _researchItems) {
|
| if ((r.source_count || 0) === 0) candidates.push(r);
|
| else needFetch.push(r);
|
| }
|
| const results = await Promise.all(needFetch.map(async r => {
|
| try {
|
| const res = await fetch('/api/research/detail/' + r.id, { credentials: 'same-origin' });
|
| if (!res.ok) return null;
|
| const d = await res.json();
|
|
|
|
|
| const body = (d.result || d.raw_report || '').trim();
|
| return body.length < 200 ? r : null;
|
| } catch { return null; }
|
| }));
|
| for (const r of results) if (r) candidates.push(r);
|
| if (candidates.length === 0) {
|
| if (uiModule) uiModule.showToast('Nothing to tidy');
|
| return;
|
| }
|
| await Promise.all(candidates.map(r => fetch('/api/research/' + r.id, { method: 'DELETE', credentials: 'same-origin' }).catch(() => {})));
|
| const ids = new Set(candidates.map(r => r.id));
|
| _researchItems = _researchItems.filter(r => !ids.has(r.id));
|
| _renderResearchGrid();
|
| if (uiModule) uiModule.showToast('Deleted ' + candidates.length);
|
| } finally {
|
| sp.stop();
|
| btn.disabled = false;
|
| btn.classList.remove('spinning');
|
| btn.innerHTML = origHTML;
|
| }
|
| });
|
| document.getElementById('doclib-research-archived-btn')?.addEventListener('click', (e) => {
|
| _researchArchivedView = !_researchArchivedView;
|
| e.currentTarget.classList.toggle('active', _researchArchivedView);
|
| e.currentTarget.title = _researchArchivedView ? 'Show active research' : 'Show archived research';
|
| if (_researchSelectMode) { _researchSelectMode = false; _researchSelected.clear(); document.getElementById('doclib-research-bulk').classList.add('hidden'); }
|
| _renderLibResearch();
|
| });
|
| document.getElementById('doclib-research-bulk-cancel')?.addEventListener('click', () => {
|
| _researchSelectMode = false;
|
| _researchSelected.clear();
|
| document.getElementById('doclib-research-bulk').classList.add('hidden');
|
| _renderResearchGrid();
|
| });
|
|
|
|
|
| document.getElementById('doclib-research-select-all')?.addEventListener('change', () => {
|
| const allCb = document.getElementById('doclib-research-select-all');
|
| const newState = allCb?.checked;
|
| _researchItems.forEach(r => { if (newState) _researchSelected.add(r.id); else _researchSelected.delete(r.id); });
|
| _updateResearchCount();
|
| _renderResearchGrid();
|
| });
|
|
|
|
|
| document.getElementById('doclib-research-bulk-delete')?.addEventListener('click', async () => {
|
| const count = _researchSelected.size;
|
| if (!count) return;
|
| if (!await window.styledConfirm(`Delete ${count} research report${count > 1 ? 's' : ''} permanently?`, { confirmText: 'Delete', danger: true })) return;
|
| const grid = document.getElementById('doclib-research-grid');
|
| if (grid) {
|
| grid.querySelectorAll('.doclib-research-card').forEach(card => {
|
| if (_researchSelected.has(card.dataset.researchId)) {
|
| card.style.transition = 'opacity 0.25s, transform 0.25s';
|
| card.style.opacity = '0';
|
| card.style.transform = 'scale(0.95)';
|
| }
|
| });
|
| }
|
| await new Promise(r => setTimeout(r, 250));
|
| await Promise.all([..._researchSelected].map(rid => fetch('/api/research/' + rid, { method: 'DELETE', credentials: 'same-origin' })));
|
| _researchItems = _researchItems.filter(r => !_researchSelected.has(r.id));
|
| _researchSelected.clear();
|
| _researchSelectMode = false;
|
| document.getElementById('doclib-research-bulk').classList.add('hidden');
|
| _renderResearchGrid();
|
| });
|
|
|
|
|
| document.getElementById('doclib-research-bulk-archive')?.addEventListener('click', async () => {
|
| const count = _researchSelected.size;
|
| if (!count) return;
|
| const toArchived = !_researchArchivedView;
|
| const grid = document.getElementById('doclib-research-grid');
|
| if (grid) {
|
| grid.querySelectorAll('.doclib-research-card').forEach(card => {
|
| if (_researchSelected.has(card.dataset.researchId)) {
|
| card.style.transition = 'opacity 0.25s, transform 0.25s';
|
| card.style.opacity = '0';
|
| card.style.transform = 'scale(0.95)';
|
| }
|
| });
|
| }
|
| await new Promise(r => setTimeout(r, 250));
|
| await Promise.all([..._researchSelected].map(rid => fetch('/api/research/' + rid + '/archive?archived=' + toArchived, { method: 'POST', credentials: 'same-origin' })));
|
| _researchItems = _researchItems.filter(r => !_researchSelected.has(r.id));
|
| _researchSelected.clear();
|
| _researchSelectMode = false;
|
| document.getElementById('doclib-research-bulk').classList.add('hidden');
|
| _renderResearchGrid();
|
| if (uiModule) uiModule.showToast(toArchived ? 'Archived' : 'Restored');
|
| });
|
|
|
|
|
|
|
|
|
|
|
| function _relTime(iso) {
|
| if (!iso) return '';
|
| const diff = Date.now() - new Date(iso).getTime();
|
| const mins = Math.floor(diff / 60000);
|
| if (mins < 60) return mins + 'm ago';
|
| const hrs = Math.floor(mins / 60);
|
| if (hrs < 24) return hrs + 'h ago';
|
| const days = Math.floor(hrs / 24);
|
| if (days < 30) return days + 'd ago';
|
| return new Date(iso).toLocaleDateString();
|
| }
|
|
|
|
|
|
|
|
|
| _switchLibTab(_activeLibTab);
|
|
|
| const searchInput = document.getElementById('doclib-search');
|
| searchInput.addEventListener('input', () => {
|
| clearTimeout(_librarySearchDebounce);
|
| _librarySearchDebounce = setTimeout(() => {
|
| _librarySearch = searchInput.value.trim();
|
| libraryFetch(false);
|
| }, 300);
|
| });
|
|
|
| document.getElementById('doclib-sort').addEventListener('change', (e) => {
|
| _librarySort = e.target.value;
|
| libraryFetch(false);
|
| });
|
|
|
| document.getElementById('doclib-load-more').addEventListener('click', () => {
|
| _libraryOffset = _libraryDocs.length;
|
| libraryFetch(true);
|
| });
|
|
|
|
|
| const grid = document.getElementById('doclib-grid');
|
| if (grid) {
|
| grid.addEventListener('scroll', () => libraryRenderLoadMore());
|
|
|
|
|
|
|
| if (typeof ResizeObserver !== 'undefined') {
|
| new ResizeObserver(() => libraryRenderLoadMore()).observe(grid);
|
| }
|
| }
|
|
|
|
|
| const importFileBtn = document.getElementById('doclib-import-file-btn');
|
| const fileInput = document.getElementById('doclib-file-input');
|
| if (importFileBtn && fileInput) {
|
| importFileBtn.addEventListener('click', () => fileInput.click());
|
| fileInput.addEventListener('change', async () => {
|
| if (fileInput.files.length === 0) return;
|
| const files = Array.from(fileInput.files);
|
| fileInput.value = '';
|
|
|
| const _orig = importFileBtn.innerHTML;
|
| importFileBtn.disabled = true;
|
| let _sp = null;
|
| try {
|
| _sp = spinnerModule.createWhirlpool(12);
|
| _sp.element.style.cssText = 'width:12px;height:12px;margin:0 4px 0 0;display:inline-block;vertical-align:middle;position:relative;top:-2px;';
|
| importFileBtn.innerHTML = '';
|
| importFileBtn.appendChild(_sp.element);
|
| importFileBtn.appendChild(document.createTextNode('Import'));
|
| } catch {}
|
| try {
|
| await libraryImportFiles(files);
|
| } finally {
|
| try { _sp && _sp.stop(); } catch {}
|
| importFileBtn.innerHTML = _orig;
|
| importFileBtn.disabled = false;
|
| }
|
| });
|
| }
|
|
|
|
|
| const createBtn = document.getElementById('doclib-create-btn');
|
| if (createBtn) {
|
| createBtn.addEventListener('click', async () => {
|
|
|
| try {
|
| const sRes = await fetch('/api/session', { method: 'POST', credentials: 'same-origin', headers: { 'Content-Type': 'application/json' }, body: JSON.stringify({ title: 'Untitled Document' }) });
|
| const sData = await sRes.json();
|
| const sessionId = sData.session_id;
|
| await _createDocument(sessionId);
|
|
|
| closeLibrary();
|
| if (window.sessionsModule) window.sessionsModule.loadSession(sessionId);
|
| setTimeout(() => _openPanel(), 300);
|
| } catch (e) {
|
| console.error('Failed to create document:', e);
|
| if (uiModule) uiModule.showError('Failed to create document');
|
| }
|
| });
|
| }
|
|
|
|
|
| const archivedBtn = document.getElementById('doclib-archived-btn');
|
| if (archivedBtn) archivedBtn.addEventListener('click', () => {
|
| _libraryArchivedView = !_libraryArchivedView;
|
| archivedBtn.classList.toggle('active', _libraryArchivedView);
|
| archivedBtn.title = _libraryArchivedView ? 'Show active documents' : 'Show archived documents';
|
| if (_librarySelectMode) libraryExitSelectMode();
|
| libraryFetch(false);
|
| });
|
|
|
|
|
| const tidyBtn = document.getElementById('doclib-tidy-btn');
|
| if (tidyBtn) tidyBtn.addEventListener('click', async () => {
|
| tidyBtn.disabled = true;
|
| tidyBtn.classList.add('spinning');
|
| const origHTML = tidyBtn.innerHTML;
|
| tidyBtn.textContent = '';
|
| const spinner = spinnerModule.create('', 'clean', 'whirlpool');
|
| const _spEl = spinner.createElement();
|
|
|
| _spEl.style.position = 'relative';
|
| _spEl.style.top = '1px';
|
| tidyBtn.appendChild(_spEl);
|
| spinner.start();
|
|
|
| let totalDeleted = 0;
|
| let totalFixed = 0;
|
| let aiMessage = '';
|
| try {
|
|
|
| const [res1] = await Promise.all([
|
| fetch(`${API_BASE}/api/documents/tidy`, { method: 'POST' }),
|
| new Promise(r => setTimeout(r, 600)),
|
| ]);
|
| if (res1.ok) {
|
| const d1 = await res1.json();
|
| totalDeleted += d1.deleted || 0;
|
| totalFixed += d1.fixed_titles || 0;
|
| }
|
|
|
|
|
| try {
|
| const res2 = await fetch(`${API_BASE}/api/documents/ai-tidy`, { method: 'POST' });
|
| if (res2.ok) {
|
| const d2 = await res2.json();
|
| totalDeleted += d2.deleted || 0;
|
| if (d2.message) aiMessage = d2.message;
|
| }
|
| } catch (_) { }
|
|
|
| spinner.destroy();
|
|
|
| if (totalDeleted === 0 && totalFixed === 0) {
|
| tidyBtn.innerHTML = '<span style="opacity:0.7">Already tidy</span>';
|
| } else {
|
| const msg = aiMessage || `Removed ${totalDeleted} document${totalDeleted !== 1 ? 's' : ''}`;
|
| if (uiModule) uiModule.showToast(msg);
|
| libraryFetch(false);
|
| }
|
| setTimeout(() => { tidyBtn.innerHTML = origHTML; tidyBtn.disabled = false; tidyBtn.classList.remove('spinning'); }, 1500);
|
| } catch (e) {
|
| spinner.destroy();
|
| console.error('Document tidy failed:', e);
|
| if (uiModule) uiModule.showToast('Tidy failed');
|
| tidyBtn.disabled = false;
|
| tidyBtn.classList.remove('spinning');
|
| tidyBtn.innerHTML = origHTML;
|
| }
|
| });
|
|
|
|
|
| const selectBtn = document.getElementById('doclib-select-btn');
|
| if (selectBtn) selectBtn.addEventListener('click', () => {
|
| if (_librarySelectMode) libraryExitSelectMode();
|
| else libraryEnterSelectMode();
|
| });
|
|
|
| const selectAll = document.getElementById('doclib-select-all');
|
| if (selectAll) selectAll.addEventListener('change', libraryToggleSelectAll);
|
|
|
|
|
| const bulkCheckLabel = modal.querySelector('.memory-bulk-check-all');
|
| if (bulkCheckLabel) {
|
| bulkCheckLabel.addEventListener('click', (e) => {
|
| if (e.target === selectAll) return;
|
| e.preventDefault();
|
| selectAll.checked = !selectAll.checked;
|
| libraryToggleSelectAll();
|
| });
|
| }
|
| const selectedCountEl = document.getElementById('doclib-selected-count');
|
| if (selectedCountEl) {
|
| selectedCountEl.style.cursor = 'pointer';
|
| selectedCountEl.addEventListener('click', () => {
|
| selectAll.checked = !selectAll.checked;
|
| libraryToggleSelectAll();
|
| });
|
| }
|
|
|
| const bulkActionsBtn = document.getElementById('doclib-bulk-actions');
|
| if (bulkActionsBtn) bulkActionsBtn.addEventListener('click', (e) => {
|
| e.stopPropagation();
|
| if (_librarySelectedIds.size === 0) {
|
| if (uiModule) uiModule.showToast('Select documents first');
|
| return;
|
| }
|
| _showLibDropdown(e.currentTarget, [
|
| { label: _libraryArchivedView ? 'Restore' : 'Archive', icon: _libraryArchivedView ? 'restore' : 'archive', action: libraryBulkArchive },
|
| { label: 'Clone', icon: 'clone', action: libraryBulkClone },
|
| { label: 'Export', icon: 'open', action: libraryBulkExport },
|
| { label: 'Delete', icon: 'delete', danger: true, action: libraryBulkDelete },
|
| ], { onCancel: libraryExitSelectMode });
|
| });
|
|
|
| const bulkCancelBtn = document.getElementById('doclib-bulk-cancel');
|
| if (bulkCancelBtn) bulkCancelBtn.addEventListener('click', libraryExitSelectMode);
|
|
|
|
|
| modal.addEventListener('click', (e) => {
|
| if (uiModule.isTouchInsideModal()) return;
|
| if (e.target === modal) closeLibrary();
|
| });
|
|
|
|
|
| _libraryEscHandler = (e) => {
|
| if (e.key === 'Escape') {
|
|
|
| const expanded = document.querySelector('#doclib-grid .doclib-card-expanded');
|
| if (expanded) {
|
| _collapseExpandedCard(expanded);
|
| } else {
|
| closeLibrary();
|
| }
|
| }
|
| };
|
| document.addEventListener('keydown', _libraryEscHandler);
|
|
|
|
|
| const btn = document.getElementById('tool-doclib-btn');
|
| if (btn) btn.classList.add('active');
|
|
|
| libraryFetch(false);
|
| if (window.innerWidth >= 768) searchInput.focus();
|
| }
|
|
|
| export function closeLibrary() {
|
| if (!_libraryOpen) return;
|
| _libraryOpen = false;
|
| _librarySelectMode = false;
|
| _librarySelectedIds.clear();
|
| _libraryImportMode = false;
|
| clearTimeout(_librarySearchDebounce);
|
|
|
| const modal = document.getElementById('doclib-modal');
|
| if (modal) {
|
| const content = modal.querySelector('.modal-content, .doclib-modal-content');
|
| if (content) {
|
| content.classList.add('modal-closing');
|
| content.addEventListener('animationend', () => modal.remove(), { once: true });
|
| setTimeout(() => { if (modal.parentElement) modal.remove(); }, 250);
|
| } else {
|
| modal.remove();
|
| }
|
| }
|
|
|
| if (_libraryEscHandler) {
|
| document.removeEventListener('keydown', _libraryEscHandler);
|
| _libraryEscHandler = null;
|
| }
|
|
|
| const btn = document.getElementById('tool-doclib-btn');
|
| if (btn) btn.classList.remove('active');
|
| }
|
|
|
| export function isLibraryOpen() {
|
| return _libraryOpen;
|
| }
|
|
|