bep40 commited on
Commit
0279878
·
verified ·
1 Parent(s): fea2760

Restore livescore shorts highlights worldcup sections without AI wall

Browse files
Files changed (1) hide show
  1. static/index.html +47 -34
static/index.html CHANGED
@@ -4,56 +4,69 @@
4
  <meta charset="utf-8">
5
  <meta name="viewport" content="width=device-width,initial-scale=1,maximum-scale=1">
6
  <title>VNEWS - Tin Tức Việt Nam</title>
7
- <meta name="description" content="Tin tức tổng hợp, video short, highlight bóng đá, livescore, tường AI.">
8
  <meta property="og:title" content="VNEWS - Tin Tức Việt Nam">
9
  <meta property="og:image" content="https://s1.vnecdn.net/vnexpress/restruct/i/v9505/logo_default.jpg">
10
  <link rel="canonical" href="https://bep40-vnews.hf.space">
11
  <script src="https://cdn.jsdelivr.net/npm/hls.js@1/dist/hls.min.js"></script>
12
  <style>
13
- *{box-sizing:border-box;margin:0;padding:0}body{background:#111;color:#eee;font-family:-apple-system,BlinkMacSystemFont,'Segoe UI',Roboto,sans-serif;overflow-x:hidden}.header{background:linear-gradient(135deg,#0d1117,#1a3a2a 50%,#8b7500);padding:12px;text-align:center}.header h1{font-size:18px;color:#fff}.header p{font-size:10px;color:rgba(255,255,255,.65);margin-top:2px}.cats{display:flex;overflow-x:auto;background:#1a1a1a;border-bottom:1px solid #333;padding:0 4px;position:sticky;top:0;z-index:50;scrollbar-width:none}.cats::-webkit-scrollbar{display:none}.cat{padding:9px 11px;color:#888;font-size:11px;cursor:pointer;white-space:nowrap;border-bottom:2px solid transparent;flex-shrink:0}.cat.active{color:#5cb87a;border-bottom-color:#5cb87a;font-weight:700}.view{display:none}.view.active{display:block}.loading{text-align:center;padding:32px;color:#666;font-size:12px}.grid{display:grid;grid-template-columns:repeat(2,1fr);gap:6px;padding:6px 4px}@media(min-width:650px){.grid{grid-template-columns:repeat(3,1fr)}}.card{background:#1a1a1a;border:1px solid #222;border-radius:8px;overflow:hidden;cursor:pointer}.card:active{transform:scale(.98)}.card-img{position:relative;width:100%;aspect-ratio:16/9;background:#333;overflow:hidden}.card-img img{width:100%;height:100%;object-fit:cover}.card-body{padding:6px 8px}.card-title{font-size:11px;font-weight:650;color:#eee;line-height:1.35;display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden}.badge{font-size:8px;padding:1px 5px;border-radius:3px;font-weight:800;display:inline-block;margin-bottom:2px}.badge-vne{background:#c0392b}.badge-bbc{background:#b80000}.badge-dt{background:#1565c0}.badge-genk{background:#6a1b9a}.badge-fpt{background:#f26522}.badge-ai{background:#2d8659}.slider-wrap{margin:6px 4px;background:#1a1a1a;border:1px solid #2a2a2a;border-radius:8px;overflow:hidden}.slider-header{padding:7px 10px;display:flex;align-items:center;justify-content:space-between}.slider-label{color:#f0c040;font-size:13px;font-weight:800}.slider-note{font-size:10px;color:#777}.slider-track{display:flex;gap:8px;padding:4px 10px 10px;overflow-x:auto;scrollbar-width:none}.slider-track::-webkit-scrollbar{display:none}.slider-item{flex:0 0 160px;cursor:pointer}.slider-thumb{position:relative;width:100%;aspect-ratio:16/9;border-radius:6px;background:#333;overflow:hidden}.slider-thumb img{width:100%;height:100%;object-fit:cover}.slider-title{font-size:10px;color:#ccc;line-height:1.25;margin-top:3px;display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden}.shorts-item{flex-basis:110px}.shorts-thumb{aspect-ratio:3/4;border-radius:8px}.card-play{position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);width:30px;height:30px;background:rgba(0,0,0,.55);border-radius:50%;display:flex;align-items:center;justify-content:center;color:#fff;font-size:12px}.section-title{font-size:13px;font-weight:800;color:#5cb87a;margin:8px 0 4px;padding-left:8px;border-left:3px solid #5cb87a}.back-btn{background:rgba(17,17,17,.96);color:#fff;border:0;padding:10px;font-size:12px;cursor:pointer;width:100%;text-align:center;position:sticky;top:0;z-index:50}.article-view{padding:12px 8px 40px;max-width:760px;margin:0 auto}.article-title{font-size:18px;font-weight:850;color:#f0f0f0;line-height:1.32;margin-bottom:8px}.article-summary{background:#1a2a1f;border-left:3px solid #2d8659;padding:10px;margin-bottom:14px;font-weight:650;color:#ccc;font-size:13px;border-radius:0 6px 6px 0}.article-p{font-size:14px;line-height:1.72;color:#ccc;margin-bottom:10px}.article-img{width:100%;border-radius:6px;margin:10px 0}.article-h2{font-size:16px;font-weight:800;color:#eee;margin:16px 0 8px}.article-actions{display:flex;gap:8px;padding:10px 0;border-top:1px solid #333;margin-top:16px;flex-wrap:wrap}.article-actions button,.article-actions select{background:#1a1a1a;border:1px solid #333;color:#ccc;padding:7px 12px;border-radius:14px;font-size:11px;cursor:pointer}.article-actions button.primary{background:#2d8659;border-color:#2d8659;color:#fff}.wall-item{flex:0 0 260px;background:#141414;border:1px solid #2b2b2b;border-radius:10px;padding:8px}.wall-thumb{width:100%;aspect-ratio:16/9;border-radius:8px;background:#222;overflow:hidden;margin-bottom:6px}.wall-thumb img{width:100%;height:100%;object-fit:cover}.wall-title{font-size:12px;color:#5cb87a;font-weight:800;line-height:1.3;margin-bottom:4px;display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden}.wall-text{font-size:11px;color:#bbb;line-height:1.4;white-space:pre-wrap;display:-webkit-box;-webkit-line-clamp:5;-webkit-box-orient:vertical;overflow:hidden}.wall-actions{display:flex;gap:6px;margin-top:8px}.wall-actions button{flex:1;border:1px solid #333;background:#222;color:#ddd;border-radius:14px;padding:6px 8px;font-size:10px;cursor:pointer}.wall-actions button.primary{background:#2d8659;border-color:#2d8659;color:#fff}.rewrite-box{margin-top:12px;background:#141414;border:1px solid #2b2b2b;border-radius:10px;padding:10px}.rewrite-title{font-size:12px;font-weight:800;color:#5cb87a;margin-bottom:6px}.rewrite-text{font-size:12px;line-height:1.55;color:#ccc;white-space:pre-wrap}.ls-section{margin:6px 4px;background:#1a1a1a;border:1px solid #2a2a2a;border-radius:8px;overflow:hidden}.ls-header{padding:8px 10px}.ls-header h3{font-size:13px;color:#fff}.ls-tabs{display:flex;gap:4px;padding:0 10px 8px;overflow-x:auto;scrollbar-width:none}.ls-tab{padding:4px 10px;background:#222;border:1px solid #333;border-radius:12px;color:#999;font-size:10px;white-space:nowrap;cursor:pointer}.ls-tab.active{background:#2d8659;color:#fff;border-color:#2d8659}.ls-content{max-height:360px;overflow-y:auto;padding:0 6px 8px}.ls-content .matchs-league{list-style:none}.ls-content .match-detail{padding:6px;border-bottom:1px solid #222;cursor:pointer}.ls-content .title-content{padding:5px;background:#222;border-radius:4px;margin:6px 0}.ls-content .title-content strong{font-size:11px;color:#eee}.ls-content .match{display:flex;flex-wrap:wrap;gap:4px;align-items:center}.ls-content .datetime{width:100%;display:flex;gap:8px;color:#888;font-size:9px}.ls-content .teams{display:flex;align-items:center;width:100%;gap:5px}.ls-content .team{flex:1;display:flex;align-items:center;gap:4px;min-width:0;text-decoration:none}.ls-content .name{font-size:11px;color:#ddd;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.ls-content .logo img{width:18px;height:18px;object-fit:contain}.ls-content .home-team{justify-content:flex-end;text-align:right}.ls-content .home-team .name{order:-1}.ls-content .status{flex:0 0 50px;text-align:center}.ls-content .status a{font-size:12px;color:#fff;text-decoration:none;font-weight:800}.ls-content .status .label{font-size:8px;color:#888;display:block}.ls-content .status .label.live{color:#e74c3c}.ls-content .info,.ls-content .btns{display:none}.match-overlay{position:fixed;inset:0;background:#111;z-index:9999;display:none;flex-direction:column;overflow-y:auto}.match-overlay.active{display:flex}.mo-header{padding:10px;background:#1a1a1a;display:flex;justify-content:space-between;align-items:center;position:sticky;top:0;z-index:1}.mo-header h3{font-size:13px;color:#fff}.mo-close{background:none;border:0;color:#fff;font-size:22px}.mo-tabs{display:flex;gap:4px;padding:8px 10px;background:#1a1a1a}.mo-tab{padding:5px 12px;background:#222;border:1px solid #333;border-radius:10px;color:#999;font-size:10px;cursor:pointer}.mo-tab.active{background:#2d8659;color:#fff}.mo-body{padding:8px;overflow-x:auto}.tiktok-container{width:100%;height:80vh;max-height:680px;min-height:400px;background:#000}.tiktok-feed{height:100%;overflow-y:auto;scroll-snap-type:y mandatory;scrollbar-width:none}.tiktok-slide{height:80vh;max-height:680px;min-height:400px;scroll-snap-align:start;position:relative;background:#000;display:flex;align-items:center;justify-content:center}.tiktok-slide video,.tiktok-slide iframe{width:100%;height:100%;object-fit:contain;border:0}.tiktok-bottom{position:absolute;bottom:0;left:0;right:60px;padding:12px 10px 16px;background:linear-gradient(transparent,rgba(0,0,0,.85));z-index:3}.tiktok-title{color:#fff;font-size:12px;font-weight:650;line-height:1.3}.tiktok-counter{position:absolute;top:8px;left:8px;background:rgba(0,0,0,.5);font-size:9px;padding:2px 7px;border-radius:8px}.tiktok-right{position:absolute;right:8px;bottom:100px;display:flex;flex-direction:column;gap:14px}.tiktok-right-btn{background:none;border:0;color:#fff}.tiktok-right-btn .icon{width:42px;height:42px;border-radius:50%;background:rgba(255,255,255,.12);display:flex;align-items:center;justify-content:center;font-size:20px}
14
  </style>
15
  </head>
16
  <body>
17
- <div class="header"><h1>📰 VNEWS</h1><p>Tin tức · Video · Livescore · Rewrite AI</p></div>
18
  <div class="cats" id="cat-bar"></div>
19
  <div id="view-home" class="view active"><div class="loading">Đang tải...</div></div>
20
  <div id="view-cat" class="view"></div>
21
  <div id="view-video" class="view"></div>
22
  <div id="view-tiktok" class="view"></div>
23
  <div id="view-article" class="view"></div>
24
- <div class="match-overlay" id="match-overlay"><div class="mo-header"><h3>Chi tiết trận đấu</h3><button class="mo-close" onclick="closeMatch()">✕</button></div><div class="mo-tabs"><span class="mo-tab active" onclick="loadMatchTab('comm')">Diễn biến</span><span class="mo-tab" onclick="loadMatchTab('stats')">Thống kê</span></div><div class="mo-body" id="mo-body"><div class="loading">Đang tải...</div></div></div>
25
  <script>
26
- const SPACE=location.origin;let _cats=[],_tikData=[],_serverWall=[],_currentArticle=null,_currentEventId='',_lsTab='today';
 
27
  function escapeHtml(s){return String(s||'').replace(/[&<>"']/g,m=>({'&':'&amp;','<':'&lt;','>':'&gt;','"':'&quot;',"'":'&#39;'}[m]));}
28
- function fetchJsonTimeout(url,ms=8000){const c=new AbortController();const t=setTimeout(()=>c.abort(),ms);return fetch(url,{signal:c.signal}).then(r=>r.json()).finally(()=>clearTimeout(t));}
29
- function doShare(title,url,img){const u=SPACE+'/s?url='+encodeURIComponent(url)+'&title='+encodeURIComponent(title)+'&img='+encodeURIComponent(img||'');if(navigator.share)navigator.share({title,url:u}).catch(()=>{});else navigator.clipboard.writeText(u).then(()=>alert('Đã sao chép link!')).catch(()=>{});}
30
- function doShareVideo(title,url,img,type){const u=SPACE+'/v?url='+encodeURIComponent(url)+'&title='+encodeURIComponent(title)+'&img='+encodeURIComponent(img||'')+'&type='+(type||'highlights');if(navigator.share)navigator.share({title,url:u}).catch(()=>{});else navigator.clipboard.writeText(u).then(()=>alert('Đã sao chép link video!')).catch(()=>{});}
31
- async function init(){_cats=await fetchJsonTimeout('/api/categories',5000).catch(()=>[]);let bar='<div class="cat active" data-cat="home">🏠</div><div class="cat" data-cat="video">🎬 Video</div>';_cats.forEach(c=>bar+=`<div class="cat" data-cat="${c.id}">${c.name}</div>`);document.getElementById('cat-bar').innerHTML=bar;document.querySelectorAll('.cat').forEach(t=>t.onclick=()=>switchCat(t.dataset.cat));await loadHomeFast();let pv=localStorage.getItem('pending_video');if(pv){localStorage.removeItem('pending_video');try{pv=JSON.parse(pv);if(pv.url)openTikTokByUrl(pv.url,pv.type||'highlights')}catch(e){}}let pa=localStorage.getItem('pending_article');if(pa){localStorage.removeItem('pending_article');readArticle(pa)}}
32
- function switchCat(id){document.querySelectorAll('.cat').forEach(x=>x.classList.remove('active'));document.querySelector(`[data-cat="${id}"]`)?.classList.add('active');document.querySelectorAll('.view').forEach(x=>x.classList.remove('active'));document.querySelectorAll('video').forEach(v=>{try{v.pause()}catch(e){}});document.querySelectorAll('iframe[data-yt-src]').forEach(f=>f.src='');if(id==='home'){document.getElementById('view-home').classList.add('active');loadHomeFast()}else if(id==='video'){document.getElementById('view-video').classList.add('active');loadVideos()}else{document.getElementById('view-cat').classList.add('active');loadCat(id)}}
33
- function showView(id){document.querySelectorAll('.view').forEach(x=>x.classList.remove('active'));document.getElementById(id).classList.add('active')}
34
- async function loadHomeFast(){const home=document.getElementById('view-home');home.innerHTML='<div id="home-wall"></div><div id="home-ls"></div><div id="home-fast"></div><div id="home-lazy"></div><div class="loading" id="home-loadnote">Đang tải tin chính...</div>';loadServerWall();renderLivescoreShell();fetchJsonTimeout('/api/homepage',9000).then(n=>renderNews(n||[])).catch(()=>{let e=document.getElementById('home-loadnote');if(e)e.textContent='Không tải được tin chính'});setTimeout(loadLazySections,80);setTimeout(()=>loadLivescore('today'),150)}
35
- async function loadServerWall(){try{let j=await fetchJsonTimeout('/api/wall',5000);_serverWall=(j&&j.posts)||[];renderServerWall()}catch(e){_serverWall=[];renderServerWall()}}
36
- function renderServerWall(){let el=document.getElementById('home-wall');if(!el)return;if(!_serverWall.length){el.innerHTML='';return}el.innerHTML=renderWallSlide(_serverWall)}
37
- function renderWallSlide(posts){let h='<div class="slider-wrap"><div class="slider-header"><span class="slider-label">🧱 Tường AI</span><span class="slider-note">Lưu trên server</span></div><div class="slider-track">';posts.slice(0,30).forEach(p=>{h+=`<div class="wall-item"><div class="wall-thumb">${p.img?`<img loading="lazy" src="${p.img}">`:''}</div><div class="wall-title">${escapeHtml(p.title)}</div><div class="wall-text">${escapeHtml(p.text)}</div><div class="wall-actions"><button class="primary" onclick="readWallPost('${p.id}')">Xem</button>${p.url?`<button onclick="readArticle('${p.url.replace(/'/g,"\\'")}')">Gốc</button>`:''}</div></div>`});h+='</div></div>';return h}
38
- function readWallPost(id){let p=_serverWall.find(x=>x.id===id);if(!p)return;showView('view-article');document.getElementById('view-article').innerHTML=`<button class="back-btn" onclick="switchCat('home')">← Quay lại</button><div class="article-view"><span class="badge badge-ai">AI Rewrite</span><h1 class="article-title">${escapeHtml(p.title)}</h1>${p.img?`<img class="article-img" src="${p.img}">`:''}<div class="article-summary">Bài viết AI tóm tắt và viết lại</div><p class="article-p" style="white-space:pre-wrap">${escapeHtml(p.text)}</p><div class="article-actions"><button onclick="doShare('${escapeHtml(p.title)}','${p.url||location.href}','${p.img||''}')">📤 Chia sẻ</button>${p.url?`<button onclick="readArticle('${p.url.replace(/'/g,"\\'")}')">Đọc bài gốc</button>`:''}</div></div>`;window.scrollTo(0,0)}
39
- function renderLivescoreShell(){document.getElementById('home-ls').innerHTML=`<div class="ls-section"><div class="ls-header"><h3>⚽ Bóng Đá</h3></div><div class="ls-tabs"><span class="ls-tab" data-tab="live" onclick="loadLivescore('live')">🔴 Live</span><span class="ls-tab" data-tab="incoming" onclick="loadLivescore('incoming')">⏰ Sắp tới</span><span class="ls-tab" data-tab="results" onclick="loadLivescore('results')">✅ Kết quả</span><span class="ls-tab active" data-tab="today" onclick="loadLivescore('today')">📅 Hôm nay</span><span class="ls-tab" data-tab="bxh_nha" onclick="loadLivescore('bxh_nha')">🏆 NHA</span><span class="ls-tab" data-tab="bxh_laliga" onclick="loadLivescore('bxh_laliga')">🏆 La Liga</span><span class="ls-tab" data-tab="bxh_seriea" onclick="loadLivescore('bxh_seriea')">🏆 Serie A</span><span class="ls-tab" data-tab="bxh_bundesliga" onclick="loadLivescore('bxh_bundesliga')">🏆 Bundesliga</span><span class="ls-tab" data-tab="bxh_ligue1" onclick="loadLivescore('bxh_ligue1')">🏆 Ligue 1</span></div><div class="ls-content" id="ls-content"><div class="loading" style="padding:20px">Đang tải...</div></div></div>`}
40
- function bindMatchClicks(c){c.querySelectorAll('.match-detail').forEach(md=>md.onclick=e=>{e.preventDefault();let a=md.querySelector('.status a');let m=a&&(a.getAttribute('href')||'').match(/\/tran-dau\/(\d+)\//);if(m)openMatch(m[1])});c.querySelectorAll('a').forEach(a=>a.onclick=e=>e.preventDefault())}
41
- async function loadLivescore(tab){_lsTab=tab;document.querySelectorAll('.ls-tab').forEach(t=>t.classList.remove('active'));document.querySelector(`.ls-tab[data-tab="${tab}"]`)?.classList.add('active');let el=document.getElementById('ls-content');if(!el)return;el.innerHTML='<div class="loading" style="padding:20px">Đang tải...</div>';let ep='/api/livescore/'+tab;if(tab.startsWith('bxh_'))ep='/api/livescore/standings/'+tab.replace('bxh_','');try{let d=await fetchJsonTimeout(ep,9000);if(d.html&&d.html.length>50){el.innerHTML=d.html;bindMatchClicks(el)}else el.innerHTML='<div class="loading" style="padding:20px">Không có dữ liệu</div>'}catch(e){el.innerHTML='<div class="loading" style="padding:20px">Lỗi tải dữ liệu</div>'}}
42
- setInterval(()=>{if(_lsTab==='live')loadLivescore('live')},60000);
43
- function openMatch(id){if(!id)return;_currentEventId=id;document.getElementById('match-overlay').classList.add('active');document.body.style.overflow='hidden';loadMatchTab('comm')}
44
- function closeMatch(){document.getElementById('match-overlay').classList.remove('active');document.body.style.overflow=''}
45
- async function loadMatchTab(tab){document.querySelectorAll('.mo-tab').forEach(t=>t.classList.remove('active'));document.querySelectorAll('.mo-tab').forEach(t=>{if(tab==='comm'&&t.textContent==='Diễn biến')t.classList.add('active');if(tab==='stats'&&t.textContent==='Thống kê')t.classList.add('active')});let el=document.getElementById('mo-body');el.innerHTML='<div class="loading">Đang tải...</div>';try{let d=await fetchJsonTimeout(tab==='stats'?`/api/match/${_currentEventId}/stats`:`/api/match/${_currentEventId}/commentaries`,9000);el.innerHTML=d.html||'<div class="loading">Không có dữ liệu</div>'}catch(e){el.innerHTML='<div class="loading">Lỗi</div>'}}
46
- function renderNews(news){let note=document.getElementById('home-loadnote');if(note)note.remove();let h='';let groups={};(news||[]).forEach(a=>{if(!groups[a.group])groups[a.group]=[];groups[a.group].push(a)});for(const[g,arts]of Object.entries(groups)){h+=`<div class="section-title">${g}</div><div class="grid">`;arts.slice(0,6).forEach(a=>{let bg=a.source==='bbc'?'badge-bbc':a.source==='dantri'?'badge-dt':a.source==='genk'?'badge-genk':'badge-vne';let lb=a.source==='bbc'?'BBC':a.source==='dantri'?'DT':a.source==='genk'?'GenK':'VnE';h+=`<div class="card" onclick="readArticle('${a.link.replace(/'/g,"\\'")}','${a.source}')"><div class="card-img">${a.img?`<img loading="lazy" src="${a.img}">`:''}</div><div class="card-body"><span class="badge ${bg}">${lb}</span><div class="card-title">${a.title}</div></div></div>`});h+='</div>'}document.getElementById('home-fast').innerHTML=h||'<div class="loading">Không có dữ liệu</div>'}
47
- async function loadLazySections(){let lazy=document.getElementById('home-lazy');Promise.allSettled([fetchJsonTimeout('/api/worldcup2026',9000),fetchJsonTimeout('/api/dantri_hot',7000),fetchJsonTimeout('/api/shorts',9000),fetchJsonTimeout('/api/vne_video',7000),fetchJsonTimeout('/api/highlights',9000)]).then(res=>{let wc=res[0].value||[],dantri=res[1].value||[],shorts=res[2].value||[],vneVid=res[3].value||[],hl=res[4].value||[];let out='';if(wc.length){out+='<div class="slider-wrap"><div class="slider-header"><span class="slider-label">🏆 Đường đến World Cup 2026</span></div><div class="slider-track">';wc.slice(0,18).forEach(a=>out+=`<div class="slider-item" onclick="readArticle('${a.link.replace(/'/g,"\\'")}','ttvh')"><div class="slider-thumb">${a.img?`<img loading="lazy" src="${a.img}">`:''}</div><div class="slider-title">${a.title}</div></div>`);out+='</div></div>'}if(dantri.length){out+='<div class="slider-wrap"><div class="slider-header"><span class="slider-label">🔥 Tin Nổi Bật</span></div><div class="slider-track">';dantri.forEach(a=>out+=`<div class="slider-item" onclick="readArticle('${a.link.replace(/'/g,"\\'")}','dantri')"><div class="slider-thumb">${a.img?`<img loading="lazy" src="${a.img}">`:''}</div><div class="slider-title">${a.title}</div></div>`);out+='</div></div>'}let all=[...shorts,...vneVid];if(all.length){out+='<div class="slider-wrap"><div class="slider-header"><span class="slider-label">📱 Video · Shorts</span></div><div class="slider-track">';all.forEach((a,i)=>{if(a.source==='vne-video')out+=`<div class="slider-item" onclick="window.open('${a.link}','_blank')"><div class="slider-thumb">${a.img?`<img loading="lazy" src="${a.img}">`:''}<div class="card-play">▶</div></div><div class="slider-title">${a.title}</div></div>`;else out+=`<div class="slider-item shorts-item" onclick="openTikTok('shorts',${i})"><div class="slider-thumb shorts-thumb">${a.img?`<img loading="lazy" src="${a.img}">`:''}<div class="card-play">▶</div></div><div class="slider-title">${a.title}</div></div>`});out+='</div></div>'}if(hl.length){out+='<div class="slider-wrap"><div class="slider-header"><span class="slider-label">🎬 Highlight</span></div><div class="slider-track">';hl.slice(0,20).forEach((a,i)=>out+=`<div class="slider-item" onclick="openTikTok('highlights',${i})"><div class="slider-thumb">${a.img?`<img loading="lazy" src="${a.img}">`:''}<div class="card-play">▶</div></div><div class="slider-title">${a.title}</div></div>`);out+='</div></div>'}lazy.innerHTML=out})}
48
- async function loadCat(id){let el=document.getElementById('view-cat');el.innerHTML='<div class="loading">Đang tải...</div>';let arts=await fetchJsonTimeout('/api/category/'+id,9000).catch(()=>[]);if(!arts.length){el.innerHTML='<div class="loading">Không có tin</div>';return}let h='<div class="grid">';arts.forEach(a=>{let bg=a.source==='bbc'?'badge-bbc':a.source==='dantri'?'badge-dt':a.source==='genk'?'badge-genk':'badge-vne';h+=`<div class="card" onclick="readArticle('${a.link.replace(/'/g,"\\'")}','${a.source}')"><div class="card-img">${a.img?`<img loading="lazy" src="${a.img}">`:''}</div><div class="card-body"><span class="badge ${bg}">${a.source}</span><div class="card-title">${a.title}</div></div></div>`});h+='</div>';el.innerHTML=h}
49
- async function readArticle(url,source){let supported=url.includes('vnexpress.net')||url.includes('bbc.com')||url.includes('dantri.com.vn')||url.includes('genk.vn')||url.includes('thethaovanhoa.vn');if(!supported){window.open(url,'_blank');return}showView('view-article');let el=document.getElementById('view-article');el.innerHTML='<div class="loading">Đang tải...</div>';let data=await fetchJsonTimeout('/api/article?url='+encodeURIComponent(url),12000).catch(()=>null);if(!data||data.error||!data.body||!data.body.length){el.innerHTML=`<button class="back-btn" onclick="switchCat('home')">← Quay lại</button><div class="loading"><a href="${url}" target="_blank" style="color:#5cb87a">Mở link gốc</a></div>`;return}_currentArticle={url,data};let h=`<button class="back-btn" onclick="switchCat('home')">← Quay lại</button><div class="article-view"><h1 class="article-title">${data.title}</h1>`;if(data.summary)h+=`<div class="article-summary">${data.summary}</div>`;let last='';data.body.forEach(b=>{if(b.type==='p')h+=`<p class="article-p">${b.text}</p>`;else if(b.type==='img'&&b.src&&b.src!==last){last=b.src;h+=`<img loading="lazy" class="article-img" src="${b.src}">`}else if(b.type==='heading')h+=`<h2 class="article-h2">${b.text}</h2>`});h+=`<div class="article-actions"><select id="rewrite-tone"><option value="vui-ve">Vui vẻ</option><option value="nghiem-tuc">Nghiêm túc</option><option value="nghi-luan">Nghị luận</option><option value="hoi-dap">Hỏi đáp</option><option value="soi-noi">Sôi nổi</option><option value="thu-hut">Thu hút</option><option value="phan-tich">Phân tích</option><option value="chuyen-gia">Chuyên gia</option></select><button class="primary" onclick="rewriteCurrentArticle()">🤖 Rewrite AI</button><button onclick="doShare('${(data.title||'').replace(/'/g,"\\'")}','${url.replace(/'/g,"\\'")}','${(data.og_image||'').replace(/'/g,"\\'")}')">📤 Chia sẻ</button><button onclick="window.open('${url}','_blank')">🔗 Gốc</button></div><div id="rewrite-result"></div></div>`;el.innerHTML=h;window.scrollTo(0,0)}
50
- function rewriteCurrentArticle(){if(!_currentArticle)return;let tone=document.getElementById('rewrite-tone')?.value||'nghiem-tuc';let btn=document.querySelector('.article-actions button.primary');if(btn){btn.textContent='Đang rewrite...';btn.disabled=true}fetch('/api/rewrite_share',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({url:_currentArticle.url,tone})}).then(r=>r.json()).then(j=>{if(j&&j.post){document.getElementById('rewrite-result').innerHTML=`<div class="rewrite-box"><div class="rewrite-title">Đã rewrite đăng lên Tường AI</div><div class="rewrite-text">${escapeHtml(j.post.text||'')}</div></div>`;alert('Đã đăng lên Tường AI');loadServerWall()}else alert(j.error||'Không tạo được bài AI')}).catch(()=>alert('Lỗi tạo bài AI')).finally(()=>{if(btn){btn.textContent='🤖 Rewrite AI';btn.disabled=false}})}
51
- async function loadVideos(){let el=document.getElementById('view-video');if(el.dataset.loaded)return;el.innerHTML='<div class="loading">Đang tải...</div>';let[hl,bdp]=await Promise.all([fetchJsonTimeout('/api/highlights',9000).catch(()=>[]),fetchJsonTimeout('/api/bdp_videos',9000).catch(()=>[])]);let h='<div class="section-title">🎬 Highlight</div><div class="grid">';hl.forEach((a,i)=>h+=`<div class="card" onclick="openTikTok('highlights',${i})"><div class="card-img">${a.img?`<img loading="lazy" src="${a.img}">`:''}<div class="card-play">▶</div></div><div class="card-body"><span class="badge badge-fpt">HL</span><div class="card-title">${a.title}</div></div></div>`);h+='</div>';if(bdp.length){h+='<div class="section-title">⚽ BDP</div><div class="grid">';bdp.forEach((a,i)=>h+=`<div class="card" onclick="openTikTok('bdp',${i})"><div class="card-img">${a.img?`<img loading="lazy" src="${a.img}">`:''}<div class="card-play">▶</div></div><div class="card-body"><span class="badge badge-fpt">BDP</span><div class="card-title">${a.title}</div></div></div>`);h+='</div>'}el.innerHTML=h;el.dataset.loaded='1'}
52
- async function openTikTokByUrl(targetUrl,type){showView('view-tiktok');document.querySelectorAll('.cat').forEach(x=>x.classList.remove('active'));let el=document.getElementById('view-tiktok');el.innerHTML='<div class="loading">Đang tải video...</div>';let articles=type==='shorts'?await fetchJsonTimeout('/api/shorts',9000).catch(()=>[]):type==='highlights'?await fetchJsonTimeout('/api/highlights',9000).catch(()=>[]):await fetchJsonTimeout('/api/bdp_videos',9000).catch(()=>[]);let startIdx=articles.findIndex(a=>a.link===targetUrl);await buildTikTokPlayer(articles,startIdx<0?0:startIdx,type)}
53
- async function openTikTok(type,startIdx){showView('view-tiktok');document.querySelectorAll('.cat').forEach(x=>x.classList.remove('active'));let el=document.getElementById('view-tiktok');el.innerHTML='<div class="loading">Đang tải video...</div>';let articles=type==='shorts'?await fetchJsonTimeout('/api/shorts',9000).catch(()=>[]):type==='highlights'?await fetchJsonTimeout('/api/highlights',9000).catch(()=>[]):await fetchJsonTimeout('/api/bdp_videos',9000).catch(()=>[]);await buildTikTokPlayer(articles,startIdx,type)}
54
- async function buildTikTokPlayer(articles,startIdx,type){let el=document.getElementById('view-tiktok');let vids=[];let res=await Promise.all(articles.map(async(a,i)=>{try{let v=await fetch('/api/video_url?url='+encodeURIComponent(a.link)).then(r=>r.json());if(v&&v.src)return{article:a,video:v,idx:i}}catch(e){}return null}));res.forEach(r=>{if(r)vids.push({...r.article,...r.video,_idx:r.idx})});vids.sort((a,b)=>a._idx-b._idx);if(!vids.length){el.innerHTML='<div class="loading">Không tìm thấy video</div>';return}let ti=vids.findIndex(v=>v._idx===startIdx);if(ti<0)ti=0;let ordered=ti>0?[...vids.slice(ti),...vids.slice(0,ti)]:vids;_tikData=ordered;let h=`<button class="back-btn" onclick="switchCat('home')">← Quay lại</button><div class="tiktok-container"><div class="tiktok-feed" id="tiktok-feed">`;ordered.forEach((v,i)=>{let isYT=v.type==='youtube';let isHLS=!isYT&&v.src&&v.src.includes('.m3u8');let poster=v.poster?` poster="${v.poster}"`:'';let tag=isYT?`<iframe data-yt-src="${v.src}" allowfullscreen allow="accelerometer;autoplay;clipboard-write;encrypted-media;gyroscope;picture-in-picture"></iframe>`:isHLS?`<video playsinline preload="none"${poster} data-hls="${v.src}" loop controls></video>`:`<video playsinline preload="none"${poster} loop controls><source src="${v.src}" type="video/mp4"></video>`;h+=`<div class="tiktok-slide">${tag}<div class="tiktok-bottom"><span class="badge badge-fpt">VIDEO</span><p class="tiktok-title">${v.title}</p></div><div class="tiktok-right"><button class="tiktok-right-btn" onclick="event.stopPropagation();shareVid(${i})"><div class="icon">📤</div></button></div><span class="tiktok-counter">${i+1}/${ordered.length}</span></div>`});h+='</div></div>';el.innerHTML=h;initFeed()}
55
- function shareVid(i){let v=_tikData[i];if(v)doShareVideo(v.title,v.link||'',v.poster||v.img||'','highlights')}
56
- function initFeed(){let feed=document.getElementById('tiktok-feed');if(!feed)return;let slides=feed.querySelectorAll('.tiktok-slide');let cur=-1;function activate(idx){if(idx===cur)return;slides.forEach((sl,i)=>{let v=sl.querySelector('video'),f=sl.querySelector('iframe');if(i===idx){if(v&&v.dataset.hls){if(!v._hls){let hls=new Hls();hls.loadSource(v.dataset.hls);hls.attachMedia(v);hls.on(Hls.Events.MANIFEST_PARSED,()=>v.play().catch(()=>{}));v._hls=hls}else v.play().catch(()=>{})}else if(v)v.play().catch(()=>{});if(f&&!f.src&&f.dataset.ytSrc)f.src=f.dataset.ytSrc}else{if(v){v.pause();if(v._hls){v._hls.destroy();v._hls=null}}if(f&&f.src)f.src=''}});cur=idx}let sT;feed.addEventListener('scroll',()=>{clearTimeout(sT);sT=setTimeout(()=>{let rect=feed.getBoundingClientRect(),ctr=rect.top+rect.height/2,best=-1,bestD=1e9;slides.forEach((sl,i)=>{let d=Math.abs(sl.getBoundingClientRect().top+sl.getBoundingClientRect().height/2-ctr);if(d<bestD){bestD=d;best=i}});if(best>=0)activate(best)},150)});setTimeout(()=>activate(0),300)}
 
 
 
 
 
 
 
 
 
 
 
 
57
  init();
58
  </script>
59
  </body>
 
4
  <meta charset="utf-8">
5
  <meta name="viewport" content="width=device-width,initial-scale=1,maximum-scale=1">
6
  <title>VNEWS - Tin Tức Việt Nam</title>
7
+ <meta name="description" content="Tin tức tổng hợp, bóng đá trực tiếp, video highlight.">
8
  <meta property="og:title" content="VNEWS - Tin Tức Việt Nam">
9
  <meta property="og:image" content="https://s1.vnecdn.net/vnexpress/restruct/i/v9505/logo_default.jpg">
10
  <link rel="canonical" href="https://bep40-vnews.hf.space">
11
  <script src="https://cdn.jsdelivr.net/npm/hls.js@1/dist/hls.min.js"></script>
12
  <style>
13
+ *{box-sizing:border-box;margin:0;padding:0}body{background:#111;color:#eee;font-family:-apple-system,BlinkMacSystemFont,'Segoe UI',Roboto,sans-serif;overflow-x:hidden}.header{background:linear-gradient(135deg,#0d1117,#1a3a2a 50%,#8b7500);padding:12px;text-align:center}.header h1{color:#fff;font-size:18px;font-weight:800}.header p{color:rgba(255,255,255,.6);font-size:10px;margin-top:2px}.cats{display:flex;overflow-x:auto;background:#1a1a1a;border-bottom:1px solid #333;padding:0 4px;scrollbar-width:none;position:sticky;top:0;z-index:50}.cats::-webkit-scrollbar{display:none}.cat{padding:9px 11px;color:#888;font-size:11px;cursor:pointer;white-space:nowrap;border-bottom:2px solid transparent;flex-shrink:0}.cat.active{color:#5cb87a;border-bottom-color:#5cb87a;font-weight:600}.view{display:none}.view.active{display:block}.loading{text-align:center;padding:40px;color:#666;font-size:12px}.grid{display:grid;grid-template-columns:repeat(2,1fr);gap:6px;padding:6px 4px}@media(min-width:650px){.grid{grid-template-columns:repeat(3,1fr)}}.card{background:#1a1a1a;border-radius:8px;overflow:hidden;cursor:pointer;border:1px solid #222}.card:active{transform:scale(.98)}.card-img{position:relative;width:100%;aspect-ratio:16/9;background:#333;overflow:hidden}.card-img img{width:100%;height:100%;object-fit:cover}.card-play{position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);width:30px;height:30px;background:rgba(0,0,0,.55);border-radius:50%;display:flex;align-items:center;justify-content:center;color:#fff;font-size:11px}.card-body{padding:6px 8px}.card-title{font-size:11px;font-weight:600;color:#eee;line-height:1.35;display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden}.badge{font-size:8px;padding:1px 5px;border-radius:3px;font-weight:700;display:inline-block;margin-bottom:2px}.badge-vne{background:#c0392b;color:#fff}.badge-bbc{background:#b80000;color:#fff}.badge-dt{background:#1565c0;color:#fff}.badge-genk{background:#6a1b9a;color:#fff}.badge-fpt{background:#f26522;color:#fff}.slider-wrap{margin:6px 4px;background:#1a1a1a;border-radius:8px;overflow:hidden;border:1px solid #2a2a2a}.slider-header{padding:7px 10px;display:flex;align-items:center;justify-content:space-between}.slider-label{color:#f0c040;font-size:13px;font-weight:700}.slider-track{display:flex;overflow-x:auto;gap:8px;padding:4px 10px 10px;scrollbar-width:none}.slider-track::-webkit-scrollbar{display:none}.slider-item{flex:0 0 160px;cursor:pointer;flex-shrink:0}.slider-thumb{position:relative;width:100%;aspect-ratio:16/9;border-radius:6px;overflow:hidden;background:#333}.slider-thumb img{width:100%;height:100%;object-fit:cover}.slider-title{color:#ccc;font-size:10px;margin-top:3px;line-height:1.25;display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden}.shorts-item{flex:0 0 110px!important}.shorts-thumb{aspect-ratio:3/4!important;border-radius:8px!important}.section-title{font-size:13px;font-weight:700;color:#5cb87a;margin:8px 0 4px;padding-left:8px;border-left:3px solid #5cb87a}.back-btn{background:rgba(17,17,17,.95);color:#fff;border:none;padding:10px;font-size:12px;cursor:pointer;width:100%;text-align:center;position:sticky;top:0;z-index:50}.article-view{padding:12px 8px 40px;max-width:760px;margin:0 auto}.article-title{font-size:18px;font-weight:800;color:#f0f0f0;line-height:1.32;margin-bottom:8px}.article-summary{background:#1a2a1f;border-left:3px solid #2d8659;padding:10px;margin-bottom:14px;font-weight:600;color:#ccc;font-size:13px;border-radius:0 6px 6px 0}.article-p{font-size:14px;line-height:1.72;color:#ccc;margin-bottom:10px}.article-img{width:100%;border-radius:6px;margin:10px 0}.article-h2{font-size:16px;font-weight:700;color:#eee;margin:16px 0 8px}.article-actions{display:flex;gap:8px;padding:10px 0;border-top:1px solid #333;margin-top:16px;flex-wrap:wrap}.article-actions button{background:#1a1a1a;border:1px solid #333;color:#ccc;padding:7px 12px;border-radius:14px;font-size:11px;cursor:pointer}.featured-match{margin:6px 4px;background:linear-gradient(135deg,#1a2a1f,#0d1117);border-radius:10px;padding:12px;border:1px solid #2d8659;cursor:pointer}.fm-league{color:#5cb87a;font-size:9px;font-weight:600;text-transform:uppercase;margin-bottom:6px;text-align:center}.fm-teams{display:flex;align-items:center;justify-content:center;gap:10px}.fm-team{display:flex;flex-direction:column;align-items:center;gap:4px;flex:1}.fm-team img{width:32px;height:32px;object-fit:contain}.fm-team span{font-size:10px;color:#ccc;text-align:center}.fm-score{font-size:22px;font-weight:900;color:#fff;min-width:60px;text-align:center}.fm-status{text-align:center;margin-top:6px;font-size:9px;color:#e74c3c;font-weight:600}.fm-status.upcoming{color:#f0c040}.ls-section{margin:6px 4px;background:#1a1a1a;border-radius:8px;overflow:hidden;border:1px solid #2a2a2a}.ls-header{padding:8px 10px}.ls-header h3{color:#fff;font-size:13px;font-weight:700}.ls-tabs{display:flex;gap:4px;padding:0 10px 8px;overflow-x:auto;scrollbar-width:none}.ls-tabs::-webkit-scrollbar{display:none}.ls-tab{padding:4px 10px;background:#222;border:1px solid #333;border-radius:12px;color:#999;font-size:10px;cursor:pointer;white-space:nowrap;flex-shrink:0}.ls-tab.active{background:#2d8659;border-color:#2d8659;color:#fff;font-weight:600}.ls-content{max-height:420px;overflow-y:auto;padding:0 6px 8px;scrollbar-width:thin}.ls-content .matchs-league{list-style:none}.ls-content .matchs-league>li{margin-bottom:8px}.ls-content .title-content{display:flex;align-items:center;gap:6px;padding:4px 6px;background:#222;border-radius:4px;margin-bottom:4px}.ls-content .title-content img{width:16px;height:16px}.ls-content .title-content strong{color:#eee;font-size:11px}.ls-content .match-detail{padding:5px 6px;border-bottom:1px solid #222;cursor:pointer}.ls-content .match-detail:active{background:#222}.ls-content .match{display:flex;flex-wrap:wrap;align-items:center;gap:4px}.ls-content .datetime{width:100%;display:flex;gap:8px;margin-bottom:2px}.ls-content .datetime .round{color:#666;font-size:9px}.ls-content .datetime .match-time{color:#aaa;font-size:9px}.ls-content .teams{display:flex;align-items:center;gap:4px;width:100%}.ls-content .team{display:flex;align-items:center;gap:4px;flex:1;min-width:0;text-decoration:none}.ls-content .team .name{font-size:11px;color:#ddd;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.ls-content .team .logo img{width:18px;height:18px;object-fit:contain}.ls-content .home-team{justify-content:flex-end;text-align:right}.ls-content .home-team .name{order:-1}.ls-content .away-team{justify-content:flex-start}.ls-content .status{flex:0 0 50px;text-align:center}.ls-content .status a{text-decoration:none;color:#fff;font-weight:700;font-size:12px}.ls-content .status .label{font-size:8px;display:block;color:#888}.ls-content .status .label.live{color:#e74c3c;font-weight:700}.ls-content .status .vs{color:#888;font-size:11px}.ls-content .status .blink{animation:blink 1s infinite}@keyframes blink{50%{opacity:0}}.ls-content .info,.ls-content .btns{display:none}.ls-content .leaderboard-item{display:flex;align-items:center;padding:5px 6px;border-bottom:1px solid #222;gap:4px}.ls-content .leaderboard-item .team{display:flex;align-items:center;gap:5px;flex:1;min-width:0}.ls-content .leaderboard-item .team span{color:#888;font-size:10px;min-width:16px}.ls-content .leaderboard-item .team .link{display:flex;align-items:center;gap:4px;text-decoration:none}.ls-content .leaderboard-item .team p{color:#ddd;font-size:11px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.ls-content .leaderboard-item .team img{width:16px;height:16px}.ls-content .leaderboard-item .copy{display:flex;gap:6px;font-size:10px;color:#aaa}.ls-content .leaderboard-item .copy p{min-width:16px;text-align:center}.ls-content .leaderboard-item .copy strong{color:#5cb87a}.ls-content .leaderboard-item .copy .form{display:flex;gap:2px}.ls-content .leaderboard-item .copy .form span{width:12px;height:12px;border-radius:50%;display:flex;align-items:center;justify-content:center;font-size:7px;color:#fff;font-weight:700}.ls-content .bg-green{background:#27ae60}.ls-content .bg-red{background:#e74c3c}.ls-content .bg-gray-5{background:#666}.match-overlay{position:fixed;top:0;left:0;right:0;bottom:0;background:#111;z-index:9999;display:none;flex-direction:column;overflow-y:auto}.match-overlay.active{display:flex}.mo-header{padding:10px;background:#1a1a1a;display:flex;justify-content:space-between;align-items:center;position:sticky;top:0;z-index:1}.mo-header h3{color:#fff;font-size:13px}.mo-close{background:none;border:none;color:#fff;font-size:22px;cursor:pointer}.mo-tabs{display:flex;gap:4px;padding:8px 10px;background:#1a1a1a}.mo-tab{padding:5px 12px;background:#222;border:1px solid #333;border-radius:10px;color:#999;font-size:10px;cursor:pointer}.mo-tab.active{background:#2d8659;border-color:#2d8659;color:#fff}.mo-body{padding:8px;flex:1;overflow-x:auto}.mo-body .time{color:#5cb87a;font-weight:700;font-size:11px;margin-right:6px}.mo-body .description{color:#ccc;font-size:12px;line-height:1.4}.mo-body li{list-style:none;padding:6px 0;border-bottom:1px solid #222}.mo-body .professional-table{width:100%;font-size:10px;border-collapse:collapse;min-width:500px}.mo-body .professional-table th{color:#888;padding:6px 4px;text-align:center;border-bottom:1px solid #333;white-space:nowrap}.mo-body .professional-table th.sticky-column{text-align:left;position:sticky;left:0;background:#111;z-index:1;min-width:120px}.mo-body .professional-table td{color:#ccc;padding:6px 4px;border-bottom:1px solid #1a1a1a;text-align:center;font-size:11px}.mo-body .professional-table td.sticky-column{text-align:left;position:sticky;left:0;background:#111;z-index:1}.mo-body .player-info{display:flex;align-items:center;gap:6px}.mo-body .player-logo{width:18px;height:18px;border-radius:50%;object-fit:contain}.mo-body .player-name{color:#ddd;text-decoration:none;font-size:11px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;max-width:100px}.mo-body .score{font-weight:700;border-radius:4px;padding:2px 5px;color:#fff}.mo-body .bg-blue{background:#2563eb}.mo-body .bg-green{background:#16a34a}.mo-body .bg-orange{background:#ea580c}.vp-wrap{background:#000;width:100%;aspect-ratio:1/1;position:relative;overflow:hidden;transition:aspect-ratio .3s}.vp-wrap.wide{aspect-ratio:16/9}.vp-wrap video,.vp-wrap iframe{object-fit:cover!important;width:100%;height:100%;border:none}.vp-ratio-btn{position:absolute;top:8px;right:8px;background:rgba(0,0,0,.6);border:1px solid rgba(255,255,255,.3);color:#fff;font-size:9px;padding:3px 8px;border-radius:12px;cursor:pointer;z-index:5}.vp-title{padding:8px 10px;font-size:13px;font-weight:700;color:#fff;background:#1a1a1a;border-bottom:1px solid #222}.pl-list{padding:4px}.pl-item{display:flex;gap:8px;padding:8px 6px;border-bottom:1px solid #1a1a1a;cursor:pointer;border-radius:6px}.pl-item.active{background:#1a2a1f;border:1px solid #2d8659}.pl-item-thumb{flex:0 0 100px;aspect-ratio:16/9;border-radius:4px;overflow:hidden;background:#333;position:relative}.pl-item-thumb img{width:100%;height:100%;object-fit:cover}.pl-item-body{flex:1;display:flex;align-items:center;min-width:0}.pl-item-title{font-size:11px;color:#ccc;line-height:1.3;display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden}.pl-item.active .pl-item-title{color:#5cb87a;font-weight:600}.msv{position:fixed;inset:0;background:#000;z-index:99999;overflow:hidden;color:#fff}.msv-stage{position:absolute;inset:0;display:flex;align-items:center;justify-content:center;background:#000}.msv-bg{position:absolute;inset:-30px;background:center/cover no-repeat;filter:blur(28px);transform:scale(1.08);opacity:.28}.msv-phone{position:relative;z-index:1;width:max(100vw,56.25dvh);height:max(100dvh,177.7778vw);max-width:none;max-height:none;aspect-ratio:9/16;background:#000;overflow:hidden}.msv-phone iframe{position:absolute;inset:0;width:100%!important;height:100%!important;border:0;background:#000}.msv-close{position:fixed;top:max(10px,env(safe-area-inset-top));right:max(10px,env(safe-area-inset-right));z-index:8;width:38px;height:38px;border-radius:50%;border:1px solid rgba(255,255,255,.16);background:rgba(0,0,0,.55);color:#fff;font-size:22px}.msv-info{position:fixed;left:12px;right:58px;bottom:max(18px,env(safe-area-inset-bottom));z-index:6;text-shadow:0 1px 7px #000;pointer-events:none}.msv-count{font-size:11px;color:#6ee78f;font-weight:700;margin-bottom:5px}.msv-title{font-size:13px;line-height:1.35;font-weight:700;display:-webkit-box;-webkit-line-clamp:2;-webkit-box-orient:vertical;overflow:hidden}.msv-nav{position:fixed;right:10px;top:50%;transform:translateY(-50%);z-index:7;display:flex;flex-direction:column;gap:10px}.msv-nav button{width:38px;height:38px;border:0;border-radius:50%;background:rgba(255,255,255,.16);color:#fff;font-size:16px}.msv-actions{position:fixed;right:10px;bottom:max(76px,env(safe-area-inset-bottom));z-index:7;display:flex;flex-direction:column;gap:10px}.msv-act{width:42px;min-height:42px;border:0;border-radius:22px;background:rgba(255,255,255,.14);color:#fff;font-size:17px;display:flex;flex-direction:column;align-items:center;justify-content:center;gap:1px}.msv-act span{font-size:9px;color:#ddd}.msv-act.liked{background:rgba(231,76,60,.9);color:#fff}.msv-comments{position:fixed;left:8px;right:8px;bottom:0;max-height:48vh;background:rgba(17,17,17,.96);border:1px solid #333;border-radius:12px 12px 0 0;z-index:9;transform:translateY(110%);transition:transform .22s;padding:10px}.msv-comments.show{transform:translateY(0)}.msv-ctitle{font-size:12px;font-weight:700;color:#fff;margin-bottom:8px}.msv-clist{max-height:32vh;overflow:auto}.msv-citem{font-size:11px;color:#ccc;border-bottom:1px solid #222;padding:6px 0}.msv-crow{display:flex;gap:6px;margin-top:8px}.msv-crow input{flex:1;background:#222;border:1px solid #333;color:#fff;border-radius:16px;padding:8px 10px;font-size:12px}.msv-crow button{background:#2d8659;border:0;color:#fff;border-radius:16px;padding:8px 12px;font-size:12px}.msv-swipe{position:absolute;inset:0;z-index:3;background:transparent}.ms-lock{overflow:hidden!important}
14
  </style>
15
  </head>
16
  <body>
17
+ <div class="header"><h1>📰 VNEWS</h1><p>Tin tức · Bóng đá trực tiếp · Shorts</p></div>
18
  <div class="cats" id="cat-bar"></div>
19
  <div id="view-home" class="view active"><div class="loading">Đang tải...</div></div>
20
  <div id="view-cat" class="view"></div>
21
  <div id="view-video" class="view"></div>
22
  <div id="view-tiktok" class="view"></div>
23
  <div id="view-article" class="view"></div>
24
+ <div class="match-overlay" id="match-overlay"><div class="mo-header"><h3 id="mo-title">Chi tiết trận đấu</h3><button class="mo-close" onclick="closeMatch()">✕</button></div><div class="mo-tabs"><span class="mo-tab active" onclick="loadMatchTab('comm')">Diễn biến</span><span class="mo-tab" onclick="loadMatchTab('stats')">Thống kê</span></div><div class="mo-body" id="mo-body"><div class="loading">Đang tải...</div></div></div>
25
  <script>
26
+ const SPACE=location.origin;let _cats=[],_tikData=[],_currentEventId='',_featuredEventId='',_hlLeagueData={},_vpHls=null,_shortsData=[],_shortStats={},_shortsPlayer=null,_ytLoading=false,_lsTab='today';
27
+ const HL_CONFIG={"premier-league":{name:"Premier League",emoji:"🏴"},"fa-cup":{name:"FA Cup",emoji:"🏆"},"champions-league":{name:"Champions League",emoji:"⭐"},"europa-league":{name:"Europa League",emoji:"🟠"},"la-liga":{name:"La Liga",emoji:"🇪🇸"},"serie-a":{name:"Serie A",emoji:"🇮🇹"},"bundesliga":{name:"Bundesliga",emoji:"🇩🇪"},"world-cup":{name:"World Cup 2026",emoji:"🌍"}};
28
  function escapeHtml(s){return String(s||'').replace(/[&<>"']/g,m=>({'&':'&amp;','<':'&lt;','>':'&gt;','"':'&quot;',"'":'&#39;'}[m]));}
29
+ function fetchJsonTimeout(url,ms=9000){const c=new AbortController();const t=setTimeout(()=>c.abort(),ms);return fetch(url,{signal:c.signal}).then(r=>r.json()).finally(()=>clearTimeout(t));}
30
+ function doShareVideo(title,url,img,type){const shareUrl=SPACE+'/v?url='+encodeURIComponent(url)+'&title='+encodeURIComponent(title)+'&img='+encodeURIComponent(img||'')+'&type='+encodeURIComponent(type||'highlights');if(navigator.share)navigator.share({title:'🎬 '+title,url:shareUrl}).catch(()=>{});else navigator.clipboard.writeText(shareUrl).then(()=>alert('Đã sao chép link!')).catch(()=>{});}
31
+ function doShare(title,url,img){const shareUrl=SPACE+'/s?url='+encodeURIComponent(url)+'&title='+encodeURIComponent(title)+'&img='+encodeURIComponent(img||'');if(navigator.share)navigator.share({title,url:shareUrl}).catch(()=>{});else navigator.clipboard.writeText(shareUrl).then(()=>alert('Đã sao chép!')).catch(()=>{});}
32
+ async function init(){_cats=await fetchJsonTimeout('/api/categories',5000).catch(()=>[]);let bar='<div class="cat active" data-cat="home">🏠</div><div class="cat" data-cat="video">🎬 Video</div>';_cats.forEach(c=>{bar+=`<div class="cat" data-cat="${c.id}">${c.name}</div>`;});document.getElementById('cat-bar').innerHTML=bar;document.querySelectorAll('.cat').forEach(t=>{t.onclick=()=>switchCat(t.dataset.cat);});await loadHome();const pending=localStorage.getItem('pending_article');if(pending){localStorage.removeItem('pending_article');readArticle(pending);}}
33
+ function switchCat(id){document.querySelectorAll('.cat').forEach(x=>x.classList.remove('active'));document.querySelector(`[data-cat="${id}"]`)?.classList.add('active');document.querySelectorAll('.view').forEach(x=>x.classList.remove('active'));document.querySelectorAll('video').forEach(v=>{try{v.pause();if(v._hls){v._hls.destroy();v._hls=null;}}catch(e){}});if(_vpHls){_vpHls.destroy();_vpHls=null;}document.querySelectorAll('iframe[data-yt-src]').forEach(f=>{f.src='';});if(id==='home')document.getElementById('view-home').classList.add('active');else if(id==='video'){document.getElementById('view-video').classList.add('active');loadVideos();}else{document.getElementById('view-cat').classList.add('active');loadCat(id);}}
34
+ function showView(id){document.querySelectorAll('.view').forEach(x=>x.classList.remove('active'));document.getElementById(id).classList.add('active');}
35
+ async function loadHome(){const home=document.getElementById('view-home');home.innerHTML='<div id="home-fast"></div><div id="home-lazy"></div><div class="loading" id="home-loadnote">Đang tải...</div>';fetchJsonTimeout('/api/homepage',9000).then(news=>renderNews(news||[])).catch(()=>{document.getElementById('home-loadnote').textContent='Không tải được tin chính';});setTimeout(loadLazySections,80);}
36
+ function renderNews(news){let note=document.getElementById('home-loadnote');if(note)note.remove();let h='';const groups={};news.forEach(a=>{if(!groups[a.group])groups[a.group]=[];groups[a.group].push(a);});for(const[g,arts] of Object.entries(groups)){h+=`<div class="section-title">${g}</div><div class="grid">`;arts.slice(0,6).forEach(a=>{const bg=a.source==='bbc'?'badge-bbc':a.source==='dantri'?'badge-dt':a.source==='genk'?'badge-genk':'badge-vne';const lb=a.source==='bbc'?'BBC':a.source==='dantri'?'DT':a.source==='genk'?'GenK':'VnE';h+=`<div class="card" onclick="readArticle('${a.link.replace(/'/g,"\\'")}','${a.source}')"><div class="card-img">${a.img?`<img loading="lazy" src="${a.img}">`:''}</div><div class="card-body"><span class="badge ${bg}">${lb}</span><div class="card-title">${a.title}</div></div></div>`;});h+='</div>';}document.getElementById('home-fast').innerHTML=h||'<div class="loading">Không có dữ liệu</div>';}
37
+ async function loadLazySections(){const lazy=document.getElementById('home-lazy');let h='';try{let featured=await fetchJsonTimeout('/api/livescore/featured',5000);if(featured&&featured.home){_featuredEventId=featured.event_id;const sc=featured.status==='live'?'':'upcoming';const st=featured.status==='live'?`🔴 ${featured.minute||'LIVE'}`:`⏰ ${featured.time}`;h+=`<div class="featured-match" onclick="openMatch('${featured.event_id}')"><div class="fm-league">${featured.league}</div><div class="fm-teams"><div class="fm-team"><img src="${featured.home_logo}" onerror="this.style.display='none'"><span>${featured.home}</span></div><div class="fm-score">${featured.score||'VS'}</div><div class="fm-team"><img src="${featured.away_logo}" onerror="this.style.display='none'"><span>${featured.away}</span></div></div><div class="fm-status ${sc}">${st}</div></div>`;}}catch(e){}
38
+ h+=`<div class="ls-section"><div class="ls-header"><h3>⚽ Bóng Đá</h3></div><div class="ls-tabs"><span class="ls-tab" data-tab="live" onclick="loadLivescore('live')">🔴 Live</span><span class="ls-tab" data-tab="incoming" onclick="loadLivescore('incoming')"> Sắp tới</span><span class="ls-tab" data-tab="results" onclick="loadLivescore('results')">✅ Kết quả</span><span class="ls-tab active" data-tab="today" onclick="loadLivescore('today')">📅 Hôm nay</span><span class="ls-tab" data-tab="bxh_nha" onclick="loadLivescore('bxh_nha')">🏆 NHA</span><span class="ls-tab" data-tab="bxh_laliga" onclick="loadLivescore('bxh_laliga')">🏆 La Liga</span><span class="ls-tab" data-tab="bxh_seriea" onclick="loadLivescore('bxh_seriea')">🏆 Serie A</span><span class="ls-tab" data-tab="bxh_bundesliga" onclick="loadLivescore('bxh_bundesliga')">🏆 Bundesliga</span><span class="ls-tab" data-tab="bxh_ligue1" onclick="loadLivescore('bxh_ligue1')">🏆 Ligue 1</span></div><div class="ls-content" id="ls-content"><div class="loading" style="padding:20px">Đang tải...</div></div></div>`;
39
+ lazy.innerHTML=h;loadLivescore('today');checkAndSwitchToLive();loadShortsAsync();loadWorldCupRoadAsync();loadHighlightsLeaguesAsync();loadDantriHotAsync();}
40
+ function loadDantriHotAsync(){fetchJsonTimeout('/api/dantri_hot',7000).then(dantri=>{if(!dantri||!dantri.length)return;let wrap=document.createElement('div');wrap.className='slider-wrap';let h='<div class="slider-header"><span class="slider-label">🔥 Tin Nổi Bật</span></div><div class="slider-track">';dantri.forEach(a=>{h+=`<div class="slider-item" onclick="readArticle('${a.link.replace(/'/g,"\\'")}','dantri')"><div class="slider-thumb">${a.img?`<img loading="lazy" src="${a.img}">`:''}</div><div class="slider-title">${a.title}</div></div>`;});h+='</div>';wrap.innerHTML=h;document.getElementById('home-lazy').prepend(wrap);}).catch(()=>{});}
41
+ function loadShortsAsync(){fetchJsonTimeout('/api/shorts',12000).then(sh=>{if(!sh||!sh.length)return;_shortsData=sh;let wrap=document.createElement('div');wrap.className='slider-wrap';let h='<div class="slider-header"><span class="slider-label">📱 Shorts · Dân Trí & Sức Khỏe</span></div><div class="slider-track">';sh.forEach((a,i)=>{h+=`<div class="slider-item shorts-item" onclick="openShortsViewer(${i})"><div class="slider-thumb shorts-thumb">${a.img?`<img loading="lazy" src="${a.img}">`:''}<div class="card-play">▶</div></div><div class="slider-title">${a.title}</div></div>`;});h+='</div>';wrap.innerHTML=h;document.getElementById('home-lazy').appendChild(wrap);}).catch(()=>{});}
42
+ function loadWorldCupRoadAsync(){fetchJsonTimeout('/api/worldcup2026',12000).then(wc=>{if(!wc||!wc.length)return;let wrap=document.createElement('div');wrap.className='slider-wrap';let h='<div class="slider-header"><span class="slider-label">🏆 Đường đến World Cup 2026</span></div><div class="slider-track">';wc.forEach(a=>{h+=`<div class="slider-item" onclick="readArticle('${a.link.replace(/'/g,"\\'")}','ttvh')"><div class="slider-thumb">${a.img?`<img loading="lazy" src="${a.img}" onerror="this.parentElement.style.background='#222'">`:''}</div><div class="slider-title">${a.title}</div></div>`;});h+='</div>';wrap.innerHTML=h;document.getElementById('home-lazy').appendChild(wrap);}).catch(()=>{});}
43
+ function loadHighlightsLeaguesAsync(){fetchJsonTimeout('/api/highlights/leagues',12000).then(hl=>{_hlLeagueData=hl||{};for(const[key,cfg] of Object.entries(HL_CONFIG)){const vids=_hlLeagueData[key];if(!vids||!vids.length)continue;let wrap=document.createElement('div');wrap.className='slider-wrap';let h=`<div class="slider-header"><span class="slider-label">${cfg.emoji} ${cfg.name}</span></div><div class="slider-track">`;vids.slice(0,8).forEach((a,i)=>{h+=`<div class="slider-item" onclick="openLeaguePlayer('${key}',${i})"><div class="slider-thumb">${a.img?`<img loading="lazy" src="${a.img}">`:''}<div class="card-play">▶</div></div><div class="slider-title">${a.title}</div></div>`;});h+='</div>';wrap.innerHTML=h;document.getElementById('home-lazy').appendChild(wrap);}}).catch(()=>{});}
44
+ async function loadLivescore(tab){_lsTab=tab;document.querySelectorAll('.ls-tab').forEach(t=>t.classList.remove('active'));document.querySelector(`.ls-tab[data-tab="${tab}"]`)?.classList.add('active');const el=document.getElementById('ls-content');if(!el)return;el.innerHTML='<div class="loading" style="padding:20px">Đang tải...</div>';let ep='/api/livescore/'+tab;if(tab.startsWith('bxh_'))ep='/api/livescore/standings/'+tab.replace('bxh_','');try{const r=await fetchJsonTimeout(ep,8000);if(r.html&&r.html.length>50){el.innerHTML=r.html;bindMatchClicks(el);}else el.innerHTML='<div class="loading" style="padding:20px">Không có dữ liệu</div>';}catch(e){el.innerHTML='<div class="loading" style="padding:20px">Lỗi tải dữ liệu</div>';}}
45
+ function bindMatchClicks(container){container.querySelectorAll('.match-detail').forEach(md=>{md.addEventListener('click',function(e){e.preventDefault();e.stopPropagation();const a=this.querySelector('.status a');if(a){const m=(a.getAttribute('href')||'').match(/\/tran-dau\/(\d+)\//);if(m)openMatch(m[1]);}});});container.querySelectorAll('a').forEach(a=>a.addEventListener('click',e=>e.preventDefault()));}
46
+ async function checkAndSwitchToLive(){try{const r=await fetchJsonTimeout('/api/livescore/live',5000);if(r.html&&r.html.length>100)loadLivescore('live');}catch(e){}}
47
+ setInterval(()=>{if(_lsTab==='live')loadLivescore('live');},60000);
48
+ async function openMatch(id){if(!id)return;_currentEventId=id;document.getElementById('match-overlay').classList.add('active');document.body.style.overflow='hidden';loadMatchTab('comm');}
49
+ function closeMatch(){document.getElementById('match-overlay').classList.remove('active');document.body.style.overflow='';}
50
+ async function loadMatchTab(tab){document.querySelectorAll('.mo-tab').forEach(t=>t.classList.remove('active'));document.querySelectorAll('.mo-tab').forEach(t=>{if((tab==='comm'&&t.textContent==='Diễn biến')||(tab==='stats'&&t.textContent==='Thống kê'))t.classList.add('active');});const el=document.getElementById('mo-body');el.innerHTML='<div class="loading">Đang tải...</div>';try{const r=await fetchJsonTimeout(tab==='stats'?`/api/match/${_currentEventId}/stats`:`/api/match/${_currentEventId}/commentaries`,8000);el.innerHTML=r.html||'<div class="loading">Không dữ liệu</div>';}catch(e){el.innerHTML='<div class="loading">Lỗi</div>';}}
51
+ async function openLeaguePlayer(league,idx){showView('view-tiktok');document.querySelectorAll('.cat').forEach(x=>x.classList.remove('active'));const el=document.getElementById('view-tiktok');el.innerHTML='<div class="loading">Đang tải video...</div>';const articles=_hlLeagueData[league]||[];if(!articles.length){el.innerHTML='<div class="loading">Không có video</div>';return;}const cfg=HL_CONFIG[league]||{name:league,emoji:'🎬'};let h=`<button class="back-btn" onclick="switchCat('home')">← ${cfg.emoji} ${cfg.name}</button><div class="vp-wrap" id="vp-wrap"><video id="vp-video" playsinline controls loop></video><button class="vp-ratio-btn" onclick="toggleRatio()">16:9</button></div><div class="vp-title" id="vp-title"></div><div class="pl-list" id="pl-list">`;articles.forEach((a,i)=>{h+=`<div class="pl-item${i===idx?' active':''}" data-idx="${i}" onclick="playFromList('${league}',${i})"><div class="pl-item-thumb">${a.img?`<img src="${a.img}">`:''}<div class="card-play">▶</div></div><div class="pl-item-body"><div class="pl-item-title">${a.title}</div></div></div>`;});h+='</div>';el.innerHTML=h;playFromList(league,idx);}
52
+ async function playFromList(league,idx){const articles=_hlLeagueData[league]||[];const a=articles[idx];if(!a)return;document.querySelectorAll('.pl-item').forEach(x=>x.classList.remove('active'));document.querySelector(`.pl-item[data-idx="${idx}"]`)?.classList.add('active');const titleEl=document.getElementById('vp-title');if(titleEl)titleEl.textContent=a.title;const video=document.getElementById('vp-video');if(!video)return;if(_vpHls){_vpHls.destroy();_vpHls=null;}video.removeAttribute('src');video.innerHTML='';const r=await fetch('/api/video_url?url='+encodeURIComponent(a.link));const v=await r.json();if(!v||!v.src){titleEl.textContent=a.title+' (lỗi)';return;}if(v.type==='youtube'){const wrap=video.parentElement;wrap.innerHTML=`<iframe src="${v.src}" allowfullscreen allow="accelerometer;autoplay;clipboard-write;encrypted-media;gyroscope;picture-in-picture" style="width:100%;height:100%;border:none"></iframe>`;return;}if(v.poster)video.poster=v.poster;if(v.type==='hls'&&Hls.isSupported()){_vpHls=new Hls({maxBufferLength:60,maxMaxBufferLength:120,maxBufferSize:30*1000*1000,startLevel:-1});_vpHls.loadSource(v.src);_vpHls.attachMedia(video);_vpHls.on(Hls.Events.MANIFEST_PARSED,()=>{video.play().catch(()=>{});});}else{video.src=v.src;video.play().catch(()=>{});}video.scrollIntoView({behavior:'smooth',block:'start'});}
53
+ function toggleRatio(){const w=document.getElementById('vp-wrap');const btn=w?.querySelector('.vp-ratio-btn');if(!w||!btn)return;if(w.classList.contains('wide')){w.classList.remove('wide');btn.textContent='16:9';}else{w.classList.add('wide');btn.textContent='1:1';}}
54
+ async function loadCat(id){const el=document.getElementById('view-cat');el.innerHTML='<div class="loading">Đang tải...</div>';const arts=await fetchJsonTimeout('/api/category/'+id,9000).catch(()=>[]);if(!arts.length){el.innerHTML='<div class="loading">Không có tin</div>';return;}let h='<div class="grid">';arts.forEach(a=>{const bg=a.source==='bbc'?'badge-bbc':a.source==='dantri'?'badge-dt':a.source==='genk'?'badge-genk':'badge-vne';h+=`<div class="card" onclick="readArticle('${a.link.replace(/'/g,"\\'")}','${a.source}')"><div class="card-img">${a.img?`<img loading="lazy" src="${a.img}">`:''}</div><div class="card-body"><span class="badge ${bg}">${a.source}</span><div class="card-title">${a.title}</div></div></div>`;});h+='</div>';el.innerHTML=h;}
55
+ async function readArticle(url,source){const supported=url.includes('vnexpress.net')||url.includes('bbc.com')||url.includes('dantri.com.vn')||url.includes('genk.vn')||url.includes('thethaovanhoa.vn');if(!supported){window.open(url,'_blank');return;}showView('view-article');const el=document.getElementById('view-article');el.innerHTML='<div class="loading">Đang tải...</div>';const data=await fetchJsonTimeout('/api/article?url='+encodeURIComponent(url),12000).catch(()=>null);if(!data||data.error||!data.body||!data.body.length){el.innerHTML=`<button class="back-btn" onclick="switchCat('home')">← Quay lại</button><div class="loading"><a href="${url}" target="_blank" style="color:#5cb87a">Mở link gốc</a></div>`;return;}let h=`<button class="back-btn" onclick="switchCat('home')">← Quay lại</button><div class="article-view"><h1 class="article-title">${data.title}</h1>`;if(data.summary)h+=`<div class="article-summary">${data.summary}</div>`;let lastImg='';data.body.forEach(b=>{if(b.type==='p')h+=`<p class="article-p">${b.text}</p>`;else if(b.type==='img'&&b.src&&b.src!==lastImg){lastImg=b.src;h+=`<img loading="lazy" class="article-img" src="${b.src}">`;}else if(b.type==='heading')h+=`<h2 class="article-h2">${b.text}</h2>`;});h+=`<div class="article-actions"><button onclick="doShare('${(data.title||'').replace(/'/g,"\\'")}','${url.replace(/'/g,"\\'")}','${(data.og_image||'').replace(/'/g,"\\'")}')">📤 Chia sẻ</button><button onclick="window.open('${url}','_blank')">🔗 Gốc</button></div></div>`;el.innerHTML=h;window.scrollTo(0,0);}
56
+ function loadYT(cb){if(window.YT&&YT.Player){cb();return;}let prev=window.onYouTubeIframeAPIReady;window.onYouTubeIframeAPIReady=function(){if(typeof prev==='function')try{prev()}catch(e){}cb();};if(!_ytLoading){_ytLoading=true;let s=document.createElement('script');s.src='https://www.youtube.com/iframe_api';document.head.appendChild(s);}}
57
+ function fmtNum(n){n=Number(n||0);return n>999?(Math.round(n/100)/10)+'k':String(n)}
58
+ function postShortAction(id,action,extra){return fetch('/api/short-action',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify(Object.assign({id,action},extra||{}))}).then(r=>r.json()).then(j=>{if(j&&j.stats){_shortStats[id]=j.stats;updateShortActions(id);}return j;}).catch(()=>null)}
59
+ function updateShortActions(id){let st=_shortStats[id]||{};let q=s=>document.querySelector(s);if(q('.msv-view span'))q('.msv-view span').textContent=fmtNum(st.views);if(q('.msv-like span'))q('.msv-like span').textContent=fmtNum(st.likes);if(q('.msv-share span'))q('.msv-share span').textContent=fmtNum(st.shares);if(q('.msv-comment span'))q('.msv-comment span').textContent=fmtNum((st.comments||[]).length);if(q('.msv-like'))q('.msv-like').classList.toggle('liked',localStorage.getItem('vnews_short_like_'+id)==='1');}
60
+ function renderShortComments(id){let box=document.querySelector('.msv-comments');if(!box)return;let st=_shortStats[id]||{};let list=box.querySelector('.msv-clist');let cs=st.comments||[];list.innerHTML=cs.length?cs.map(c=>'<div class="msv-citem">'+escapeHtml(c.text)+'</div>').join(''):'<div class="msv-citem">Chưa có bình luận</div>';}
61
+ function toggleShortComments(id){let box=document.querySelector('.msv-comments');if(box){box.classList.toggle('show');renderShortComments(id);}}
62
+ function shareShort(id,title){let url=location.origin+'/v?url='+encodeURIComponent('https://www.youtube.com/watch?v='+id)+'&title='+encodeURIComponent(title||'Short')+'&type=shorts';postShortAction(id,'share');if(navigator.share)navigator.share({title:title||'Short',url}).catch(()=>{});else navigator.clipboard.writeText(url).then(()=>alert('Đã sao chép link!')).catch(()=>{});}
63
+ function openShortsViewer(i){const data=_shortsData||[];if(!data.length)return;let idx=i||0;document.documentElement.classList.add('ms-lock');document.body.classList.add('ms-lock');const v=document.createElement('div');v.className='msv';v.innerHTML='<div class="msv-bg"></div><div class="msv-stage"><div class="msv-phone"><div id="msv-player"></div></div></div><div class="msv-swipe"></div><button class="msv-close">×</button><div class="msv-info"><div class="msv-count"></div><div class="msv-title"></div></div><div class="msv-nav"><button class="msv-up">▲</button><button class="msv-down">▼</button></div><div class="msv-actions"><button class="msv-act msv-view">👁<span>0</span></button><button class="msv-act msv-like">♥<span>0</span></button><button class="msv-act msv-comment">💬<span>0</span></button><button class="msv-act msv-share">↗<span>0</span></button></div><div class="msv-comments"><div class="msv-ctitle">Bình luận</div><div class="msv-clist"></div><div class="msv-crow"><input class="msv-cinput" placeholder="Nhập bình luận..."><button class="msv-send">Gửi</button></div></div>';document.body.appendChild(v);let ids=data.map(x=>x.id).filter(Boolean).join(',');fetch('/api/short-stats?ids='+encodeURIComponent(ids)).then(r=>r.json()).then(j=>{_shortStats=j.stats||{};updateShortActions(currentId());renderShortComments(currentId());}).catch(()=>{});function current(){return data[idx]||{}}function currentId(){let it=current();return it.id||((it.link||'').match(/(?:v=|youtu\.be\/)([A-Za-z0-9_-]{11})/)||[])[1]||'';}function countView(id){let k='vnews_short_view_'+id;let now=Date.now(),last=Number(sessionStorage.getItem(k)||0);if(now-last>25000){sessionStorage.setItem(k,now);postShortAction(id,'view');}}function play(n){idx=(n+data.length)%data.length;const it=current();const id=currentId();v.querySelector('.msv-bg').style.backgroundImage='url('+(it.img||'')+')';v.querySelector('.msv-count').textContent=(idx+1)+'/'+data.length;v.querySelector('.msv-title').textContent=it.title||'';updateShortActions(id);renderShortComments(id);loadYT(()=>{if(_shortsPlayer&&_shortsPlayer.loadVideoById){_shortsPlayer.loadVideoById(id);}else{_shortsPlayer=new YT.Player('msv-player',{videoId:id,playerVars:{autoplay:1,playsinline:1,rel:0,controls:1},events:{onReady:e=>{try{e.target.playVideo()}catch(_){ }},onStateChange:e=>{if(e.data===YT.PlayerState.ENDED)play(idx+1);}}});}countView(id);});}function close(){try{if(_shortsPlayer&&_shortsPlayer.destroy)_shortsPlayer.destroy()}catch(e){} _shortsPlayer=null;v.remove();document.documentElement.classList.remove('ms-lock');document.body.classList.remove('ms-lock');document.removeEventListener('keydown',key);}function key(e){if(e.key==='Escape')close();if(e.key==='ArrowUp')play(idx-1);if(e.key==='ArrowDown')play(idx+1);}v.querySelector('.msv-close').onclick=close;v.querySelector('.msv-up').onclick=()=>play(idx-1);v.querySelector('.msv-down').onclick=()=>play(idx+1);v.querySelector('.msv-like').onclick=()=>{let id=currentId();if(localStorage.getItem('vnews_short_like_'+id)==='1')return;localStorage.setItem('vnews_short_like_'+id,'1');postShortAction(id,'like');};v.querySelector('.msv-share').onclick=()=>shareShort(currentId(),current().title);v.querySelector('.msv-comment').onclick=()=>toggleShortComments(currentId());v.querySelector('.msv-send').onclick=()=>{let inp=v.querySelector('.msv-cinput');let txt=(inp.value||'').trim();if(!txt)return;inp.value='';postShortAction(currentId(),'comment',{text:txt}).then(()=>renderShortComments(currentId()));};let sy=0;v.querySelector('.msv-swipe').addEventListener('touchstart',e=>{sy=e.touches[0].clientY},{passive:true});v.querySelector('.msv-swipe').addEventListener('touchend',e=>{let dy=(e.changedTouches[0].clientY-sy);if(Math.abs(dy)>45)play(idx+(dy<0?1:-1));},{passive:true});document.addEventListener('keydown',key);play(idx);}
64
+ async function loadVideos(){const el=document.getElementById('view-video');if(el.dataset.loaded)return;el.innerHTML='<div class="loading">Đang tải...</div>';const[hl,bdp]=await Promise.all([fetchJsonTimeout('/api/highlights',9000).catch(()=>[]),fetchJsonTimeout('/api/bdp_videos',9000).catch(()=>[])]);let h='<div class="section-title">🎬 Highlight</div><div class="grid">';hl.forEach((a,i)=>{h+=`<div class="card" onclick="openTikTok('highlights',${i})"><div class="card-img">${a.img?`<img loading="lazy" src="${a.img}">`:''}<div class="card-play">▶</div></div><div class="card-body"><span class="badge badge-fpt">HL</span><div class="card-title">${a.title}</div></div></div>`;});h+='</div>';if(bdp.length){h+='<div class="section-title">⚽ BDP</div><div class="grid">';bdp.forEach((a,i)=>{h+=`<div class="card" onclick="openTikTok('bdp',${i})"><div class="card-img">${a.img?`<img loading="lazy" src="${a.img}">`:''}<div class="card-play">▶</div></div><div class="card-body"><span class="badge badge-fpt">BDP</span><div class="card-title">${a.title}</div></div></div>`;});h+='</div>';}el.innerHTML=h;el.dataset.loaded='1';}
65
+ async function openTikTok(type,startIdx){showView('view-tiktok');document.querySelectorAll('.cat').forEach(x=>x.classList.remove('active'));const el=document.getElementById('view-tiktok');el.innerHTML='<div class="loading">Đang tải video...</div>';let articles;if(type==='shorts'){_shortsData=await fetchJsonTimeout('/api/shorts',9000).catch(()=>[]);openShortsViewer(startIdx);return;}else if(type==='highlights')articles=await fetchJsonTimeout('/api/highlights',9000).catch(()=>[]);else articles=await fetchJsonTimeout('/api/bdp_videos',9000).catch(()=>[]);await buildTikTokPlayer(articles,startIdx,type);}
66
+ async function openTikTokByUrl(targetUrl,type){showView('view-tiktok');document.querySelectorAll('.cat').forEach(x=>x.classList.remove('active'));const el=document.getElementById('view-tiktok');el.innerHTML='<div class="loading">Đang tải video...</div>';let articles=await fetchJsonTimeout(type==='shorts'?'/api/shorts':type==='highlights'?'/api/highlights':'/api/bdp_videos',9000).catch(()=>[]);let startIdx=0;for(let i=0;i<articles.length;i++){if(articles[i].link===targetUrl){startIdx=i;break;}}await buildTikTokPlayer(articles,startIdx,type);}
67
+ async function buildTikTokPlayer(articles,startIdx,type){const el=document.getElementById('view-tiktok');const vids=[];const results=await Promise.all(articles.map(async(a,i)=>{try{const r=await fetch('/api/video_url?url='+encodeURIComponent(a.link));const v=await r.json();if(v&&v.src)return{article:a,video:v,idx:i};}catch(e){}return null;}));results.forEach(r=>{if(!r)return;const{article:a,video:v,idx:i}=r;vids.push({...a,...v,_idx:i});});vids.sort((a,b)=>a._idx-b._idx);if(!vids.length){el.innerHTML='<div class="loading">Không tìm thấy video</div>';return;}let ti=vids.findIndex(v=>v._idx===startIdx);if(ti<0)ti=0;const ordered=ti>0?[...vids.slice(ti),...vids.slice(0,ti)]:vids;_tikData=ordered;let h=`<button class="back-btn" onclick="switchCat('home')">← Quay lại</button><div class="tiktok-container"><div class="tiktok-feed" id="tiktok-feed">`;ordered.forEach((v,i)=>{const isYT=v.type==='youtube';const isHLS=!isYT&&v.src&&v.src.includes('.m3u8');const poster=v.poster?` poster="${v.poster}"`:'';const vtag=isYT?`<iframe data-yt-src="${v.src}" allowfullscreen allow="accelerometer;autoplay;clipboard-write;encrypted-media;gyroscope;picture-in-picture"></iframe>`:isHLS?`<video playsinline preload="none"${poster} data-hls="${v.src}" loop controls></video>`:`<video playsinline preload="none"${poster} loop controls><source src="${v.src}" type="video/mp4"></video>`;h+=`<div class="tiktok-slide" id="tslide-${i}">${vtag}<div class="tiktok-bottom"><span class="badge badge-fpt">VIDEO</span><p class="tiktok-title">${v.title}</p></div><div class="tiktok-right"><button class="tiktok-right-btn" onclick="event.stopPropagation();shareVid(${i})"><div class="icon">📤</div></button></div><span class="tiktok-counter">${i+1}/${ordered.length}</span></div>`;});h+='</div></div>';el.innerHTML=h;initFeed();}
68
+ function shareVid(i){const v=_tikData[i];if(!v)return;doShareVideo(v.title,v.link||'',v.poster||v.img||'','highlights');}
69
+ function initFeed(){const feed=document.getElementById('tiktok-feed');if(!feed)return;const slides=feed.querySelectorAll('.tiktok-slide');let cur=-1;function activate(idx){if(idx===cur)return;slides.forEach((sl,i)=>{const v=sl.querySelector('video');const f=sl.querySelector('iframe');if(i===idx){if(v&&v.dataset.hls){if(!v._hls){const hls=new Hls();hls.loadSource(v.dataset.hls);hls.attachMedia(v);hls.on(Hls.Events.MANIFEST_PARSED,()=>v.play().catch(()=>{}));v._hls=hls;}else v.play().catch(()=>{});}else if(v)v.play().catch(()=>{});if(f&&!f.src&&f.dataset.ytSrc)f.src=f.dataset.ytSrc;}else{if(v){v.pause();if(v._hls){v._hls.destroy();v._hls=null;}}if(f&&f.src)f.src='';}});cur=idx;}let sT;feed.addEventListener('scroll',()=>{clearTimeout(sT);sT=setTimeout(()=>{const rect=feed.getBoundingClientRect();const ctr=rect.top+rect.height/2;let best=-1,bestD=1e9;slides.forEach((sl,i)=>{const d=Math.abs(sl.getBoundingClientRect().top+sl.getBoundingClientRect().height/2-ctr);if(d<bestD){bestD=d;best=i;}});if(best>=0)activate(best);},150);});setTimeout(()=>activate(0),300);}
70
  init();
71
  </script>
72
  </body>