Add lightweight AI rewrite tone selector and local AI wall slide
Browse files- static/index.html +30 -62
static/index.html
CHANGED
|
@@ -4,85 +4,53 @@
|
|
| 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,
|
| 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}.grid{display:grid;grid-template-columns:repeat(2,1fr);gap:6px;padding:6px 4px}@media(min-width:600px){.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(.97)}.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:28px;height:28px;background:rgba(0,0,0,.5);border-radius:50%;color:#fff;display:flex;align-items:center;justify-content:center;font-size:11px}.card-body{padding:6px 8px}.card-title{font-size:11px;font-weight:600;color:#eee;line-height:1.3;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-bdp{background:#1a5c35;color:#fff}.badge-fpt{background:#f26522;color:#fff}.badge-dt{background:#1565c0;color:#fff}.slider-wrap{margin:6px 4px;background:#1a1a1a;border-radius:8px;overflow:hidden;border:1px solid #2a2a2a}.slider-header{padding:7px 10px}.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;scroll-snap-align:start;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.2;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}.featured-match{margin:6px 4px;background:linear-gradient(135deg,#1a2a1f,#0d1117);border-radius:10px;padding:12px;border:1px solid #2d8659;cursor:pointer}.featured-match .fm-league{color:#5cb87a;font-size:9px;font-weight:600;text-transform:uppercase;margin-bottom:6px;text-align:center}.featured-match .fm-teams{display:flex;align-items:center;justify-content:center;gap:10px}.featured-match .fm-team{display:flex;flex-direction:column;align-items:center;gap:4px;flex:1}.featured-match .fm-team img{width:32px;height:32px;object-fit:contain}.featured-match .fm-team span{font-size:10px;color:#ccc;text-align:center}.featured-match .fm-score{font-size:22px;font-weight:900;color:#fff;min-width:60px;text-align:center}.featured-match .fm-status{text-align:center;margin-top:6px;font-size:9px;color:#e74c3c;font-weight:600}.featured-match .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::-webkit-scrollbar{width:3px}.ls-content::-webkit-scrollbar-thumb{background:#333;border-radius:3px}.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}.match-overlay .mo-header{padding:10px;background:#1a1a1a;display:flex;justify-content:space-between;align-items:center;position:sticky;top:0;z-index:1}.match-overlay .mo-header h3{color:#fff;font-size:13px}.match-overlay .mo-close{background:none;border:none;color:#fff;font-size:22px;cursor:pointer}.match-overlay .mo-tabs{display:flex;gap:4px;padding:8px 10px;background:#1a1a1a}.match-overlay .mo-tab{padding:5px 12px;background:#222;border:1px solid #333;border-radius:10px;color:#999;font-size:10px;cursor:pointer}.match-overlay .mo-tab.active{background:#2d8659;border-color:#2d8659;color:#fff}.match-overlay .mo-body{padding:8px;flex:1;overflow-x:auto}.match-overlay .mo-body .time{color:#5cb87a;font-weight:700;font-size:11px;margin-right:6px}.match-overlay .mo-body .description{color:#ccc;font-size:12px;line-height:1.4}.match-overlay .mo-body li{list-style:none;padding:6px 0;border-bottom:1px solid #222}.match-overlay .mo-body .tab-content{margin-bottom:8px}.match-overlay .mo-body .professional-table{width:100%;font-size:10px;border-collapse:collapse;min-width:500px}.match-overlay .mo-body .professional-table th{color:#888;padding:6px 4px;text-align:center;border-bottom:1px solid #333;white-space:nowrap}.match-overlay .mo-body .professional-table th.sticky-column{text-align:left;position:sticky;left:0;background:#111;z-index:1;min-width:120px}.match-overlay .mo-body .professional-table th svg{width:14px;height:14px;vertical-align:middle;fill:#888}.match-overlay .mo-body .professional-table th svg g{fill:#888}.match-overlay .mo-body .professional-table th svg path{fill:#888}.match-overlay .mo-body .professional-table th span{display:none}.match-overlay .mo-body .professional-table td{color:#ccc;padding:6px 4px;border-bottom:1px solid #1a1a1a;text-align:center;font-size:11px}.match-overlay .mo-body .professional-table td.sticky-column{text-align:left;position:sticky;left:0;background:#111;z-index:1}.match-overlay .mo-body .professional-table .player-info{display:flex;align-items:center;gap:6px}.match-overlay .mo-body .professional-table .player-logo{width:18px;height:18px;border-radius:50%;object-fit:contain}.match-overlay .mo-body .professional-table .player-name{color:#ddd;text-decoration:none;font-size:11px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis;max-width:100px}.match-overlay .mo-body .professional-table .score{font-weight:700;border-radius:4px;padding:2px 5px;color:#fff}.match-overlay .mo-body .professional-table .bg-blue{background:#2563eb}.match-overlay .mo-body .professional-table .bg-green{background:#16a34a}.match-overlay .mo-body .professional-table .bg-orange{background:#ea580c}.loading{text-align:center;padding:40px;color:#666;font-size:12px}.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:700px;margin:0 auto}.article-title{font-size:18px;font-weight:800;color:#f0f0f0;line-height:1.3;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.7;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:10px;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:6px 12px;border-radius:14px;font-size:11px;cursor:pointer}.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}.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-wrap video,.vp-wrap iframe{width:100%;height:100%;object-fit:contain;border:none}.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:#222}.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}.wall-wrap{margin:8px 4px;background:#1a1a1a;border:1px solid #2a2a2a;border-radius:8px;padding:8px}.wall-post{border-top:1px solid #222;padding:8px 0}.wall-post:first-child{border-top:0}.wall-post-title{font-size:12px;color:#5cb87a;font-weight:700;margin-bottom:4px}.wall-post-text{font-size:12px;color:#ccc;line-height:1.45;white-space:pre-wrap}.wall-post-img{width:100%;max-height:180px;object-fit:cover;border-radius:6px;margin:6px 0}.tiktok-container{width:100%;height:80vh;max-height:680px;min-height:400px;background:#000}.tiktok-feed{height:100%;overflow-y:scroll;scroll-snap-type:y mandatory;scrollbar-width:none}.tiktok-feed::-webkit-scrollbar{display:none}.tiktok-slide{height:80vh;max-height:680px;min-height:400px;scroll-snap-align:start;scroll-snap-stop:always;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:none}.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:600;margin:3px 0 5px;line-height:1.3}.tiktok-counter{position:absolute;top:8px;left:8px;background:rgba(0,0,0,.5);color:#fff;font-size:9px;padding:2px 7px;border-radius:8px;z-index:4}.tiktok-right{position:absolute;right:8px;bottom:100px;display:flex;flex-direction:column;align-items:center;gap:14px;z-index:5}.tiktok-right-btn{display:flex;flex-direction:column;align-items:center;gap:2px;cursor:pointer;background:none;border:none;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 ·
|
| 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=[],
|
| 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 |
-
const LS={get(k){try{return JSON.parse(localStorage.getItem(k))}catch{return null}},set(k,v){try{localStorage.setItem(k,JSON.stringify(v))}catch{}}};
|
| 29 |
-
function hid(s){let h=0;for(let i=0;i<s.length;i++){h=((h<<5)-h)+s.charCodeAt(i);h|=0;}return Math.abs(h).toString(36).slice(0,10);}
|
| 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 openMatch(id){if(!id)return;_currentEventId=id;document.getElementById('match-overlay').classList.add('active');document.body.style.overflow='hidden';loadMatchTab('comm');}
|
| 33 |
-
function closeMatch(){document.getElementById('match-overlay').classList.remove('active');document.body.style.overflow='';}
|
| 34 |
-
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 fetch(tab==='stats'?`/api/match/${_currentEventId}/stats`:`/api/match/${_currentEventId}/commentaries`);const d=await r.json();el.innerHTML=d.html||'<div class="loading">Không có dữ liệu</div>';}catch(e){el.innerHTML='<div class="loading">Lỗi</div>';}}
|
| 35 |
-
let _lsTab='today';
|
| 36 |
-
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',function(e){e.preventDefault();})});}
|
| 37 |
-
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');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 fetch(ep);const d=await r.json();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</div>';}}
|
| 38 |
-
setInterval(()=>{if(_lsTab==='live')loadLivescore('live');},60000);
|
| 39 |
-
async function checkAndSwitchToLive(){try{const r=await fetch('/api/livescore/live');const d=await r.json();if(d.html&&d.html.length>100)loadLivescore('live');}catch(e){}}
|
| 40 |
-
async function refreshFeatured(){try{const r=await fetch('/api/livescore/featured');const f=await r.json();if(!f||!f.home)return;_featuredEventId=f.event_id;const el=document.querySelector('.featured-match');if(!el)return;const sc=f.status==='live'?'':'upcoming';const st=f.status==='live'?`🔴 ${f.minute||'LIVE'}`:`⏰ ${f.time}`;el.querySelector('.fm-league').textContent=f.league;el.querySelector('.fm-score').textContent=f.score||'VS';el.querySelector('.fm-status').className='fm-status '+sc;el.querySelector('.fm-status').textContent=st;el.setAttribute('onclick',`openMatch('${f.event_id}')`);}catch(e){}}
|
| 41 |
-
setInterval(refreshFeatured,30000);
|
| 42 |
-
async function init(){_cats=await fetch('/api/categories').then(r=>r.json());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();}
|
| 43 |
-
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=>{v.pause();if(v._hls){v._hls.destroy();v._hls=null;}});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);}}
|
| 44 |
-
function showView(id){document.querySelectorAll('.view').forEach(x=>x.classList.remove('active'));document.getElementById(id).classList.add('active');}
|
| 45 |
-
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>`;h+=`<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>`;h+=`<div class="vp-title" id="vp-title"></div>`;h+=`<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);}
|
| 46 |
-
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(()=>{});});_vpHls.on(Hls.Events.ERROR,(e,d)=>{if(d.fatal&&d.type===Hls.ErrorTypes.NETWORK_ERROR)_vpHls.startLoad();});}else{video.src=v.src;video.play().catch(()=>{});}video.scrollIntoView({behavior:'smooth',block:'start'});}
|
| 47 |
-
async function loadHome(){const[news,sh,dantri,featured,hlLeagues]=await Promise.all([fetch('/api/homepage').then(r=>r.json()),Promise.resolve([]),Promise.resolve([]),fetch('/api/livescore/featured').then(r=>r.json()).catch(()=>null),fetch('/api/highlights/leagues').then(r=>r.json()).catch(()=>({}))]);_hlLeagueData=hlLeagues;let h='';
|
| 48 |
-
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>`;}
|
| 49 |
-
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>`;
|
| 50 |
-
if(dantri&&dantri.length){h+='<div class="slider-wrap"><div class="slider-header"><span class="slider-label">🤖 Ứng dụng AI</span></div><div class="slider-track">';dantri.forEach(a=>{h+=`<div class="slider-item" onclick="readArticle('${a.link.replace(/'/g,"\\'")}','genk')"><div class="slider-thumb">${a.img?`<img src="${a.img}" onerror="this.parentElement.style.background='#222'">`:''}</div><div class="slider-title">${a.title}</div></div>`;});h+='</div></div>';}
|
| 51 |
-
|
| 52 |
-
for(const[key,cfg] of Object.entries(HL_CONFIG)){const vids=hlLeagues[key];if(!vids||!vids.length)continue;h+=`<div class="slider-wrap"><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 src="${a.img}">`:''}<div class="card-play">▶</div></div><div class="slider-title">${a.title}</div></div>`;});h+='</div></div>';}
|
| 53 |
-
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':'badge-vne';const lb=a.source==='bbc'?'BBC':'VnE';h+=`<div class="card" onclick="readArticle('${a.link.replace(/'/g,"\\'")}','${a.source}')"><div class="card-img">${a.img?`<img 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>';}
|
| 54 |
-
document.getElementById('view-home').innerHTML=h;loadWallAsync();loadShortsAsync();loadAiAsync();loadWorldCupRoadAsync();loadLivescore('today');checkAndSwitchToLive();}
|
| 55 |
-
|
| 56 |
-
function fetchJsonTimeout(url,ms){const c=new AbortController();const t=setTimeout(()=>c.abort(),ms);return fetch(url,{signal:c.signal}).then(r=>r.json()).finally(()=>clearTimeout(t));}
|
| 57 |
-
function loadWallAsync(){fetchJsonTimeout('/api/wall',8000).then(j=>{let ps=(j&&j.posts)||[];if(!ps.length)return;let wrap=document.createElement('div');wrap.className='wall-wrap';let html='<div class="slider-header" style="padding:0 0 6px"><span class="slider-label">🧱 Tường trang chủ</span></div>';ps.slice(0,8).forEach(p=>{html+='<div class="wall-post">'+(p.img?'<img class="wall-post-img" src="'+p.img+'">':'')+'<div class="wall-post-title">'+escapeHtml(p.title||'')+'</div><div class="wall-post-text">'+escapeHtml(p.text||'')+'</div></div>';});wrap.innerHTML=html;wrap.dataset.wall='1';const home=document.getElementById('view-home');if(home&&!home.querySelector('[data-wall="1"]'))home.insertBefore(wrap,home.children[1]||null);}).catch(()=>{});}
|
| 58 |
-
function rewriteAndShare(url){if(!confirm('Dùng AI viết lại bài này và đăng lên tường trang chủ?'))return;fetch('/api/rewrite_share',{method:'POST',headers:{'Content-Type':'application/json'},body:JSON.stringify({url})}).then(r=>r.json()).then(j=>{if(j&&j.post){alert('Đã đăng lên tường trang chủ');switchCat('home');setTimeout(loadWallAsync,500);}else alert(j.error||'Lỗi rewrite');}).catch(()=>alert('Lỗi rewrite'));}
|
| 59 |
-
function loadAiAsync(){fetchJsonTimeout('/api/genk_ai',12000).then(ai=>{if(!ai||!ai.length)return;let wrap=document.createElement('div');wrap.className='slider-wrap';let h='<div class="slider-header"><span class="slider-label">🤖 Ứng dụng AI</span></div><div class="slider-track">';ai.forEach(a=>{h+=`<div class="slider-item" onclick="readArticle('${a.link.replace(/'/g,"\\'")}','genk')"><div class="slider-thumb">${a.img?`<img src="${a.img}" onerror="this.parentElement.style.background='#222'">`:''}</div><div class="slider-title">${a.title}</div></div>`;});h+='</div>';wrap.innerHTML=h;wrap.dataset.ai='1';const home=document.getElementById('view-home');if(!home||home.querySelector('[data-ai="1"]'))return;home.insertBefore(wrap,home.children[2]||null);}).catch(()=>{});}
|
| 60 |
-
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 src="${a.img}" onerror="this.parentElement.style.background='#222'">`:''}</div><div class="slider-title">${a.title}</div></div>`;});h+='</div>';wrap.innerHTML=h;wrap.dataset.wcroad='1';const home=document.getElementById('view-home');if(!home||home.querySelector('[data-wcroad="1"]'))return;const ai=[...home.querySelectorAll('.slider-label')].find(x=>x.textContent.includes('Ứng dụng AI'));const aiWrap=ai?ai.closest('.slider-wrap'):null;if(aiWrap&&aiWrap.nextSibling)home.insertBefore(wrap,aiWrap.nextSibling);else home.insertBefore(wrap,home.children[3]||null);}).catch(()=>{});}
|
| 61 |
-
function loadShortsAsync(){fetchJsonTimeout('/api/shorts',12000).then(sh=>{if(!sh||!sh.length)return;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 src="${a.img}">`:''}<div class="card-play">▶</div></div><div class="slider-title">${a.title}</div></div>`;});h+='</div>';wrap.innerHTML=h;wrap.dataset.shorts='1';_shortsData=sh;const home=document.getElementById('view-home');if(!home||home.querySelector('[data-shorts="1"]'))return;const ai=[...home.querySelectorAll('.slider-label')].find(x=>x.textContent.includes('Ứng dụng AI'));const aiWrap=ai?ai.closest('.slider-wrap'):null;if(aiWrap&&aiWrap.nextSibling)home.insertBefore(wrap,aiWrap.nextSibling);else home.insertBefore(wrap,home.children[2]||null);}).catch(()=>{});}
|
| 62 |
-
let _shortsData=[];
|
| 63 |
-
|
| 64 |
-
let _shortStats={},_shortsPlayer=null,_ytReady=false,_ytLoading=false;
|
| 65 |
-
function fmtNum(n){n=Number(n||0);return n>999?(Math.round(n/100)/10)+'k':String(n)}
|
| 66 |
-
function loadYT(cb){if(window.YT&&YT.Player){cb();return;}let prev=window.onYouTubeIframeAPIReady;window.onYouTubeIframeAPIReady=function(){_ytReady=true;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);}}
|
| 67 |
function escapeHtml(s){return String(s||'').replace(/[&<>"']/g,m=>({'&':'&','<':'<','>':'>','"':'"',"'":'''}[m]));}
|
| 68 |
-
function
|
| 69 |
-
function
|
| 70 |
-
function
|
| 71 |
-
function
|
| 72 |
-
function
|
| 73 |
-
|
| 74 |
-
function
|
| 75 |
-
|
| 76 |
-
async function
|
| 77 |
-
|
| 78 |
-
let
|
| 79 |
-
let
|
| 80 |
-
|
| 81 |
-
|
| 82 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 83 |
function shareVid(i){const v=_tikData[i];if(!v)return;doShareVideo(v.title,v.link||'',v.poster||v.img||'','highlights');}
|
| 84 |
-
function initFeed(){const feed=document.getElementById('tiktok-feed');if(!feed)return;const slides=feed.querySelectorAll('.tiktok-slide');let cur=-1;function
|
| 85 |
-
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';}}
|
| 86 |
init();
|
| 87 |
</script>
|
| 88 |
</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, video short, highlight bóng đá.">
|
| 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-ai{background:#2d8659;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-note{font-size:10px;color:#777}.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,.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-tone{font-size:9px;color:#f0c040;margin-bottom:4px}.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}.tiktok-container{width:100%;height:80vh;max-height:680px;min-height:400px;background:#000}.tiktok-feed{height:100%;overflow-y:scroll;scroll-snap-type:y mandatory;scrollbar-width:none}.tiktok-feed::-webkit-scrollbar{display:none}.tiktok-slide{height:80vh;max-height:680px;min-height:400px;scroll-snap-align:start;scroll-snap-stop:always;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:none}.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:600;margin:3px 0 5px;line-height:1.3}.tiktok-counter{position:absolute;top:8px;left:8px;background:rgba(0,0,0,.5);color:#fff;font-size:9px;padding:2px 7px;border-radius:8px;z-index:4}.tiktok-right{position:absolute;right:8px;bottom:100px;display:flex;flex-direction:column;align-items:center;gap:14px;z-index:5}.tiktok-right-btn{display:flex;flex-direction:column;align-items:center;gap:2px;cursor:pointer;background:none;border:none;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}.ms-lock{overflow:hidden!important}
|
| 14 |
</style>
|
| 15 |
</head>
|
| 16 |
<body>
|
| 17 |
+
<div class="header"><h1>📰 VNEWS</h1><p>Tin tức · Video · 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 |
<script>
|
| 25 |
+
const SPACE=location.origin;let _cats=[],_tikData=[],_localWall=[],_currentArticle=null;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 26 |
function escapeHtml(s){return String(s||'').replace(/[&<>"']/g,m=>({'&':'&','<':'<','>':'>','"':'"',"'":'''}[m]));}
|
| 27 |
+
function doShareVideo(title,videoUrl,img,type){const shareUrl=SPACE+'/v?url='+encodeURIComponent(videoUrl)+'&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 video!')).catch(()=>{});}
|
| 28 |
+
function doShare(title,articleUrl,img){const shareUrl=SPACE+'/s?url='+encodeURIComponent(articleUrl)+'&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 link!')).catch(()=>{});}
|
| 29 |
+
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));}
|
| 30 |
+
function loadLocalWall(){try{return JSON.parse(localStorage.getItem('vnews_ai_wall')||'[]')}catch{return[]}}
|
| 31 |
+
function saveLocalWall(arr){try{localStorage.setItem('vnews_ai_wall',JSON.stringify(arr.slice(0,80)))}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 loadHomeFast();const pendingVideo=localStorage.getItem('pending_video');if(pendingVideo){localStorage.removeItem('pending_video');try{const pv=JSON.parse(pendingVideo);if(pv.url)openTikTokByUrl(pv.url,pv.type||'highlights');}catch(e){}}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()}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);}}
|
| 34 |
+
function showView(id){document.querySelectorAll('.view').forEach(x=>x.classList.remove('active'));document.getElementById(id).classList.add('active');}
|
| 35 |
+
async function loadHomeFast(){const home=document.getElementById('view-home');home.innerHTML='<div id="home-wall"></div><div id="home-fast"></div><div id="home-lazy"></div><div class="loading" id="home-loadnote">Đang tải tin chính...</div>';renderLocalWall();fetchJsonTimeout('/api/homepage',9000).then(news=>renderNews(news||[])).catch(()=>{let n=document.getElementById('home-loadnote');if(n)n.textContent='Không tải được tin chính';});setTimeout(loadLazySections,80);}
|
| 36 |
+
function renderLocalWall(){_localWall=loadLocalWall();const el=document.getElementById('home-wall');if(!el)return;if(!_localWall.length){el.innerHTML='';return;}el.innerHTML=renderWallSlide(_localWall);}
|
| 37 |
+
function renderNews(news){const 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>';}
|
| 38 |
+
async function loadLazySections(){const lazy=document.getElementById('home-lazy');Promise.allSettled([fetchJsonTimeout('/api/dantri_hot',7000),fetchJsonTimeout('/api/shorts',9000),fetchJsonTimeout('/api/vne_video',7000),fetchJsonTimeout('/api/highlights',9000)]).then(res=>{let dantri=res[0].value||[],shorts=res[1].value||[],vneVid=res[2].value||[],hl=res[3].value||[];let out='';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;});}
|
| 39 |
+
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">Rewrite dạng tóm tắt</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-tone">Giọng: ${escapeHtml(p.toneLabel||p.tone||'Tóm tắt')}</div><div class="wall-text">${escapeHtml(p.text)}</div><div class="wall-actions"><button class="primary" onclick="readWallPost('${p.id}')">Xem</button><button onclick="deleteWallPost('${p.id}')">Xóa</button></div></div>`;});h+='</div></div>';return h;}
|
| 40 |
+
function readWallPost(id){let p=loadLocalWall().find(x=>x.id===id);if(!p)return;showView('view-article');let h=`<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">Giọng: ${escapeHtml(p.toneLabel||p.tone||'Tóm tắt')}</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>`;document.getElementById('view-article').innerHTML=h;window.scrollTo(0,0);}
|
| 41 |
+
function deleteWallPost(id){if(!confirm('Xóa bài khỏi Tường AI?'))return;let arr=loadLocalWall().filter(x=>x.id!==id);saveLocalWall(arr);renderLocalWall();}
|
| 42 |
+
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;}
|
| 43 |
+
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;}_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 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"><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);}
|
| 44 |
+
const toneLabels={'vui-ve':'Vui vẻ','nghiem-tuc':'Nghiêm túc','nghi-luan':'Nghị luận','hoi-dap':'Hỏi đáp','soi-noi':'Sôi nổi','thu-hut':'Thu hút','phan-tich':'Phân tích','chuyen-gia':'Chuyên gia'};
|
| 45 |
+
function summarizeSentences(text,n){let s=text.replace(/\s+/g,' ').split(/(?<=[.!?])\s+/).map(x=>x.trim()).filter(x=>x.length>35);return s.slice(0,n);}
|
| 46 |
+
function rewriteByTone(data,tone){let title=data.title||'';let summary=data.summary||'';let ps=(data.body||[]).filter(b=>b.type==='p').map(b=>b.text).join(' ');let points=summarizeSentences((summary?summary+'. ':'')+ps,4);if(!points.length)points=[title];let lead=points[0]||title;let rest=points.slice(1,4);if(tone==='vui-ve')return `😊 ${title}\n\n${lead}\n\nTóm lại, câu chuyện này có điểm đáng chú ý là: ${rest.join(' ')}\n\nMột bản tin nhanh, dễ hiểu và đáng để theo dõi.`;if(tone==='nghiem-tuc')return `TÓM TẮT NGHIÊM TÚC\n\n${title}\n\n${lead}\n\nCác ý chính:\n${rest.map(x=>'• '+x).join('\n')}`;if(tone==='nghi-luan')return `Góc nhìn nghị luận: ${title}\n\n${lead}\n\nVấn đề đặt ra là ${rest.join(' ')}\n\nĐiều này cho thấy sự việc cần được nhìn nhận trong bối cảnh rộng hơn, thay vì chỉ dừng ở một chi tiết đơn lẻ.`;if(tone==='hoi-dap')return `Hỏi nhanh đáp gọn về: ${title}\n\nHỏi: Sự việc chính là gì?\nĐáp: ${lead}\n\nHỏi: Có điểm nào đáng chú ý?\nĐáp: ${rest.join(' ')}\n\nHỏi: Vì sao nên quan tâm?\nĐáp: Vì thông tin này có thể ảnh hưởng đến cách chúng ta nhìn nhận vấn đề hiện tại.`;if(tone==='soi-noi')return `🔥 Đáng chú ý: ${title}\n\n${lead}\n\n${rest.join(' ')}\n\nDiễn biến này đang tạo nhiều sự quan tâm và có thể còn tiếp tục được cập nhật.`;if(tone==='thu-hut')return `Bạn có thể đã bỏ lỡ tin này: ${title}\n\n${lead}\n\nĐiểm khiến câu chuyện đáng chú ý nằm ở chỗ: ${rest.join(' ')}\n\nTheo dõi tiếp để không bỏ qua các diễn biến mới.`;if(tone==='phan-tich')return `Phân tích nhanh: ${title}\n\n1. Bối cảnh: ${lead}\n\n2. Diễn biến chính: ${rest[0]||''}\n\n3. Điểm đáng chú ý: ${rest.slice(1).join(' ')}\n\nNhìn chung, đây là thông tin cần được theo dõi thêm.`;if(tone==='chuyen-gia')return `Nhận định chuyên gia: ${title}\n\n${lead}\n\nTừ dữ liệu hiện có, có thể rút ra một số điểm chính:\n${rest.map(x=>'• '+x).join('\n')}\n\nĐánh giá sơ bộ: sự việc có giá trị thông tin cao và nên được tiếp tục cập nhật.`;return `${title}\n\n${points.join('\n')}`;}
|
| 47 |
+
function rewriteCurrentArticle(){if(!_currentArticle)return;let tone=document.getElementById('rewrite-tone')?.value||'nghiem-tuc';let data=_currentArticle.data;let text=rewriteByTone(data,tone);let img=data.og_image||((data.body||[]).find(b=>b.type==='img')||{}).src||'';let post={id:'ai_'+Date.now(),url:_currentArticle.url,title:data.title||'AI Rewrite',img,text,tone,toneLabel:toneLabels[tone]||tone,ts:Date.now()};let arr=loadLocalWall();arr.unshift(post);saveLocalWall(arr);document.getElementById('rewrite-result').innerHTML=`<div class="rewrite-box"><div class="rewrite-title">Đã rewrite và đăng lên Tường AI · ${post.toneLabel}</div><div class="rewrite-text">${escapeHtml(text)}</div></div>`;alert('Đã đăng bài rewrite lên Tường AI');}
|
| 48 |
+
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';}
|
| 49 |
+
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;if(type==='shorts')articles=await fetchJsonTimeout('/api/shorts',9000).catch(()=>[]);else if(type==='highlights')articles=await fetchJsonTimeout('/api/highlights',9000).catch(()=>[]);else articles=await fetchJsonTimeout('/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);}
|
| 50 |
+
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')articles=await fetchJsonTimeout('/api/shorts',9000).catch(()=>[]);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);}
|
| 51 |
+
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();}
|
| 52 |
function shareVid(i){const v=_tikData[i];if(!v)return;doShareVideo(v.title,v.link||'',v.poster||v.img||'','highlights');}
|
| 53 |
+
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);}
|
|
|
|
| 54 |
init();
|
| 55 |
</script>
|
| 56 |
</body>
|