forked from UKSOURCE/cms.hailearning.edu.vn
231 lines
9.3 KiB
Plaintext
231 lines
9.3 KiB
Plaintext
<div class="container">
|
|
<div class="d-flex justify-content-between align-items-center mt-4 mb-4">
|
|
<div>
|
|
<h1 class="h3 mb-0" style="color: var(--primary-dark)">
|
|
Homepage Management
|
|
</h1>
|
|
<p class="text-muted mb-0">Edit content displayed on homepage</p>
|
|
</div>
|
|
<div>
|
|
<a href="<%= frontendUrl %>" class="btn btn-outline-primary" target="_blank">
|
|
<i class="fas fa-external-link-alt me-2"></i>View Homepage
|
|
</a>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="row">
|
|
<div class="col-12">
|
|
<form action="/admin/home/update" method="POST" class="content-with-fixed-buttons">
|
|
<!-- Hidden inputs for JSON data -->
|
|
<input type="hidden" name="hero" id="heroJson" />
|
|
<input type="hidden" name="whyChooseUs" id="whyChooseUsJson" />
|
|
<input type="hidden" name="visaSolutions" id="visaSolutionsJson" />
|
|
<input type="hidden" name="visaCountries" id="visaCountriesJson" />
|
|
<input type="hidden" name="testimonials" id="testimonialsJson" />
|
|
<input type="hidden" name="videoGallery" id="videoGalleryJson" />
|
|
<input type="hidden" name="faq" id="faqJson" />
|
|
<input type="hidden" name="achievements" id="achievementsJson" />
|
|
<input type="hidden" name="partners" id="partnersJson" />
|
|
<input type="hidden" name="blogPreview" id="blogPreviewJson" />
|
|
|
|
<!-- Navigation Tabs -->
|
|
<div class="card shadow-sm border-0 mb-4">
|
|
<div class="card-header bg-white border-bottom">
|
|
<ul class="nav nav-tabs card-header-tabs" role="tablist">
|
|
<li class="nav-item">
|
|
<a class="nav-link active" data-bs-toggle="tab" href="#hero" role="tab">
|
|
<i class="fas fa-home me-2"></i>Hero
|
|
</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" data-bs-toggle="tab" href="#whychooseus" role="tab">
|
|
<i class="fas fa-star me-2"></i>Why Choose Us
|
|
</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" data-bs-toggle="tab" href="#visasolutions" role="tab">
|
|
<i class="fas fa-concierge-bell me-2"></i>Visa Solutions
|
|
</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" data-bs-toggle="tab" href="#visacountries" role="tab">
|
|
<i class="fas fa-globe-americas me-2"></i>Visa Countries
|
|
</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" data-bs-toggle="tab" href="#testimonials" role="tab">
|
|
<i class="fas fa-comments me-2"></i>Testimonials
|
|
</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" data-bs-toggle="tab" href="#videogallery" role="tab">
|
|
<i class="fas fa-video me-2"></i>Video Gallery
|
|
</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" data-bs-toggle="tab" href="#faq" role="tab">
|
|
<i class="fas fa-question-circle me-2"></i>FAQ
|
|
</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" data-bs-toggle="tab" href="#achievements" role="tab">
|
|
<i class="fas fa-chart-pie me-2"></i>Achievements
|
|
</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" data-bs-toggle="tab" href="#partners" role="tab">
|
|
<i class="fas fa-handshake me-2"></i>Partners
|
|
</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" data-bs-toggle="tab" href="#blogpreview" role="tab">
|
|
<i class="fas fa-blog me-2"></i>Blog Preview
|
|
</a>
|
|
</li>
|
|
</ul>
|
|
</div>
|
|
|
|
<div class="card-body">
|
|
<div class="tab-content">
|
|
<%- include('sections/hero') %>
|
|
<%- include('sections/whyChooseUs') %>
|
|
<%- include('sections/visaSolutions') %>
|
|
<%- include('sections/visaCountries') %>
|
|
<%- include('sections/testimonials') %>
|
|
<%- include('sections/videoGallery') %>
|
|
<%- include('sections/faq') %>
|
|
<%- include('sections/achievements') %>
|
|
<%- include('sections/partners') %>
|
|
<%- include('sections/blogPreview') %>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Move buttons to fixed bottom -->
|
|
<div class="fixed-bottom-buttons">
|
|
<button type="reset" class="btn btn-secondary">
|
|
<i class="fas fa-undo"></i>
|
|
<span>Reset</span>
|
|
</button>
|
|
<button type="submit" class="btn btn-primary">
|
|
<i class="fas fa-save"></i>
|
|
<span>Save Changes</span>
|
|
</button>
|
|
</div>
|
|
</form>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Image upload input -->
|
|
<input type="file" id="directImageUpload" style="display: none" />
|
|
<input type="hidden" id="currentImageType" name="imageType" />
|
|
<input type="hidden" id="currentTargetInput" name="targetInput" />
|
|
|
|
<script>
|
|
/**
|
|
* BRIDGE SCRIPT: Cho phép các section lẻ tự đăng ký logic lấy dữ liệu.
|
|
* Cách dùng trong file lẻ (vị dụ hero.ejs):
|
|
* <script>
|
|
* window.homeScrapers = window.homeScrapers || {};
|
|
* window.homeScrapers.hero = () => ({ title: document.getElementById('heroTitle').value, ... });
|
|
* <\/script>
|
|
*/
|
|
window.homeScrapers = window.homeScrapers || {};
|
|
|
|
document.addEventListener("DOMContentLoaded", function () {
|
|
const form = document.querySelector("form");
|
|
if (form) {
|
|
form.addEventListener("submit", function(e) {
|
|
console.log("Form submitting, collecting data from scrapers...");
|
|
|
|
// Tự động thu gom dữ liệu từ các section đã đăng ký
|
|
Object.keys(window.homeScrapers).forEach(section => {
|
|
const input = document.getElementById(section + 'Json');
|
|
if (input) {
|
|
try {
|
|
const data = window.homeScrapers[section]();
|
|
console.log(`- Collected data for [${section}]:`, data);
|
|
input.value = JSON.stringify(data);
|
|
} catch (err) {
|
|
console.error(`Error scraping section [${section}]:`, err);
|
|
}
|
|
}
|
|
});
|
|
|
|
// Để form tự submit tự nhiên sau khi đã điền xong các hidden inputs
|
|
});
|
|
}
|
|
|
|
// Khởi tạo các nút upload ảnh (dùng chung cho toàn bộ các section)
|
|
initImageUploads();
|
|
});
|
|
|
|
// --- UTILITIES (Dùng chung) ---
|
|
|
|
function initImageUploads() {
|
|
document.addEventListener("click", function(e) {
|
|
const btn = e.target.closest(".btn-upload-image");
|
|
if (btn) {
|
|
document.getElementById("currentImageType").value = btn.dataset.imageType;
|
|
document.getElementById("currentTargetInput").value = btn.dataset.targetInput;
|
|
document.getElementById("directImageUpload").click();
|
|
}
|
|
});
|
|
|
|
const fileInput = document.getElementById("directImageUpload");
|
|
if (fileInput) {
|
|
fileInput.addEventListener("change", handleDirectImageUpload);
|
|
}
|
|
}
|
|
|
|
async function handleDirectImageUpload(e) {
|
|
if (!this.files || !this.files[0]) return;
|
|
const file = this.files[0];
|
|
const imageType = document.getElementById("currentImageType").value;
|
|
const targetInput = document.getElementById("currentTargetInput").value;
|
|
|
|
try {
|
|
const formData = new FormData();
|
|
formData.append("image", file);
|
|
const response = await fetch(`/admin/upload/image?imageType=${imageType}`, { method: "POST", body: formData });
|
|
const result = await response.json();
|
|
|
|
if (result.success && result.path) {
|
|
const input = document.getElementById(targetInput);
|
|
if (input) {
|
|
input.value = result.path;
|
|
// Cập nhật preview nếu có img ngay sau input group
|
|
const previewImg = input.closest('.input-group')?.nextElementSibling?.querySelector('img');
|
|
if (previewImg) {
|
|
previewImg.src = result.path;
|
|
previewImg.classList.remove('d-none');
|
|
}
|
|
}
|
|
showToast("Success", "Image uploaded successfully", "success");
|
|
} else {
|
|
throw new Error(result.error || "Upload failed");
|
|
}
|
|
} catch (error) {
|
|
showToast("Error", "Upload failed: " + error.message, "error");
|
|
}
|
|
this.value = "";
|
|
}
|
|
|
|
function showToast(title, message, type = "info") {
|
|
let container = document.querySelector(".toast-container") || (() => {
|
|
const c = document.createElement("div");
|
|
c.className = "toast-container position-fixed top-0 end-0 p-3";
|
|
document.body.appendChild(c);
|
|
return c;
|
|
})();
|
|
|
|
const toast = document.createElement("div");
|
|
toast.className = `toast align-items-center text-white bg-${type === "error" ? "danger" : type} border-0`;
|
|
toast.setAttribute("role", "alert");
|
|
toast.innerHTML = `<div class="d-flex"><div class="toast-body"><strong>${title}:</strong> ${message}</div><button type="button" class="btn-close btn-close-white me-2 m-auto" data-bs-dismiss="toast"></button></div>`;
|
|
container.appendChild(toast);
|
|
new bootstrap.Toast(toast, { autohide: true, delay: 3000 }).show();
|
|
toast.addEventListener("hidden.bs.toast", () => toast.remove());
|
|
}
|
|
</script> |