// Phase 6.1 — Documents UI
// All DOM interaction for the docs sidebar.
// Uses only: getEditorText(), loadDocumentText(), documents-api, documents-state.
let _renameDocId = null;
/* ── Initialization ── */
function initDocumentsCloud() {
_renderSidebar();
_bindSidebarToggle();
_bindEditorDirty();
_loadAndRenderList();
// Listen to auth changes — reload list when user signs in
window.addEventListener('bayan:docstate', () => _updateTitleBar());
window.addEventListener('bayan:authchange', () => {
_loadAndRenderList();
});
// Warn user if they try to refresh with unsaved changes
window.addEventListener('beforeunload', (e) => {
const state = getDocState();
if (state.hasUnsavedChanges) {
e.preventDefault();
e.returnValue = '';
}
});
// Sync Manager UI Updates
window.addEventListener('bayan:syncstate', (e) => {
const { state } = e.detail;
const saveBtn = document.getElementById('doc-save-btn');
if (!saveBtn) return;
if (state === 'saving') {
saveBtn.title = 'جاري الحفظ...';
saveBtn.classList.add('is-saving');
if (typeof showAutoSaveStatus === 'function') showAutoSaveStatus('جاري الحفظ...');
} else if (state === 'saved') {
saveBtn.title = 'تم الحفظ';
if (typeof showAutoSaveStatus === 'function') showAutoSaveStatus('✓ تم الحفظ');
saveBtn.classList.remove('is-saving', 'doc-save-btn--dirty');
saveBtn.classList.add('is-saved');
setTimeout(() => {
saveBtn.classList.remove('is-saved');
const currentState = getDocState();
saveBtn.title = currentState.hasUnsavedChanges ? 'حفظ (يوجد تغييرات غير محفوظة)' : 'حفظ';
}, 2000);
} else if (state === 'saved_locally') {
saveBtn.title = 'محفوظ محلياً (أنت غير متصل)';
saveBtn.classList.add('doc-save-btn--dirty');
} else if (state === 'error') {
saveBtn.title = 'خطأ في الحفظ';
saveBtn.classList.add('doc-save-btn--dirty');
}
});
}
/* ── Sidebar toggle ── */
function _bindSidebarToggle() {
const toggleBtn = document.getElementById('docs-sidebar-toggle');
const sidebar = document.getElementById('docs-sidebar');
if (!toggleBtn || !sidebar) return;
toggleBtn.addEventListener('click', () => {
const isOpen = sidebar.classList.toggle('is-open');
toggleBtn.setAttribute('aria-expanded', isOpen ? 'true' : 'false');
});
// Close when clicking outside
document.addEventListener('click', (e) => {
if (sidebar && sidebar.classList.contains('is-open') &&
!sidebar.contains(e.target) &&
!toggleBtn.contains(e.target)) {
sidebar.classList.remove('is-open');
toggleBtn.setAttribute('aria-expanded', 'false');
}
});
}
/* ── Mark dirty on editor input ── */
function _bindEditorDirty() {
const editor = document.getElementById('editor-container');
if (!editor) return;
editor.addEventListener('input', () => {
if (hasOpenDocument()) {
markDirty();
const state = getDocState();
const content = getEditorText();
if (typeof SyncManager !== 'undefined') {
SyncManager.queueChange(state.currentDocumentId, content);
}
}
});
}
/* ── Autosave Removed (Handled by SyncManager) ── */
/* ── Load & render list ── */
async function _loadAndRenderList() {
const listEl = document.getElementById('docs-list');
if (!listEl) return;
const isAuthenticated = window.__bayanAuth &&
window.__bayanAuth.userId &&
!window.__bayanAuth.isOfflineMode;
console.log('[DEBUG_AUTH]', JSON.stringify(window.__bayanAuth));
if (!isAuthenticated) {
listEl.innerHTML = `
سجّل دخولك لحفظ مستنداتك في السحابة
`;
return;
}
listEl.innerHTML = `جاري التحميل...
`;
const docs = await loadDocuments();
if (!docs.length) {
listEl.innerHTML = `
\u0644\u0627 \u062a\u0648\u062c\u062f \u0645\u0633\u062a\u0646\u062f\u0627\u062a \u0628\u0639\u062f
\u0623\u0646\u0634\u0626 \u0645\u0633\u062a\u0646\u062f\u064b\u0627 \u062c\u062f\u064a\u062f\u064b\u0627 \u0644\u0628\u062f\u0621 \u0627\u0644\u0643\u062a\u0627\u0628\u0629
`;
return;
}
const state = getDocState();
listEl.innerHTML = docs.map(doc => _buildDocItemHTML(doc, doc.id === state.currentDocumentId)).join('');
_bindDocItemEvents(listEl);
}
function _buildDocItemHTML(doc, isActive) {
const date = new Date(doc.updated_at).toLocaleDateString('ar-EG', {
month: 'short', day: 'numeric'
});
return `
`;
}
function _bindDocItemEvents(container) {
container.querySelectorAll('.doc-list-item__open').forEach(btn => {
btn.addEventListener('click', () => _openDocument(btn.dataset.docId));
});
container.querySelectorAll('.doc-rename-btn').forEach(btn => {
btn.addEventListener('click', (e) => {
e.stopPropagation();
_startRename(btn.dataset.docId, btn.dataset.docTitle);
});
});
container.querySelectorAll('.doc-delete-btn').forEach(btn => {
btn.addEventListener('click', (e) => {
e.stopPropagation();
_confirmDelete(btn.dataset.docId, btn.dataset.docTitle);
});
});
}
/* ── Open document ── */
async function _openDocument(id) {
const doc = await loadDocument(id);
if (!doc) {
if (typeof showDocToast === 'function') showDocToast('تعذّر تحميل المستند', 'error');
return;
}
let contentToLoad = doc.content;
if (typeof SyncManager !== 'undefined') {
contentToLoad = await SyncManager.loadAndResolveDocument(id);
if (contentToLoad === null) return;
}
loadDocumentText(contentToLoad, { analyze: true });
setDocState({
currentDocumentId: doc.id,
currentDocumentTitle: doc.title,
hasUnsavedChanges: false
});
_updateTitleBar();
_refreshActiveItem(id);
// Navigate to editor
if (typeof showPage === 'function') showPage('editor');
// Close sidebar on mobile
const sidebar = document.getElementById('docs-sidebar');
if (sidebar && window.innerWidth < 1024) {
sidebar.classList.remove('is-open');
}
}
/* ── Create document ── */
async function _createNewDocument() {
const isAuthenticated = window.__bayanAuth &&
window.__bayanAuth.userId &&
!window.__bayanAuth.isOfflineMode;
if (!isAuthenticated) {
if (typeof showDocToast === 'function') showDocToast('سجّل دخولك لحفظ المستندات', 'info');
return;
}
const titleInput = prompt('اسم المستند الجديد:', 'مستند جديد');
if (titleInput === null) return; // User pressed Cancel
const title = titleInput.trim() || 'مستند جديد';
const doc = await createDocument(title, '');
if (!doc) {
if (typeof showDocToast === 'function') showDocToast('تعذّر إنشاء المستند', 'error');
return;
}
// Clear the editor for the new empty document
loadDocumentText('', { analyze: false });
setDocState({
currentDocumentId: doc.id,
currentDocumentTitle: doc.title,
hasUnsavedChanges: false
});
_updateTitleBar();
await _loadAndRenderList();
if (typeof showDocToast === 'function') showDocToast('تم إنشاء المستند ✓', 'success');
}
/* ── Save current document manually ── */
async function saveCurrentDocument() {
const state = getDocState();
if (!state.currentDocumentId) {
await _createNewDocument();
return;
}
const content = getEditorText();
if (typeof SyncManager !== 'undefined') {
SyncManager.queueChange(state.currentDocumentId, content);
await SyncManager.syncNow();
if (typeof showDocToast === 'function') showDocToast('تم الحفظ ✓', 'success');
} else {
const ok = await saveDocument(state.currentDocumentId, content);
if (ok) {
markClean();
_updateTitleBar();
await _loadAndRenderList();
if (typeof showDocToast === 'function') showDocToast('تم الحفظ ✓', 'success');
} else {
if (typeof showDocToast === 'function') showDocToast('تعذّر الحفظ', 'error');
}
}
}
/* ── Rename ── */
function _startRename(id, currentTitle) {
const newTitle = prompt('الاسم الجديد للمستند:', currentTitle);
if (!newTitle || newTitle === currentTitle) return;
_doRename(id, newTitle);
}
async function _doRename(id, newTitle) {
const ok = await renameDocument(id, newTitle);
if (!ok) {
if (typeof showDocToast === 'function') showDocToast('تعذّر إعادة التسمية', 'error');
return;
}
const state = getDocState();
if (state.currentDocumentId === id) {
setDocState({ currentDocumentTitle: newTitle });
_updateTitleBar();
}
await _loadAndRenderList();
}
/* ── Delete ── */
async function _confirmDelete(id, title) {
if (typeof showConfirmDialog === 'function') {
showConfirmDialog(
'\u062d\u0630\u0641 \u0627\u0644\u0645\u0633\u062a\u0646\u062f',
'\u0647\u0644 \u062a\u0631\u064a\u062f \u062d\u0630\u0641 "' + title + '"\u061f \u0644\u0627 \u064a\u0645\u0643\u0646 \u0627\u0644\u062a\u0631\u0627\u062c\u0639 \u0639\u0646 \u0647\u0630\u0627 \u0627\u0644\u0625\u062c\u0631\u0627\u0621.',
function() { _doDelete(id); }
);
} else {
if (!confirm('\u0647\u0644 \u062a\u0631\u064a\u062f \u062d\u0630\u0641 "' + title + '"\u061f \u0644\u0627 \u064a\u0645\u0643\u0646 \u0627\u0644\u062a\u0631\u0627\u062c\u0639 \u0639\u0646 \u0647\u0630\u0627 \u0627\u0644\u0625\u062c\u0631\u0627\u0621.')) return;
_doDelete(id);
}
}
async function _doDelete(id) {
const ok = await deleteDocument(id);
if (!ok) {
if (typeof showDocToast === 'function') showDocToast('\u062a\u0639\u0630\u0651\u0631 \u0627\u0644\u062d\u0630\u0641', 'error');
return;
}
const state = getDocState();
if (state.currentDocumentId === id) {
setDocState({ currentDocumentId: null, currentDocumentTitle: '\u0645\u0633\u062a\u0646\u062f \u062c\u062f\u064a\u062f', hasUnsavedChanges: false });
_updateTitleBar();
}
await _loadAndRenderList();
if (typeof showDocToast === 'function') showDocToast('\u062a\u0645 \u062d\u0630\u0641 \u0627\u0644\u0645\u0633\u062a\u0646\u062f', 'success');
}
/* ── Title bar ── */
function _updateTitleBar() {
const titleEl = document.getElementById('doc-current-title');
const saveBtn = document.getElementById('doc-save-btn');
if (!titleEl) return;
const state = getDocState();
titleEl.textContent = state.currentDocumentTitle;
if (saveBtn) {
const unsaved = state.hasUnsavedChanges && state.currentDocumentId;
saveBtn.classList.toggle('doc-save-btn--dirty', !!unsaved);
const isTempState = saveBtn.classList.contains('is-saving') || saveBtn.classList.contains('is-saved');
if (!isTempState) {
saveBtn.title = unsaved ? 'حفظ (يوجد تغييرات غير محفوظة)' : 'حفظ';
}
}
}
/* ── Refresh active item in list without full reload ── */
function _refreshActiveItem(activeId) {
document.querySelectorAll('.doc-list-item').forEach(el => {
el.classList.toggle('doc-list-item--active', el.dataset.docId === activeId);
});
}
/* ── Render sidebar HTML into DOM ── */
function _renderSidebar() {
// The sidebar div is already in HTML — just wire the new-doc button
const newDocBtn = document.getElementById('docs-new-btn');
if (newDocBtn) {
newDocBtn.addEventListener('click', _createNewDocument);
}
const saveBtn = document.getElementById('doc-save-btn');
if (saveBtn) {
saveBtn.addEventListener('click', saveCurrentDocument);
}
}
/* ── Helpers ── */
function _escapeHtml(str) {
return String(str || '')
.replace(/&/g, '&')
.replace(//g, '>')
.replace(/"/g, '"');
}
function _escapeAttr(str) {
return String(str || '').replace(/"/g, '"').replace(/'/g, ''');
}