mirror of
https://github.com/Dvorinka/bizoni.git
synced 2026-06-03 18:22:57 +00:00
update
This commit is contained in:
+312
@@ -0,0 +1,312 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="cs">
|
||||
<head>
|
||||
<meta charset="UTF-8" />
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<title>Nový článek – Bizoni UH</title>
|
||||
<link rel="icon" type="image/x-icon" href="../img/logo.png" />
|
||||
<link rel="stylesheet" href="../css/bootstrap.css" />
|
||||
<link rel="stylesheet" href="../css/bizoni.css" />
|
||||
<link rel="stylesheet" href="../css/admin.css" />
|
||||
<style>
|
||||
body { padding: 24px; max-width: 980px; margin: 0 auto; }
|
||||
header { display:flex; justify-content: space-between; align-items:center; margin-bottom: 16px; }
|
||||
.badge { background: #111827; color: #fff; padding: 6px 10px; border-radius: 999px; font-size: 12px; text-decoration: none; }
|
||||
form { background: #fff; border:1px solid #e5e7eb; border-radius: 10px; padding: 16px; }
|
||||
.row { display: grid; grid-template-columns: 1fr; gap: 12px; }
|
||||
label { font-weight: 600; }
|
||||
input[type="text"], textarea { width: 100%; padding: 10px; border:1px solid #d1d5db; border-radius: 8px; font-size: 14px; }
|
||||
input[type="file"] { padding: 6px; }
|
||||
button { background: #111827; color: #fff; padding: 10px 16px; border: 0; border-radius: 8px; cursor: pointer; }
|
||||
.muted { color: #6b7280; }
|
||||
.result { margin-top: 16px; padding: 12px; border-radius: 8px; display:none; }
|
||||
.ok { background: #ecfdf5; color: #065f46; }
|
||||
.err { background: #fef2f2; color: #991b1b; }
|
||||
.preview { margin-top: 12px; display:flex; gap: 12px; align-items: center; }
|
||||
.preview img { width: 160px; height: 100px; object-fit: cover; border:1px solid #e5e7eb; border-radius: 6px; }
|
||||
.note { background:#FFF7D6; border:1px solid #F7E6A7; padding:10px; border-radius:8px; margin-bottom:12px; }
|
||||
/* Quill tweaks */
|
||||
.ql-container { min-height: 320px; }
|
||||
</style>
|
||||
<!-- Quill (no API key required) -->
|
||||
<link href="https://cdn.jsdelivr.net/npm/quill@1.3.7/dist/quill.snow.css" rel="stylesheet">
|
||||
<script src="https://cdn.jsdelivr.net/npm/quill@1.3.7/dist/quill.min.js"></script>
|
||||
<script src="../js/admin-auth.js"></script>
|
||||
</head>
|
||||
<body class="admin-with-sidenav">
|
||||
<aside class="admin-sidenav">
|
||||
<div class="brand"><img src="../img/logo.png" alt=""/> Bizoni UH</div>
|
||||
<nav>
|
||||
<a href="/admin/dashboard.html">Dashboard</a>
|
||||
<a href="/admin/posts.html">Příspěvky</a>
|
||||
<a class="active" href="/admin/new.html">Nový článek</a>
|
||||
<a href="/admin/index.html">Přehled</a>
|
||||
<a href="/" target="_blank">↗ Zpět na web</a>
|
||||
</nav>
|
||||
<div class="spacer"></div>
|
||||
<div class="footer">Admin</div>
|
||||
</aside>
|
||||
<header>
|
||||
<h1 id="page-title" style="margin:0; font-size: 20px;">Nový článek</h1>
|
||||
<nav style="display:flex; gap:8px;">
|
||||
<a href="/admin/" class="badge">← Přehled</a>
|
||||
<a href="/" class="badge">Domů</a>
|
||||
</nav>
|
||||
</header>
|
||||
|
||||
<!-- Note removed: uploading is supported and handled by the backend -->
|
||||
|
||||
<form id="new-post" enctype="multipart/form-data">
|
||||
<div class="row">
|
||||
<div>
|
||||
<label for="title">Titulek</label>
|
||||
<input type="text" id="title" name="title" placeholder="Např. FC Bizoni vyhráli finále" required />
|
||||
</div>
|
||||
<div>
|
||||
<label for="categories">Kategorie (oddělené čárkou)</label>
|
||||
<input type="text" id="categories" name="categories" placeholder="Zápasy, O nás" />
|
||||
<div class="muted" style="margin-top:6px">Předdefinované:</div>
|
||||
<div class="row" style="gap:12px; align-items:center; grid-template-columns: repeat(auto-fill, minmax(140px,1fr));">
|
||||
<label><input type="checkbox" class="cat-predef" value="Zápasy"/> Zápasy</label>
|
||||
<label><input type="checkbox" class="cat-predef" value="O nás"/> O nás</label>
|
||||
<label><input type="checkbox" class="cat-predef" value="Novinky"/> Novinky</label>
|
||||
</div>
|
||||
</div>
|
||||
<div>
|
||||
<label for="image">Obrázek (.png / .jpg)</label>
|
||||
<input type="file" id="image" name="image" accept="image/png,image/jpeg" />
|
||||
<div class="muted" style="margin-top:6px">Tip: můžete také vložit obrázek přes Ctrl+V nebo načíst z URL.</div>
|
||||
<div class="row" style="grid-template-columns: 1fr auto; align-items: end; gap:8px; margin-top:6px;">
|
||||
<input type="url" id="image-url" placeholder="https://… (URL obrázku)" />
|
||||
<button type="button" id="btn-load-url">Načíst z URL</button>
|
||||
</div>
|
||||
<div class="preview" id="preview" style="display:none">
|
||||
<img id="preview-img" alt="náhled" />
|
||||
<span class="muted" id="preview-name"></span>
|
||||
</div>
|
||||
</div>
|
||||
<div>
|
||||
<label for="editor">Obsah (vizuální editor)</label>
|
||||
<div id="editor"></div>
|
||||
<!-- Hidden textarea to submit HTML (kept focusable-safe by moving offscreen) -->
|
||||
<textarea id="content" name="content" rows="12" style="position:absolute; left:-10000px; width:1px; height:1px; overflow:hidden;"></textarea>
|
||||
<div class="muted">Obsah bude vložen do sekce <code><div class="text lte-text-page clearfix">...</div></code> podle šablony <code>blog/0030.html</code>.</div>
|
||||
</div>
|
||||
<div>
|
||||
<button id="submit-btn" type="submit">Vytvořit článek</button>
|
||||
</div>
|
||||
</div>
|
||||
<!-- hidden id for edit mode -->
|
||||
<input type="hidden" id="post-id" name="id" />
|
||||
</form>
|
||||
|
||||
<div id="result" class="result"></div>
|
||||
|
||||
<script>
|
||||
// Reconcile checkboxes into the text input before submit
|
||||
function reconcileCategories(){
|
||||
const input = document.getElementById('categories');
|
||||
const checks = document.querySelectorAll('.cat-predef');
|
||||
const set = new Set();
|
||||
if (input && input.value.trim()) {
|
||||
input.value.split(',').forEach(s => { const v = s.trim(); if (v) set.add(v); });
|
||||
}
|
||||
checks.forEach(ch => { if (ch.checked) set.add(ch.value); });
|
||||
if (input) input.value = Array.from(set).join(', ');
|
||||
}
|
||||
// Initialize Quill editor
|
||||
const quill = new Quill('#editor', {
|
||||
theme: 'snow',
|
||||
modules: {
|
||||
toolbar: [
|
||||
[{ header: [1, 2, 3, false] }],
|
||||
['bold', 'italic', 'underline', 'strike'],
|
||||
[{ 'color': [] }, { 'background': [] }],
|
||||
[{ 'list': 'ordered'}, { 'list': 'bullet' }],
|
||||
['link', 'blockquote', 'code-block', 'clean']
|
||||
]
|
||||
}
|
||||
});
|
||||
const form = document.getElementById('new-post');
|
||||
const result = document.getElementById('result');
|
||||
const pageTitle = document.getElementById('page-title');
|
||||
const imageInput = document.getElementById('image');
|
||||
const preview = document.getElementById('preview');
|
||||
const previewImg = document.getElementById('preview-img');
|
||||
const previewName = document.getElementById('preview-name');
|
||||
const submitBtn = document.getElementById('submit-btn');
|
||||
const inputUrl = document.getElementById('image-url');
|
||||
const btnLoadUrl = document.getElementById('btn-load-url');
|
||||
const inputId = document.getElementById('post-id');
|
||||
const inputTitle = document.getElementById('title');
|
||||
const inputCats = document.getElementById('categories');
|
||||
|
||||
let pastedBlob = null; // holds clipboard/fetched blob if provided
|
||||
|
||||
function setPreviewFromBlob(blob, name){
|
||||
pastedBlob = blob;
|
||||
preview.style.display = 'flex';
|
||||
previewName.textContent = name || 'Vložený obrázek';
|
||||
const url = URL.createObjectURL(blob);
|
||||
previewImg.onload = () => URL.revokeObjectURL(url);
|
||||
previewImg.src = url;
|
||||
}
|
||||
|
||||
imageInput.addEventListener('change', () => {
|
||||
const file = imageInput.files && imageInput.files[0];
|
||||
if (!file) { preview.style.display = 'none'; return; }
|
||||
pastedBlob = null; // prefer explicit file over pasted blob
|
||||
setPreviewFromBlob(file, file.name);
|
||||
});
|
||||
|
||||
// Paste handler (Ctrl+V) – prefer image from clipboard; fallback to URL text
|
||||
document.addEventListener('paste', async (ev) => {
|
||||
try {
|
||||
const items = ev.clipboardData && ev.clipboardData.items ? Array.from(ev.clipboardData.items) : [];
|
||||
const imgItem = items.find(it => it.type && it.type.startsWith('image/'));
|
||||
if (imgItem) {
|
||||
const blob = imgItem.getAsFile();
|
||||
if (blob) {
|
||||
imageInput.value = '';
|
||||
setPreviewFromBlob(blob, 'vložený obrázek');
|
||||
return;
|
||||
}
|
||||
}
|
||||
const txt = ev.clipboardData && ev.clipboardData.getData ? ev.clipboardData.getData('text/plain') : '';
|
||||
if (txt && /^https?:\/\//i.test(txt)) {
|
||||
inputUrl.value = txt.trim();
|
||||
await (btnLoadUrl.click());
|
||||
}
|
||||
} catch (_) {}
|
||||
});
|
||||
|
||||
// Load from URL handler
|
||||
btnLoadUrl.addEventListener('click', async () => {
|
||||
const u = (inputUrl.value || '').trim();
|
||||
if (!u) return;
|
||||
try {
|
||||
const res = await fetch(u, {mode: 'cors'});
|
||||
if (!res.ok) throw new Error('HTTP '+res.status);
|
||||
const blob = await res.blob();
|
||||
if (!blob.type.startsWith('image/')) throw new Error('URL nevrací obrázek');
|
||||
imageInput.value = '';
|
||||
setPreviewFromBlob(blob, 'z URL');
|
||||
} catch (e) {
|
||||
alert('Nelze načíst obrázek z URL: ' + (e.message || e));
|
||||
}
|
||||
});
|
||||
|
||||
// --- Edit mode support ---
|
||||
const params = new URLSearchParams(location.search);
|
||||
const editId = (params.get('edit') || '').trim();
|
||||
let editMode = false;
|
||||
|
||||
async function loadForEdit(id){
|
||||
try {
|
||||
result.style.display = 'none';
|
||||
const res = await fetch('/api/blog/get?id='+encodeURIComponent(id), { headers: window.AdminAuth ? window.AdminAuth.getHeaders() : {} });
|
||||
if (!res.ok) throw new Error('HTTP '+res.status);
|
||||
const data = await res.json();
|
||||
inputId.value = data.id || id;
|
||||
inputTitle.value = data.title || '';
|
||||
inputCats.value = Array.isArray(data.categories) ? data.categories.join(', ') : '';
|
||||
quill.root.innerHTML = data.content_html || '';
|
||||
// show current image preview
|
||||
preview.style.display = 'flex';
|
||||
previewImg.src = '/img/blog/' + (data.id || id) + '.png';
|
||||
previewName.textContent = 'Aktuální obrázek';
|
||||
// pre-check category checkboxes based on loaded categories
|
||||
const set = new Set((Array.isArray(data.categories)? data.categories : []).map(v => v.toLowerCase()));
|
||||
document.querySelectorAll('.cat-predef').forEach(ch => {
|
||||
ch.checked = set.has(ch.value.toLowerCase());
|
||||
});
|
||||
} catch (e) {
|
||||
console.error(e);
|
||||
result.textContent = 'Chyba načítání článku: ' + (e.message || e);
|
||||
result.className = 'result err';
|
||||
result.style.display = 'block';
|
||||
}
|
||||
}
|
||||
|
||||
if (editId) {
|
||||
editMode = true;
|
||||
pageTitle.textContent = 'Upravit článek ' + editId;
|
||||
submitBtn.textContent = 'Uložit změny';
|
||||
// image optional in edit
|
||||
imageInput.removeAttribute('required');
|
||||
// content still required, but keep UX flexible
|
||||
loadForEdit(editId);
|
||||
}
|
||||
|
||||
form.addEventListener('submit', async (e) => {
|
||||
e.preventDefault();
|
||||
result.style.display = 'none';
|
||||
result.className = 'result';
|
||||
// Move Quill HTML into the hidden textarea
|
||||
// Merge category checkboxes into text input
|
||||
reconcileCategories();
|
||||
const html = quill.root.innerHTML;
|
||||
document.getElementById('content').value = html;
|
||||
// Validate content is not empty (avoid browser required on hidden field)
|
||||
const plain = quill.getText().trim();
|
||||
if (!plain) {
|
||||
result.textContent = 'Vyplňte obsah článku.';
|
||||
result.classList.add('err');
|
||||
result.style.display = 'block';
|
||||
return;
|
||||
}
|
||||
const fd = new FormData(form);
|
||||
// Prefer pasted/fetched blob if present when no file was chosen
|
||||
if (pastedBlob && !(imageInput.files && imageInput.files[0])) {
|
||||
const ext = (pastedBlob.type === 'image/jpeg') ? 'jpg' : 'png';
|
||||
fd.set('image', new File([pastedBlob], 'pasted.'+ext, {type: pastedBlob.type||'image/png'}));
|
||||
}
|
||||
// Validate presence of image (file or pasted/url)
|
||||
if (!fd.get('image')) {
|
||||
result.textContent = 'Přidejte obrázek (soubor, vložení přes Ctrl+V, nebo URL).';
|
||||
result.classList.add('err');
|
||||
result.style.display = 'block';
|
||||
return;
|
||||
}
|
||||
try {
|
||||
let url = '/api/blog/new';
|
||||
if (editMode) {
|
||||
// ensure id is present in payload
|
||||
if (!fd.get('id')) fd.set('id', editId);
|
||||
url = '/api/blog/edit';
|
||||
}
|
||||
const res = await fetch(url, { method: 'POST', body: fd, headers: window.AdminAuth ? window.AdminAuth.getHeaders() : {} });
|
||||
if (!editMode && !res.ok) {
|
||||
const txt = await res.text();
|
||||
throw new Error(txt || ('HTTP '+res.status));
|
||||
}
|
||||
if (editMode && res.status !== 204 && !res.ok) {
|
||||
const txt = await res.text();
|
||||
throw new Error(txt || ('HTTP '+res.status));
|
||||
}
|
||||
// Success UI
|
||||
if (editMode) {
|
||||
result.textContent = 'Uloženo';
|
||||
result.classList.add('ok');
|
||||
result.style.display = 'block';
|
||||
} else {
|
||||
const data = await res.json();
|
||||
result.textContent = `Vytvořeno: ${data.id}`;
|
||||
result.classList.add('ok');
|
||||
result.style.display = 'block';
|
||||
// Offer a link to open the new post
|
||||
const a = document.createElement('a');
|
||||
a.href = data.link; a.target = '_blank'; a.style.marginLeft = '8px'; a.textContent = 'Otevřít';
|
||||
result.appendChild(a);
|
||||
form.reset(); preview.style.display = 'none';
|
||||
quill.setContents([]);
|
||||
}
|
||||
} catch (err) {
|
||||
result.textContent = 'Chyba: ' + (err.message || err);
|
||||
result.classList.add('err');
|
||||
result.style.display = 'block';
|
||||
}
|
||||
});
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
||||
Reference in New Issue
Block a user