/* ========= منطق التحليل والتصدير والنسخ (Static فقط) ========= */ const EXPORT_COLUMNS = [ "نوع المشكلة","وقت حدوث المشكلة","اسم صاحب المشكلة", "رقم الهوية","رقم الجهاز","رقم الجوال","المسح","المنطقة" ]; const FIELD_ALIASES = { "نوع المشكلة": ["نوع المشكله","نوع المشكلة","المشكلة"], "وقت حدوث المشكلة": ["وقت حدوث المشكله","وقت حدوث المشكلة","وقت المشكلة","وقت حدوث"], "اسم صاحب المشكلة": ["اسم صاحب المشكله","اسم صاحب المشكلة","اسم صاحب البلاغ","الاسم"], "رقم الهوية": ["رقم الهويه","رقم الهوية","الهوية"], "رقم الجهاز": ["رقم الجهاز","الجهاز"], "رقم الجوال": ["رقم الجوال","الجوال","الهاتف"], "المسح": ["المسح","اسم المسح"], "المنطقة": ["المنطقة","المنطقه","المدينة","المحافظة","منطقة"], }; const LABEL_SEP = "(?::|:)?\\s*"; const TICKET_SEP = /\n\s*(?:\n|—+|-{3,}|={3,}|🔴+)+\s*\n/; const arabicDigitsMap = {"٠":"0","١":"1","٢":"2","٣":"3","٤":"4","٥":"5","٦":"6","٧":"7","٨":"8","٩":"9"}; function normalizeText(s){ if(typeof s!=="string") return ""; return s.replace(/[\u200f\u200e\u202a-\u202e\u2066-\u2069\u00a0]/g," ") .replace(/[٠-٩]/g, d => arabicDigitsMap[d] ) .replace(/[ــ]+/g,"") .trim(); } function normalizeTime(val){ const m = (val||"").match(/(\d{1,2})[:٫\.\-:](\d{2})\s*(ص|م)?/i); if(!m) return (val||"").trim(); let h = parseInt(m[1],10), mn = m[2], ampm = m[3]; if(ampm){ if(/م|pm/i.test(ampm) && h<12) h+=12; if(/ص|am/i.test(ampm) && h===12) h=0; } return `${String(h).padStart(2,"0")}:${mn}`; } function normalizeDate(v){ v=(v||"").trim(); let m=v.match(/(\d{1,2})[\/\-](\d{1,2})[\/\-](\d{2,4})/); if(m){ let d=+m[1], mo=+m[2], y=+m[3]; if(y<100) y+=2000; return `${y.toString().padStart(4,"0")}-${String(mo).padStart(2,"0")}-${String(d).padStart(2,"0")}`; } m=v.match(/(\d{4})[\/\-](\d{1,2})[\/\-](\d{1,2})/); if(m){ let y=+m[1], mo=+m[2], d=+m[3]; return `${y.toString().padStart(4,"0")}-${String(mo).padStart(2,"0")}-${String(d).padStart(2,"0")}`; } return v; } function splitTickets(raw){ raw = normalizeText(raw); if(!raw) return []; let parts = raw.split(TICKET_SEP); if(parts.length===1){ parts = raw.split(/\n\s*\n+/).filter(p=>p.trim()); } return parts.map(p=>p.trim()).filter(Boolean); } function compileFieldPatterns(){ const pats = {}; for(const [canonical, labels] of Object.entries(FIELD_ALIASES)){ const lbls = labels.map(l => l.replace(/[.*+?^${}()|[\]\\]/g,'\\$&')).join("|"); pats[canonical] = [ new RegExp(`(?:^|\\n)\\s*(?:${lbls})\\s*${LABEL_SEP}(.+)$`, "mi"), new RegExp(`(?:^|\\n)\\s*(?:${lbls})\\s*${LABEL_SEP}\\n\\s*(.+)`, "mi"), ]; } return pats; } const FIELD_PATTERNS = compileFieldPatterns(); function extractFields(ticketText){ const data = {}; for(const k of Object.keys(FIELD_ALIASES)) data[k]=""; const text = normalizeText(ticketText); for(const [fname, patterns] of Object.entries(FIELD_PATTERNS)){ for(const pat of patterns){ const m = text.match(pat); if(m){ let val = normalizeText(m[1]); if(fname==="وقت حدوث المشكلة") val = normalizeTime(val); if(!data[fname]) data[fname]=val; break; } } } if(!data["رقم الجهاز"]){ const m = text.match(/(?:رقم\s*الجهاز|الجهاز)\D*([0-9][0-9\-\s]{2,})/i); if(m) data["رقم الجهاز"]=m[1].replace(/\D/g,"").slice(0,20); } if(!data["رقم الجوال"]){ const m = text.match(/(05[0-9\-\s]{8,12})/); if(m) data["رقم الجوال"]=m[1].replace(/\D/g,"").slice(0,10); } if(!data["رقم الهوية"]){ const m = text.match(/(1[0-9\-\s]{9,12})/); if(m) data["رقم الهوية"]=m[1].replace(/\D/g,"").slice(0,10); } if(!data["المسح"]){ const m = text.match(/(?:اسم\s*المسح|المسح)\s*[::]?\s*(.+)/); if(m) data["المسح"]=normalizeText(m[1].split(/\r?\n/)[0]); } const dm = text.match(/(\d{1,2}[\/\-]\d{1,2}[\/\-]\d{2,4}|\d{4}[\/\-]\d{1,2}[\/\-]\d{1,2})/); if(dm){ const date = normalizeDate(dm[1]); const tm = data["وقت حدوث المشكلة"] || ""; data["وقت حدوث المشكلة"] = `${date} ${tm}`.trim(); } return data; } function parseTickets(raw){ return splitTickets(raw||"").map(extractFields); } function buildTable(rows){ const theadRow = document.getElementById("theadRow"); const tbody = document.getElementById("tbody"); theadRow.innerHTML = ""; EXPORT_COLUMNS.forEach(col=>{ const th=document.createElement("th"); th.textContent=col; theadRow.appendChild(th); }); tbody.innerHTML = ""; rows.forEach(r=>{ const tr=document.createElement("tr"); EXPORT_COLUMNS.forEach(col=>{ const td=document.createElement("td"); td.contentEditable="true"; td.textContent = r[col]||""; tr.appendChild(td); }); tbody.appendChild(tr); }); } /* قراءة الجدول */ function readTable(){ const tbody = document.getElementById("tbody"); const rows = []; [...tbody.querySelectorAll("tr")].forEach(tr=>{ const obj={}; [...tr.children].forEach((td,idx)=>{ obj[EXPORT_COLUMNS[idx]] = td.textContent.trim(); }); rows.push(obj); }); return rows; } /* شارة عدّاد داخل زر التحليل */ function updateBadge(n){ const b = document.getElementById("countBadge"); b.textContent = n; b.hidden = (n===0); } /* تمكين/تعطيل الأزرار حسب وجود صفوف */ function setButtonsEnabled(hasRows){ document.getElementById("btn-export").disabled = !hasRows; document.getElementById("btn-copy").disabled = !hasRows; } /* إبراز الخلايا غير الصحيحة (هوية/جوال) */ function validateCells(){ const tbody=document.getElementById("tbody"); const idxPhone = EXPORT_COLUMNS.indexOf("رقم الجوال"); const idxID = EXPORT_COLUMNS.indexOf("رقم الهوية"); [...tbody.rows].forEach(tr=>{ if(idxPhone>=0){ const td=tr.children[idxPhone], v=td.textContent.trim(); const ok=/^05\d{8}$/.test(v); td.classList.toggle("invalid", v && !ok); } if(idxID>=0){ const td=tr.children[idxID], v=td.textContent.trim(); const ok=/^1\d{9}$/.test(v); td.classList.toggle("invalid", v && !ok); } }); } document.addEventListener("input",(e)=>{ if(e.target && e.target.closest && e.target.closest("#tbody")){ validateCells(); } }); /* Toast */ function toast(msg){ const t = document.getElementById("toast"); t.textContent = msg; t.hidden = false; t.classList.remove("show"); void t.offsetWidth; t.classList.add("show"); setTimeout(()=>{ t.hidden = true; }, 2400); } /* تصدير/مشاركة Excel: Web Share أولاً ثم تنزيل كبديل */ async function exportExcel(){ const rows = readTable(); if(!rows.length){ toast("لا يوجد بيانات لتصديرها."); return; } const aoa = [EXPORT_COLUMNS, ...rows.map(r=>EXPORT_COLUMNS.map(c=>r[c]||""))]; const ws = XLSX.utils.aoa_to_sheet(aoa); ws["!rtl"] = true; const wb = XLSX.utils.book_new(); XLSX.utils.book_append_sheet(wb, ws, "التذاكر"); const now = new Date(); const ts = now.toISOString().replace(/\D/g,'').slice(0,14); const base = (document.getElementById("fname").value || "Ticket").trim() || "Ticket"; const filename = `${base}_${ts}.xlsx`; const wbArray = XLSX.write(wb, { bookType: "xlsx", type: "array" }); const blob = new Blob([wbArray], { type: "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet" }); const file = new File([blob], filename, { type: blob.type }); if (navigator.canShare && navigator.canShare({ files: [file] })) { try { await navigator.share({ files: [file], title: "ملف التذاكر" }); toast("تمت المشاركة/الحفظ بنجاح."); return; } catch(e){ /* المستخدم أغلق ورقة المشاركة */ } } const url = URL.createObjectURL(blob); const a = document.createElement("a"); a.href = url; a.download = filename; document.body.appendChild(a); a.click(); a.remove(); setTimeout(()=>URL.revokeObjectURL(url), 1000); toast("تم تنزيل الملف."); } /* نسخ الجدول كـTSV للصق مباشر في Excel */ async function copyToClipboardTSV(){ const rows = readTable(); if(!rows.length){ toast("لا يوجد بيانات لنسخها."); return; } const header = EXPORT_COLUMNS.join("\t"); const body = rows.map(r=>EXPORT_COLUMNS.map(c=>(r[c]||"").replace(/\t/g," ")).join("\t")).join("\n"); const tsv = `${header}\n${body}`; try{ await navigator.clipboard.writeText(tsv); toast("تم النسخ — الصق/ي مباشرة في Excel."); } catch(e){ const ta = document.createElement("textarea"); ta.value = tsv; document.body.appendChild(ta); ta.select(); document.execCommand("copy"); document.body.removeChild(ta); toast("تم النسخ — الصق/ي مباشرة في Excel."); } } /* المثال المطلوب (تذكرة واحدة) */ const SAMPLE = `نوع المشكلة : لا استطيع اكمال الاستمارة بسبب تعليق وقت حدوث المشكلة: 21/8/2025 اسم صاحب المشكلة : نوف الناصر رقم الهوية: 1234567890 رقم الجهاز: 01234 رقم الجوال: 0558174717 اسم المسح: الخبر المنطقة: الشرقية`; /* تهيئة وربط الأزرار + اختصارات + حفظ تلقائي */ function init(){ const parseBtn = document.getElementById("btn-parse"); const exportBtn = document.getElementById("btn-export"); const copyBtn = document.getElementById("btn-copy"); const clearBtn = document.getElementById("btn-clear"); const sampleBtn = document.getElementById("btn-sample"); const rawEl = document.getElementById("raw"); // استرجاع الإدخال السابق إن وُجد const saved = localStorage.getItem("rawTickets"); if(!rawEl.value && saved){ rawEl.value = saved; } parseBtn.addEventListener("click", ()=>{ const raw = rawEl.value || SAMPLE; const rows = parseTickets(raw); buildTable(rows); validateCells(); updateBadge(rows.length); setButtonsEnabled(rows.length>0); localStorage.setItem("rawTickets", raw); toast(`تم استخراج ${rows.length} ${rows.length===1 ? "تذكرة" : "تذاكر"}.`); }); exportBtn.addEventListener("click", exportExcel); copyBtn.addEventListener("click", copyToClipboardTSV); clearBtn.addEventListener("click", ()=>{ rawEl.value = ""; document.getElementById("tbody").innerHTML = ""; updateBadge(0); setButtonsEnabled(false); }); sampleBtn.addEventListener("click", ()=>{ rawEl.value = SAMPLE; }); // اختصارات لوحة المفاتيح document.addEventListener("keydown", (e)=>{ const ctrl = e.ctrlKey || e.metaKey; if(ctrl && e.key === "Enter"){ e.preventDefault(); parseBtn.click(); } else if(ctrl && e.key.toLowerCase() === "e"){ e.preventDefault(); exportBtn.click(); } else if(ctrl && e.shiftKey && e.key.toLowerCase() === "c"){ e.preventDefault(); copyBtn.click(); } else if(e.key === "Escape"){ e.preventDefault(); clearBtn.click(); } }); setButtonsEnabled(false); // مبدئيًا } init(); // لدينا defer في index.html