many fixes in design and add shirt story type for creation book

This commit is contained in:
mirivlad 2026-05-01 20:51:42 +03:00
parent 9294399517
commit ce6fffaa57
24 changed files with 1167 additions and 735 deletions

View File

@ -1,5 +1,5 @@
/*!
* Quill Editor v2.0.3
* Quill Editor v2.0.2
* https://quilljs.com
* Copyright (c) 2017-2024, Slab
* Copyright (c) 2014, Jason Chen

View File

@ -43,3 +43,6 @@
.writer-editor-container.fullscreen {
transition: all 0.3s ease;
}
.ql-editor p {
text-indent: 2em;
}

View File

@ -1,4 +1,86 @@
document.addEventListener('DOMContentLoaded', () => {
// Поддержка WriterEditor (для глав и рассказов)
const writerEditor = window.writerEditor;
if (writerEditor) {
const textarea = writerEditor.textarea;
if (!textarea) return;
// Для рассказов textarea может быть hidden input
const contentField = textarea.tagName === 'TEXTAREA' ? textarea :
(document.getElementById('story-content') ? document.getElementById('story-content') : textarea);
let lastSavedContent = contentField.value;
let saveTimeout;
function showMessage(message, isError = false) {
let msgEl = document.getElementById('autosave-message');
if (!msgEl) {
msgEl = document.createElement('div');
msgEl.id = 'autosave-message';
msgEl.style.cssText = `
position: fixed;
top: 70px;
right: 10px;
padding: 8px 12px;
background: ${isError ? '#dc3545' : '#28a745'};
color: white;
border-radius: 3px;
z-index: 10000;
font-size: 0.8rem;
box-shadow: 0 2px 5px rgba(0,0,0,0.2);
`;
document.body.appendChild(msgEl);
}
msgEl.textContent = message;
msgEl.style.background = isError ? '#dc3545' : '#28a745';
msgEl.style.display = 'block';
setTimeout(() => msgEl.style.display = 'none', 2000);
}
const autoSave = () => {
const currentContent = contentField.value;
if (currentContent === lastSavedContent) return;
const form = writerEditor.form;
if (!form) return;
const formData = new FormData(form);
formData.append('autosave', 'true');
formData.append('content', currentContent);
showMessage('Сохранение...');
fetch(window.location.href, {
method: 'POST',
body: formData
})
.then(res => {
if (!res.ok) throw new Error(`HTTP ${res.status}`);
return res.json();
})
.then(data => {
if (data.success) {
lastSavedContent = currentContent;
showMessage('Автосохранено: ' + new Date().toLocaleTimeString());
} else {
throw new Error(data.error || 'Ошибка сервера');
}
})
.catch(err => {
console.error(err);
showMessage('Ошибка автосохранения: ' + err.message, true);
});
};
writerEditor.quill.on('text-change', () => {
clearTimeout(saveTimeout);
saveTimeout = setTimeout(autoSave, 2000);
});
// Периодическая автосохранение
setInterval(autoSave, 30000);
} else {
// Старая поддержка через window.quillEditorInstance
const quill = window.quillEditorInstance;
const textarea = window.quillTextarea;
if (!quill || !textarea) return;
@ -38,6 +120,7 @@ document.addEventListener('DOMContentLoaded', () => {
const form = document.getElementById('chapter-form');
const formData = new FormData(form);
formData.append('autosave', 'true');
formData.append('content', currentContent);
showMessage('Сохранение...');
@ -70,4 +153,5 @@ document.addEventListener('DOMContentLoaded', () => {
// Периодическая автосохранение
setInterval(autoSave, 30000);
}
});

View File

@ -1,4 +1,34 @@
// editor.js
// Глобальная функция переключения редактора рассказа
function toggleStoryEditor() {
const checkbox = document.getElementById('is_short_story');
const storyEditorCard = document.getElementById('story-editor-card');
const chaptersCard = document.getElementById('chapters-card');
if (!checkbox || !storyEditorCard) return;
const isChecked = checkbox.checked;
if (isChecked) {
storyEditorCard.style.display = 'block';
chaptersCard.style.display = 'none';
if (!window.writerEditor) {
const textarea = document.getElementById('content');
if (textarea) {
window.writerEditor = new WriterEditor('#book-form', 'story-editor', 'content', true);
}
}
} else {
storyEditorCard.style.display = 'none';
chaptersCard.style.display = 'block';
if (window.writerEditor) {
window.writerEditor.quill.destroy();
window.writerEditor = null;
}
}
}
class DialogueFormatter {
constructor(quill) {
this.quill = quill;
@ -79,10 +109,11 @@ class DialogueFormatter {
}
class WriterEditor {
constructor(formSelector = '#chapter-form', editorContainerId = 'quill-editor', textareaId = 'content') {
constructor(formSelector = '#chapter-form', editorContainerId = 'quill-editor', textareaId = 'content', isShortStory = false) {
this.form = document.querySelector(formSelector);
this.editorContainer = document.getElementById(editorContainerId);
this.textarea = document.getElementById(textareaId);
this.isShortStory = isShortStory;
this.isFullscreen = false;
this.originalStyles = {};
this.init();
@ -91,11 +122,16 @@ class WriterEditor {
init() {
if (!this.editorContainer || !this.textarea || !this.form) return;
this.quill = new Quill(this.editorContainer, {
theme: 'snow',
modules: {
toolbar: {
container: [
// Упрощённый тулбар для рассказов
const toolbarOptions = this.isShortStory
? [
[{ 'header': [1, 2, 3, false] }],
['bold', 'italic', 'underline'],
[{ 'list': 'ordered' }, { 'list': 'bullet' }],
['link', 'blockquote', 'code-block'],
['clean']
]
: [
[{ 'header': [1, 2, 3, 4, 5, 6, false] }],
['bold','italic','underline','strike'],
[{ 'align': [] }],
@ -109,8 +145,14 @@ class WriterEditor {
[{ 'font': [] }],
['link','image','video'],
['clean'],
['fullscreen'] // Добавляем кнопку полноэкранного режима
],
['fullscreen']
];
this.quill = new Quill(this.editorContainer, {
theme: 'snow',
modules: {
toolbar: {
container: toolbarOptions,
handlers: {
'dialogue': () => {
if (this.dialogueFormatter) {
@ -142,17 +184,18 @@ class WriterEditor {
}
}
}
},
placeholder: 'Введите текст главы...'
}, // Упрощённый тулбар для рассказов
placeholder: this.isShortStory ? 'Напишите здесь ваш рассказ...' : 'Введите текст главы...'
});
this.addCustomButtonsToToolbar();
this.dialogueFormatter = new DialogueFormatter(this.quill);
this.dialogueFormatter = this.isShortStory ? null : new DialogueFormatter(this.quill);
const rawContent = this.editorContainer.dataset.content || '';
if (rawContent.trim()) this.quill.root.innerHTML = rawContent.trim();
if (!this.isShortStory) {
setTimeout(() => this.formatExistingDialogues(), 100);
}
const sync = () => {
let html = this.quill.root.innerHTML;
@ -161,7 +204,10 @@ class WriterEditor {
};
this.quill.on('text-change', sync);
this.form.addEventListener('submit', sync);
const submitHandler = function(e) {
sync();
};
this.form.addEventListener('submit', submitHandler);
// Обработчик изменения ориентации для мобильных устройств
window.addEventListener('orientationchange', () => {
@ -349,8 +395,47 @@ class WriterEditor {
totalOffset += line.length + 1;
});
}
addCustomButtonsToToolbar() {
if (this.isShortStory) return;
const toolbar = this.quill.container.previousSibling;
if (!toolbar) return;
const dialogueBtn = toolbar.querySelector('.ql-dialogue');
const undoDialogueBtn = toolbar.querySelector('.ql-undodialogue');
const fullscreenBtn = toolbar.querySelector('.ql-fullscreen');
if (dialogueBtn) {
dialogueBtn.innerHTML = '—';
dialogueBtn.title = 'Форматировать диалоги (—)';
dialogueBtn.style.fontWeight = 'bold';
}
if (undoDialogueBtn) {
undoDialogueBtn.innerHTML = '-';
undoDialogueBtn.title = 'Убрать форматирование диалогов (-)';
undoDialogueBtn.style.fontWeight = 'bold';
}
if (fullscreenBtn) {
fullscreenBtn.innerHTML = '⛶';
fullscreenBtn.title = 'Полноэкранный режим';
}
}
}
document.addEventListener('DOMContentLoaded', () => {
// Проверяем, это редактор рассказа или главы
const storyEditor = document.getElementById('story-editor');
const quillEditor = document.getElementById('quill-editor');
if (storyEditor) {
const textarea = document.getElementById('story-content');
if (textarea) {
window.writerEditor = new WriterEditor('#book-form', 'story-editor', 'story-content', true);
}
} else if (quillEditor) {
window.writerEditor = new WriterEditor();
}
});

9
assets/js/quill.min.js vendored Normal file

File diff suppressed because one or more lines are too long

View File

View File

@ -19,8 +19,12 @@ class AuthController extends BaseController {
} else {
$username = trim($_POST['username'] ?? '');
$password = $_POST['password'] ?? '';
$ip = $_SERVER['REMOTE_ADDR'] ?? '0.0.0.0';
if (empty($username) || empty($password)) {
// Проверка Rate Limit
if (!checkRateLimit($this->pdo, $ip)) {
$error = 'Слишком много попыток входа. Попробуйте позже.';
} elseif (empty($username) || empty($password)) {
$error = 'Пожалуйста, введите имя пользователя и пароль';
} else {
$userModel = new User($this->pdo);
@ -29,8 +33,12 @@ class AuthController extends BaseController {
if ($user && $userModel->verifyPassword($password, $user['password_hash'])) {
if (!$user['is_active']) {
$error = 'Ваш аккаунт деактивирован или ожидает активации администратором.';
logLoginAttempt($this->pdo, $ip, $username);
} else {
// Успешный вход
// Успешный вход - очищаем попытки
clearLoginAttempts($this->pdo, $ip);
// Обновляем сессию
session_regenerate_id(true);
$_SESSION['user_id'] = $user['id'];
$_SESSION['username'] = $user['username'];
@ -45,6 +53,7 @@ class AuthController extends BaseController {
}
} else {
$error = 'Неверное имя пользователя или пароль';
logLoginAttempt($this->pdo, $ip, $username);
}
}
}

View File

@ -36,6 +36,16 @@ class BookController extends BaseController {
$this->redirect('/books/create');
}
$is_short_story = isset($_POST['is_short_story']) ? 1 : 0;
$content = $is_short_story ? ($_POST['content'] ?? '') : null;
// Валидация: для рассказа проверяем, что есть хоть какое-то содержание
if ($is_short_story && empty(trim(strip_tags($content)))) {
$_SESSION['error'] = "Для рассказа необходимо заполнить содержание";
$_SESSION['form_data'] = $_POST;
$this->redirect('/books/create');
}
$bookModel = new Book($this->pdo);
$data = [
'title' => $title,
@ -44,7 +54,9 @@ class BookController extends BaseController {
'user_id' => $_SESSION['user_id'],
'series_id' => !empty($_POST['series_id']) ? (int)$_POST['series_id'] : null,
'sort_order_in_series' => !empty($_POST['sort_order_in_series']) ? (int)$_POST['sort_order_in_series'] : null,
'published' => isset($_POST['published']) ? 1 : 0
'published' => isset($_POST['published']) ? 1 : 0,
'is_short_story' => $is_short_story,
'content' => $content
];
if ($bookModel->create($data)) {
@ -62,8 +74,15 @@ class BookController extends BaseController {
}
}
$_SESSION['success'] = "Книга успешно создана" . ($cover_error ? ", но возникла ошибка с обложкой: " . $cover_error : "");
// Если это рассказ с содержанием - редиректим на редактирование
// Если рассказ без содержания - тоже на редактирование (чтобы добавить текст)
if ($is_short_story) {
$_SESSION['success'] = "Рассказ успешно создан! Теперь можно добавить содержание.";
$this->redirect("/books/{$new_book_id}/edit");
} else {
$_SESSION['success'] = "Книга успешно создана" . ($cover_error ? ", но возникла ошибка с обложкой: " . $cover_error : "");
$this->redirect("/books/{$new_book_id}/chapters/create");
}
} else {
$_SESSION['error'] = "Ошибка при создании книги";
}
@ -91,11 +110,63 @@ class BookController extends BaseController {
$seriesModel = new Series($this->pdo);
$series = $seriesModel->findByUser($_SESSION['user_id']);
$error = '';
$cover_error = '';
if ($_SERVER['REQUEST_METHOD'] === 'POST') {
// Обработка запроса автосохранения
if (!empty($_POST['autosave']) && $_POST['autosave'] === 'true') {
header('Content-Type: application/json');
// Подавляем ошибки для автосейва
error_reporting(0);
if (!verify_csrf_token($_POST['csrf_token'] ?? '')) {
echo json_encode(['success' => false, 'error' => 'Ошибка безопасности']);
exit;
}
$book = $bookModel->findById($id);
if (!$book || $book['user_id'] != $_SESSION['user_id']) {
echo json_encode(['success' => false, 'error' => 'Книга не найдена']);
exit;
}
$is_short_story = isset($_POST['is_short_story']) ? 1 : 0;
$content = $is_short_story ? ($_POST['content'] ?? '') : null;
// Для autosave обновляем только title и user_id из сессии, остальное получаем из базы
$title = trim($_POST['title'] ?? $book['title']);
$description = trim($_POST['description'] ?? ($book['description'] ?? ''));
$genre = trim($_POST['genre'] ?? ($book['genre'] ?? ''));
$user_id = $_SESSION['user_id'];
$series_id = !empty($_POST['series_id']) ? (int)$_POST['series_id'] : $book['series_id'];
$sort_order_in_series = !empty($_POST['sort_order_in_series']) ? (int)$_POST['sort_order_in_series'] : $book['sort_order_in_series'];
$published = isset($_POST['published']) ? 1 : $book['published'];
$data = [
'title' => $title,
'description' => $description,
'genre' => $genre,
'user_id' => $user_id,
'series_id' => $series_id,
'sort_order_in_series' => $sort_order_in_series,
'published' => $published,
'is_short_story' => $is_short_story,
'content' => $content
];
$success = $bookModel->update($id, $data);
header('Content-Type: application/json');
if ($success) {
echo json_encode(['success' => true]);
} else {
echo json_encode(['success' => false, 'error' => 'Ошибка при сохранении']);
}
exit;
}
if (!verify_csrf_token($_POST['csrf_token'] ?? '')) {
$error = "Ошибка безопасности";
} else {
@ -103,6 +174,9 @@ class BookController extends BaseController {
if (empty($title)) {
$error = "Название книги обязательно";
} else {
$is_short_story = isset($_POST['is_short_story']) ? 1 : 0;
$content = $is_short_story ? ($_POST['content'] ?? '') : null;
$data = [
'title' => $title,
'description' => trim($_POST['description'] ?? ''),
@ -110,7 +184,9 @@ class BookController extends BaseController {
'user_id' => $_SESSION['user_id'],
'series_id' => !empty($_POST['series_id']) ? (int)$_POST['series_id'] : null,
'sort_order_in_series' => !empty($_POST['sort_order_in_series']) ? (int)$_POST['sort_order_in_series'] : null,
'published' => isset($_POST['published']) ? 1 : 0
'published' => isset($_POST['published']) ? 1 : 0,
'is_short_story' => $is_short_story,
'content' => $content
];
// Обработка обложки
@ -252,7 +328,15 @@ class BookController extends BaseController {
$this->render('errors/404');
return;
}
// Если это рассказ - не загружаем главы
if ($book['is_short_story']) {
$chapters = [];
$is_short_story = true;
} else {
$chapters = $chapterModel->getPublishedChapters($book['id']);
$is_short_story = false;
}
// Получаем информацию об авторе
$stmt = $this->pdo->prepare("SELECT id, username, display_name FROM users WHERE id = ?");
@ -263,6 +347,7 @@ class BookController extends BaseController {
'book' => $book,
'chapters' => $chapters,
'author' => $author,
'is_short_story' => $is_short_story,
'page_title' => $book['title']
]);
}
@ -279,7 +364,15 @@ class BookController extends BaseController {
$this->render('errors/404');
return;
}
// Если это рассказ - не загружаем главы
if ($book['is_short_story']) {
$chapters = [];
$is_short_story = true;
} else {
$chapters = $chapterModel->findByBook($book['id']);
$is_short_story = false;
}
// Получаем информацию об авторе
$stmt = $this->pdo->prepare("SELECT id, username, display_name FROM users WHERE id = ?");
@ -290,6 +383,7 @@ class BookController extends BaseController {
'book' => $book,
'chapters' => $chapters,
'author' => $author,
'is_short_story' => $is_short_story,
'page_title' => $book['title']
]);
}

View File

@ -26,13 +26,19 @@ class ExportController extends BaseController {
$this->redirect('/books');
}
// Для автора - все главы
// Для автора - все главы или контент рассказа
if ($book['is_short_story']) {
$chapters = [];
$is_short_story = true;
} else {
$chapters = $chapterModel->findByBook($book_id);
$is_short_story = false;
}
// Получаем информацию об авторе
$author_name = $this->getAuthorName($book['user_id']);
$this->handleExport($book, $chapters, false, $author_name, $format);
$this->handleExport($book, $chapters, false, $author_name, $format, $is_short_story);
}
public function exportShared($share_token, $format = 'pdf') {
@ -45,13 +51,19 @@ class ExportController extends BaseController {
$this->redirect('/');
}
// Для публичного доступа - только опубликованные главы
// Для публичного доступа - только опубликованные главы или контент рассказа
if ($book['is_short_story']) {
$chapters = [];
$is_short_story = true;
} else {
$chapters = $chapterModel->getPublishedChapters($book['id']);
$is_short_story = false;
}
// Получаем информацию об авторе
$author_name = $this->getAuthorName($book['user_id']);
$this->handleExport($book, $chapters, true, $author_name, $format);
$this->handleExport($book, $chapters, true, $author_name, $format, $is_short_story);
}
private function getAuthorName($user_id) {
@ -68,21 +80,20 @@ class ExportController extends BaseController {
return "Неизвестный автор";
}
private function handleExport($book, $chapters, $is_public, $author_name, $format) {
private function handleExport($book, $chapters, $is_public, $author_name, $format, $is_short_story = false) {
switch ($format) {
case 'pdf':
$this->exportPDF($book, $chapters, $is_public, $author_name);
$this->exportPDF($book, $chapters, $is_public, $author_name, $is_short_story);
break;
case 'docx':
$this->exportDOCX($book, $chapters, $is_public, $author_name);
$this->exportDOCX($book, $chapters, $is_public, $author_name, $is_short_story);
break;
case 'html':
$this->exportHTML($book, $chapters, $is_public, $author_name);
$this->exportHTML($book, $chapters, $is_public, $author_name, $is_short_story);
break;
case 'txt':
$this->exportTXT($book, $chapters, $is_public, $author_name);
$this->exportTXT($book, $chapters, $is_public, $author_name, $is_short_story);
break;
default:
$_SESSION['error'] = "Неверный формат экспорта";
@ -93,8 +104,7 @@ class ExportController extends BaseController {
}
}
function exportPDF($book, $chapters, $is_public, $author_name) {
function exportPDF($book, $chapters, $is_public, $author_name, $is_short_story = false) {
$pdf = new TCPDF(PDF_PAGE_ORIENTATION, PDF_UNIT, PDF_PAGE_FORMAT, true, 'UTF-8', false);
@ -157,9 +167,18 @@ class ExportController extends BaseController {
$pdf->Ln(10);
}
// Интерактивное оглавление
// Если это рассказ - сразу выводим контент
if ($is_short_story && !empty($book['content'])) {
$pdf->SetFont('dejavusans', '', 11);
$pdf->writeHTML($book['content'], true, false, true, false, '');
$pdf->Ln(8);
}
// Интерактивное оглавление (только для обычных книг)
if (!$is_short_story) {
$chapterLinks = [];
if (!empty($chapters)) {
if (!empty($chapters)) {
$pdf->SetFont('dejavusans', 'B', 14);
$pdf->Cell(0, 10, 'Оглавление', 0, 1, 'C');
$pdf->Ln(5);
@ -220,7 +239,7 @@ class ExportController extends BaseController {
exit;
}
function exportDOCX($book, $chapters, $is_public, $author_name) {
function exportDOCX($book, $chapters, $is_public, $author_name, $is_short_story = false) {
$phpWord = new PhpWord();
@ -271,8 +290,19 @@ class ExportController extends BaseController {
$section->addTextBreak(2);
}
// Интерактивное оглавление
if (!empty($chapters)) {
// Если это рассказ - сразу выводим контент
if ($is_short_story && !empty($book['content'])) {
$paragraphs = $this->htmlToParagraphs($book['content']);
foreach ($paragraphs as $paragraph) {
if (!empty(trim($paragraph))) {
$section->addText($paragraph);
$section->addTextBreak(1);
}
}
}
// Интерактивное оглавление (только для обычных книг)
if (!$is_short_story && !empty($chapters)) {
$section->addText('Оглавление', ['bold' => true, 'size' => 14], ['alignment' => 'center']);
$section->addTextBreak(1);
@ -286,9 +316,12 @@ class ExportController extends BaseController {
}
// Разделитель
if (!$is_short_story) {
$section->addPageBreak();
}
// Главы с закладками
// Главы с закладками (только для обычных книг)
if (!$is_short_story) {
foreach ($chapters as $index => $chapter) {
// Добавляем закладку для главы
$section->addBookmark("chapter_{$chapter['id']}");
@ -332,7 +365,7 @@ class ExportController extends BaseController {
exit;
}
function exportHTML($book, $chapters, $is_public, $author_name) {
function exportHTML($book, $chapters, $is_public, $author_name, $is_short_story = false) {
$html = '<!DOCTYPE html>
<html lang="ru">
@ -514,8 +547,13 @@ class ExportController extends BaseController {
$html .= '</div>';
}
// Интерактивное оглавление
if (!empty($chapters)) {
// Если это рассказ - выводим контент сразу
if ($is_short_story && !empty($book['content'])) {
$html .= '<div class="chapter-content">' . $book['content'] . '</div>';
}
// Интерактивное оглавление (только для обычных книг)
if (!$is_short_story && !empty($chapters)) {
$html .= '<div class="table-of-contents">';
$html .= '<h3>Оглавление</h3>';
$html .= '<ul>';
@ -554,7 +592,7 @@ class ExportController extends BaseController {
exit;
}
function exportTXT($book, $chapters, $is_public, $author_name) {
function exportTXT($book, $chapters, $is_public, $author_name, $is_short_story = false) {
$content = "=" . str_repeat("=", 80) . "=\n";
$content .= str_pad($book['title'], 80, " ", STR_PAD_BOTH) . "\n";
$content .= str_pad($author_name, 80, " ", STR_PAD_BOTH) . "\n";
@ -578,8 +616,25 @@ class ExportController extends BaseController {
$content .= "\n";
}
// Оглавление
if (!empty($chapters)) {
// Если это рассказ - выводим контент сразу
if ($is_short_story && !empty($book['content'])) {
$content .= "СОДЕРЖАНИЕ РАССКАЗА:\n";
$content .= str_repeat("-", 60) . "\n\n";
$plainText = $this->htmlToPlainText($book['content']);
$paragraphs = explode("\n", $plainText);
foreach ($paragraphs as $paragraph) {
$trimmed = trim($paragraph);
if (!empty($trimmed)) {
$wrapped = wordwrap($trimmed, 144);
$content .= $wrapped . "\n\n";
}
}
}
// Оглавление (только для обычных книг)
if (!$is_short_story && !empty($chapters)) {
$content .= "ОГЛАВЛЕНИЕ:\n";
$content .= str_repeat("-", 60) . "\n";
foreach ($chapters as $index => $chapter) {
@ -591,6 +646,8 @@ class ExportController extends BaseController {
$content .= str_repeat("-", 144) . "\n\n";
// Главы (только для обычных книг)
if (!$is_short_story) {
foreach ($chapters as $index => $chapter) {
$content .= $chapter['title'] . "\n";
$content .= str_repeat("-", 60) . "\n\n";

View File

@ -352,4 +352,25 @@ function deleteUserAvatar($user_id) {
return true;
}
function checkRateLimit($pdo, $ip, $max_attempts = 5, $timeframe = 60) {
$stmt = $pdo->prepare("
SELECT COUNT(*) FROM login_attempts
WHERE ip_address = ? AND attempted_at > NOW() - INTERVAL ? SECOND
");
$stmt->execute([$ip, $timeframe]);
$count = $stmt->fetchColumn();
return $count < $max_attempts;
}
function logLoginAttempt($pdo, $ip, $username) {
$stmt = $pdo->prepare("INSERT INTO login_attempts (ip_address, username) VALUES (?, ?)");
return $stmt->execute([$ip, $username]);
}
function clearLoginAttempts($pdo, $ip) {
$stmt = $pdo->prepare("DELETE FROM login_attempts WHERE ip_address = ?");
return $stmt->execute([$ip]);
}
?>

View File

@ -1,6 +1,6 @@
<?php
// index.php - единая точка входа
require_once 'config/config.php';
require_once __DIR__ . '/../config/config.php';
// Получаем путь к запрашиваемому ресурсу
$requestUri = $_SERVER['REQUEST_URI'];
@ -53,6 +53,7 @@ if (pathinfo($physicalPath, PATHINFO_EXTENSION) != 'php') {
// Простой роутер
class Router {
private $routes = [];
public $params = [];
public function add($pattern, $handler) {
$this->routes[$pattern] = $handler;

View File

@ -61,6 +61,8 @@ CREATE TABLE `books` (
`updated_at` timestamp NOT NULL DEFAULT current_timestamp() ON UPDATE current_timestamp(),
`share_token` varchar(32) DEFAULT NULL,
`published` tinyint(1) NOT NULL DEFAULT 0,
`is_short_story` tinyint(1) NOT NULL DEFAULT 0,
`content` text DEFAULT NULL,
PRIMARY KEY (`id`),
UNIQUE KEY `share_token` (`share_token`),
KEY `user_id` (`user_id`),
@ -198,32 +200,27 @@ if ($_SERVER['REQUEST_METHOD'] === 'POST') {
}
function generate_config($db) {
$site_url = (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] === 'on' ? "https" : "http") . "://$_SERVER[HTTP_HOST]";
$base_path = str_replace('/install.php', '', $_SERVER['PHP_SELF']);
$site_url .= $base_path;
return <<<EOT
<?php
// config/config.php - автоматически сгенерирован установщиком
// Подключаем функции
require_once __DIR__ . '/../includes/functions.php';
session_start();
require_once __DIR__ . '/../public/includes/functions.php';
// Настройки базы данных
define('DB_HOST', '{$db['db_host']}');
define('DB_USER', '{$db['db_user']}');
define('DB_PASS', '{$db['db_pass']}');
define('DB_NAME', '{$db['db_name']}');
define('SITE_URL', '{$site_url}');
}
// динамическое определение SITE_URL
$protocol = (!empty($_SERVER['HTTPS']) && $_SERVER['HTTPS'] !== 'off') ? 'https://' : 'http://';
$host = $_SERVER['HTTP_HOST'] ?? 'localhost';
define('SITE_URL', $protocol . $host);
// Настройки приложения
define('APP_NAME', 'Web Writer');
define('CONTROLLERS_PATH', __DIR__ . '/../controllers/');
define('VIEWS_PATH', __DIR__ . '/../views/');
define('LAYOUTS_PATH', VIEWS_PATH . 'layouts/');
define('UPLOAD_PATH', __DIR__ . '/../uploads/');
define('UPLOAD_PATH', __DIR__ . '/../public/uploads/');
define('COVERS_PATH', UPLOAD_PATH . 'covers/');
define('COVERS_URL', SITE_URL . '/uploads/covers/');
define('AVATARS_PATH', UPLOAD_PATH . 'avatars/');
@ -237,25 +234,28 @@ if (!file_exists(AVATARS_PATH)) {
mkdir(AVATARS_PATH, 0755, true);
}
// Подключение к базе данных
try {
\$pdo = new PDO("mysql:host=" . DB_HOST . ";dbname=" . DB_NAME . ";charset=utf8mb4", DB_USER, DB_PASS);
\$pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
} catch(PDOException \$e) {
error_log("DB Error: " . \$e->getMessage());
$pdo = new PDO("mysql:host=" . DB_HOST . ";dbname=" . DB_NAME . ";charset=utf8mb4", DB_USER, DB_PASS);
$pdo->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
} catch(PDOException $e) {
error_log("DB Error: " . $e->getMessage());
die("Ошибка подключения к базе данных");
}
// Добавляем константы для новых путей
define('CONTROLLERS_PATH', __DIR__ . '/../public/controllers/');
define('VIEWS_PATH', __DIR__ . '/../public/views/');
define('LAYOUTS_PATH', VIEWS_PATH . 'layouts/');
// Автозагрузка моделей
spl_autoload_register(function (\$class_name) {
\$model_file = __DIR__ . '/../models/' . \$class_name . '.php';
if (file_exists(\$model_file)) {
require_once \$model_file;
// Автозагрузка контроллеров
spl_autoload_register(function ($class_name) {
$controller_file = CONTROLLERS_PATH . $class_name . '.php';
if (file_exists($controller_file)) {
require_once $controller_file;
}
});
?>
EOT;
}
?>

View File

@ -28,7 +28,11 @@ class Book {
$sql = "
SELECT b.*,
COUNT(c.id) as chapter_count,
COALESCE(SUM(c.word_count), 0) as total_words
COALESCE(
CASE WHEN b.is_short_story = 1
THEN CHAR_LENGTH(b.content) - CHAR_LENGTH(REPLACE(b.content, ' ', '')) + 1
ELSE COALESCE(SUM(c.word_count), 0) END, 0
) as total_words
FROM books b
LEFT JOIN chapters c ON b.id = c.book_id
WHERE b.user_id = ?
@ -45,10 +49,12 @@ class Book {
public function create($data) {
$share_token = bin2hex(random_bytes(16));
$published = isset($data['published']) ? (int)$data['published'] : 0;
$is_short_story = isset($data['is_short_story']) ? (int)$data['is_short_story'] : 0;
$content = $is_short_story ? ($data['content'] ?? null) : null;
$stmt = $this->pdo->prepare("
INSERT INTO books (title, description, genre, user_id, series_id, sort_order_in_series, share_token, published)
VALUES (?, ?, ?, ?, ?, ?, ?, ?)
INSERT INTO books (title, description, genre, user_id, series_id, sort_order_in_series, share_token, published, is_short_story, content)
VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?, ?)
");
return $stmt->execute([
$data['title'],
@ -58,12 +64,16 @@ class Book {
$data['series_id'] ?? null,
$data['sort_order_in_series'] ?? null,
$share_token,
$published
$published,
$is_short_story,
$content
]);
}
public function update($id, $data) {
$published = isset($data['published']) ? (int)$data['published'] : 0;
$is_short_story = isset($data['is_short_story']) ? (int)$data['is_short_story'] : 0;
$content = $is_short_story ? ($data['content'] ?? null) : null;
// Преобразуем пустые строки в NULL для integer полей
$series_id = !empty($data['series_id']) ? (int)$data['series_id'] : null;
@ -71,7 +81,7 @@ class Book {
$stmt = $this->pdo->prepare("
UPDATE books
SET title = ?, description = ?, genre = ?, series_id = ?, sort_order_in_series = ?, published = ?
SET title = ?, description = ?, genre = ?, series_id = ?, sort_order_in_series = ?, published = ?, is_short_story = ?, content = ?
WHERE id = ? AND user_id = ?
");
return $stmt->execute([
@ -81,6 +91,8 @@ class Book {
$series_id, // Теперь это либо integer, либо NULL
$sort_order_in_series, // Теперь это либо integer, либо NULL
$published,
$is_short_story,
$content,
$id,
$data['user_id']
]);
@ -200,17 +212,17 @@ class Book {
public function getBookStats($book_id, $only_published_chapters = false) {
$sql = "
SELECT
COUNT(c.id) as chapter_count,
COALESCE(SUM(c.word_count), 0) as total_words
b.is_short_story,
(SELECT COUNT(*) FROM chapters c WHERE c.book_id = b.id " . ($only_published_chapters ? "AND c.status = 'published'" : "") . ") as chapter_count,
CASE
WHEN b.is_short_story = 1 AND b.content IS NOT NULL THEN CHAR_LENGTH(b.content) - CHAR_LENGTH(REPLACE(b.content, ' ', '')) + 1
WHEN b.is_short_story = 0 THEN (SELECT COALESCE(SUM(word_count), 0) FROM chapters WHERE book_id = b.id " . ($only_published_chapters ? "AND status = 'published'" : "") . ")
ELSE 0
END as total_words
FROM books b
LEFT JOIN chapters c ON b.id = c.book_id
WHERE b.id = ?
";
if ($only_published_chapters) {
$sql .= " AND c.status = 'published'";
}
$stmt = $this->pdo->prepare($sql);
$stmt->execute([$book_id]);
return $stmt->fetch(PDO::FETCH_ASSOC);

View File

@ -12,12 +12,17 @@ class Series {
$stmt = $this->pdo->prepare("
SELECT s.*,
COUNT(b.id) as book_count,
COALESCE((
SELECT SUM(c.word_count)
FROM chapters c
JOIN books b2 ON c.book_id = b2.id
(
SELECT COALESCE(SUM(
CASE
WHEN b2.is_short_story = 1
THEN CHAR_LENGTH(b2.content) - CHAR_LENGTH(REPLACE(b2.content, ' ', '')) + 1
ELSE (SELECT COALESCE(SUM(word_count), 0) FROM chapters WHERE book_id = b2.id)
END
), 0)
FROM books b2
WHERE b2.series_id = s.id AND b2.published = 1
), 0) as total_words
) as total_words
FROM series s
LEFT JOIN books b ON s.id = b.series_id AND b.published = 1
WHERE s.id = ?
@ -32,11 +37,12 @@ class Series {
$sql = "
SELECT s.*,
COUNT(b.id) as book_count,
COALESCE((
SELECT SUM(c.word_count)
FROM chapters c
JOIN books b2 ON c.book_id = b2.id
WHERE b2.series_id = s.id AND b2.user_id = ?
COALESCE(SUM(
CASE
WHEN b.is_short_story = 1
THEN CHAR_LENGTH(b.content) - CHAR_LENGTH(REPLACE(b.content, ' ', '')) + 1
ELSE (SELECT COALESCE(SUM(word_count), 0) FROM chapters WHERE book_id = b.id)
END
), 0) as total_words
FROM series s
LEFT JOIN books b ON s.id = b.series_id
@ -45,7 +51,7 @@ class Series {
ORDER BY s.created_at DESC
";
$stmt = $this->pdo->prepare($sql);
$stmt->execute([$user_id, $user_id]);
$stmt->execute([$user_id]);
} else {
$sql = "SELECT * FROM series WHERE user_id = ? ORDER BY created_at DESC";
$stmt = $this->pdo->prepare($sql);
@ -128,8 +134,14 @@ class Series {
$sql = "
SELECT
COUNT(b.id) as book_count,
COALESCE(SUM(stats.chapter_count), 0) as chapter_count,
COALESCE(SUM(stats.total_words), 0) as total_words
COALESCE(SUM(
CASE WHEN b.is_short_story = 1 THEN 0 ELSE stats.chapter_count END
), 0) as chapter_count,
COALESCE(SUM(
CASE WHEN b.is_short_story = 1
THEN CHAR_LENGTH(b.content) - CHAR_LENGTH(REPLACE(b.content, ' ', '')) + 1
ELSE COALESCE(stats.total_words, 0) END
), 0) as total_words
FROM series s
LEFT JOIN books b ON s.id = b.series_id
LEFT JOIN (

View File

@ -34,7 +34,7 @@ include 'views/layouts/header.php';
<div class="card">
<div class="card-body">
<form method="post" enctype="multipart/form-data">
<form method="post" enctype="multipart/form-data" id="book-form">
<input type="hidden" name="csrf_token" value="<?= generate_csrf_token() ?>">
<div class="mb-3">
@ -94,6 +94,28 @@ include 'views/layouts/header.php';
</div>
</div>
<div class="mb-4">
<div class="form-check">
<input class="form-check-input" type="checkbox" id="is_short_story" name="is_short_story" value="1"
<?= (!empty($_POST['is_short_story']) && $_POST['is_short_story']) ? 'checked' : '' ?> onchange="toggleStoryEditor()">
<label class="form-check-label" for="is_short_story">
Это рассказ (без разбивки на главы)
</label>
</div>
</div>
<div class="card mb-4" id="story-editor-card" style="display: none;">
<div class="card-header">
<h5 class="card-title mb-0">Содержание рассказа</h5>
</div>
<div class="card-body">
<div id="story-editor" style="height: 400px;" data-content="<?= htmlspecialchars($_POST['content'] ?? '', ENT_QUOTES) ?>">
<?= $_POST['content'] ?? '' ?>
</div>
<input type="hidden" name="content" id="story-content" <?= !empty($_POST['content']) ? 'value="' . htmlspecialchars($_POST['content'], ENT_QUOTES) . '"' : '' ?>>
</div>
</div>
<div class="d-flex gap-2">
<button type="submit" class="btn btn-primary">
<i class="bi bi-journal-plus"></i> Создать книгу
@ -107,6 +129,44 @@ include 'views/layouts/header.php';
</div>
</div>
</div>
</div>
<script src="<?= SITE_URL ?>/assets/js/editor.js"></script>
<script src="<?= SITE_URL ?>/assets/js/autosave.js"></script>
<script>
function toggleStoryEditor() {
var isShortStory = document.getElementById('is_short_story').checked;
var storyCard = document.getElementById('story-editor-card');
if (storyCard) {
storyCard.style.display = isShortStory ? 'block' : 'none';
if (isShortStory && !window.writerEditor) {
initQuillEditor(true);
} else if (!isShortStory && window.writerEditor) {
window.writerEditor.quill.destroy();
window.writerEditor = null;
}
if (isShortStory && window.writerEditor && window.writerEditor.quill) {
setTimeout(() => {
window.writerEditor.quill.root.style.height = '400px';
window.writerEditor.quill.container.style.height = '400px';
}, 100);
}
}
}
document.addEventListener('DOMContentLoaded', function() {
toggleStoryEditor();
window.addEventListener('resize', function() {
var isShortStory = document.getElementById('is_short_story').checked;
if (isShortStory && window.writerEditor && window.writerEditor.quill) {
window.writerEditor.quill.root.style.height = '400px';
window.writerEditor.quill.container.style.height = '400px';
}
});
});
</script>
<?php include 'views/layouts/footer.php'; ?>

View File

@ -18,16 +18,19 @@ include 'views/layouts/header.php';
</div>
<?php endif; ?>
<div class="row">
<div class="col-lg-8">
<div class="card mb-4">
<div class="card-header">
<h5 class="card-title mb-0">Основная информация</h5>
</div>
<div class="card-body">
<form method="post" enctype="multipart/form-data">
<form method="post" enctype="multipart/form-data" id="book-form">
<input type="hidden" name="csrf_token" value="<?= generate_csrf_token() ?>">
<!-- Аккордеон для основной информации (свернут) -->
<div class="accordion mb-4" id="bookInfoAccordion">
<div class="accordion-item">
<h2 class="accordion-header">
<button class="accordion-button collapsed" type="button" data-bs-toggle="collapse" data-bs-target="#bookInfoCollapse">
<i class="bi bi-info-circle me-2"></i> Основная информация
</button>
</h2>
<div id="bookInfoCollapse" class="accordion-collapse collapse" data-bs-parent="#bookInfoAccordion">
<div class="accordion-body">
<div class="mb-3">
<label for="title" class="form-label">Название книги *</label>
<input type="text" class="form-control" id="title" name="title"
@ -83,14 +86,38 @@ include 'views/layouts/header.php';
</div>
</div>
<button type="submit" class="btn btn-primary">
<i class="bi bi-check-circle"></i> Сохранить изменения
</button>
</form>
<div class="mb-4">
<div class="form-check">
<input class="form-check-input" type="checkbox" id="is_short_story" name="is_short_story" value="1"
<?= !empty($book['is_short_story']) ? 'checked' : '' ?> onchange="toggleStoryEditor()">
<label class="form-check-label" for="is_short_story">
Это рассказ (без разбивки на главы)
</label>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="card mb-4">
<!-- Редактор рассказа (всегда видимый) -->
<div class="mb-4" id="story-editor-card" style="display: <?= !empty($book['is_short_story']) ? 'block' : 'none' ?>">
<div class="card">
<div class="card-header bg-success bg-opacity-10">
<h5 class="card-title mb-0 text-success">
<i class="bi bi-book me-2"></i>Содержание рассказа
</h5>
</div>
<div class="card-body">
<div id="story-editor" style="height: 500px;" data-content="<?= htmlspecialchars($book['content'] ?? '', ENT_QUOTES) ?>"></div>
<textarea name="content" id="content" style="display: none;"><?= htmlspecialchars($book['content'] ?? '', ENT_QUOTES) ?></textarea>
</div>
</div>
</div>
<!-- Главы книги (всегда видимые) -->
<div class="mb-4" id="chapters-card" style="display: <?= !empty($book['is_short_story']) ? 'none' : 'block' ?>">
<div class="card">
<div class="card-header">
<h5 class="card-title mb-0">Главы книги</h5>
</div>
@ -145,136 +172,22 @@ include 'views/layouts/header.php';
</div>
</div>
<div class="col-lg-4">
<div class="card mb-4">
<div class="card-header">
<h5 class="card-title mb-0">Обложка книги</h5>
</div>
<div class="card-body">
<?php if (!empty($book['cover_image'])): ?>
<div class="text-center mb-3">
<img src="<?= COVERS_URL . e($book['cover_image']) ?>"
alt="Обложка"
class="img-fluid rounded"
style="max-height: 200px;">
</div>
<div class="form-check mb-3">
<input class="form-check-input" type="checkbox" name="delete_cover" value="1" id="delete_cover">
<label class="form-check-label" for="delete_cover">
Удалить обложку
</label>
</div>
<?php endif; ?>
<div class="mb-3">
<label for="cover_image" class="form-label">Загрузить новую обложку</label>
<input type="file" class="form-control" id="cover_image" name="cover_image"
accept="image/jpeg, image/png, image/gif, image/webp">
<div class="form-text">
Разрешены: JPG, PNG, GIF, WebP. Максимальный размер: 5MB.
</div>
</div>
<?php if (!empty($cover_error)): ?>
<div class="alert alert-danger">
<i class="bi bi-exclamation-triangle"></i> <?= e($cover_error) ?>
</div>
<?php endif; ?>
</div>
</div>
<div class="card mb-4">
<div class="card-header">
<h5 class="card-title mb-0">Публичная ссылка</h5>
</div>
<div class="card-body">
<div class="input-group mb-2">
<input type="text"
id="share-link"
value="<?= e(SITE_URL . '/book/' . $book['share_token']) ?>"
readonly
class="form-control">
<button type="button" onclick="copyShareLink()" class="btn btn-outline-secondary">
<i class="bi bi-clipboard"></i>
<!-- Кнопка сохранения -->
<div class="mb-4">
<button type="submit" class="btn btn-primary btn-lg">
<i class="bi bi-check-circle"></i> Сохранить изменения
</button>
</div>
<form method="post" action="<?= SITE_URL ?>/books/<?= $book['id'] ?>/regenerate-token" class="d-inline">
<input type="hidden" name="csrf_token" value="<?= generate_csrf_token() ?>">
<button type="submit" class="btn btn-outline-warning btn-sm" onclick="return confirm('Создать новую ссылку? Старая ссылка перестанет работать.')">
<i class="bi bi-arrow-repeat"></i> Обновить ссылку
</button>
</form>
<div class="form-text">
В публичном просмотре отображаются только опубликованные главы
</div>
</div>
</div>
<div class="card mb-4">
<div class="card-header">
<h5 class="card-title mb-0">Экспорт книги</h5>
</div>
<div class="card-body">
<div class="d-grid gap-2">
<a href="<?= SITE_URL ?>/export/<?= $book['id'] ?>/pdf" class="btn btn-outline-danger" target="_blank">
<i class="bi bi-file-pdf"></i> PDF
</a>
<a href="<?= SITE_URL ?>/export/<?= $book['id'] ?>/docx" class="btn btn-outline-primary" target="_blank">
<i class="bi bi-file-word"></i> DOCX
</a>
<a href="<?= SITE_URL ?>/export/<?= $book['id'] ?>/html" class="btn btn-outline-success" target="_blank">
<i class="bi bi-file-code"></i> HTML
</a>
<a href="<?= SITE_URL ?>/export/<?= $book['id'] ?>/txt" class="btn btn-outline-secondary" target="_blank">
<i class="bi bi-file-text"></i> TXT
</a>
</div>
<div class="form-text mt-2">
Экспортируются все главы книги (включая черновики)
</div>
</div>
</div>
<div class="card border-danger">
<div class="card-header bg-danger text-white">
<h5 class="card-title mb-0">Опасная зона</h5>
</div>
<div class="card-body">
<p class="card-text small text-muted">
Удаление книги приведет к удалению всех глав и обложки. Это действие нельзя отменить.
</p>
<form method="post" action="<?= SITE_URL ?>/books/<?= $book['id'] ?>/delete"
onsubmit="return confirm('Вы уверены, что хотите удалить книгу «<?= e($book['title']) ?>»? Все главы также будут удалены.');">
<input type="hidden" name="csrf_token" value="<?= generate_csrf_token() ?>">
<button type="submit" class="btn btn-danger w-100">
<i class="bi bi-trash"></i> Удалить книгу
</button>
</form>
</div>
</div>
</div>
</div>
</div>
<script src="<?= SITE_URL ?>/assets/js/editor.js"></script>
<script>
function copyShareLink() {
const shareLink = document.getElementById('share-link');
shareLink.select();
document.execCommand('copy');
// Показать уведомление
const button = event.target;
const originalHTML = button.innerHTML;
button.innerHTML = '<i class="bi bi-check"></i>';
button.classList.remove('btn-outline-secondary');
button.classList.add('btn-success');
setTimeout(() => {
button.innerHTML = originalHTML;
button.classList.remove('btn-success');
button.classList.add('btn-outline-secondary');
}, 2000);
document.addEventListener('DOMContentLoaded', function() {
const checkbox = document.getElementById('is_short_story');
if (checkbox && checkbox.checked) {
toggleStoryEditor();
}
});
</script>
<?php include 'views/layouts/footer.php'; ?>
</div>

View File

@ -49,6 +49,11 @@ include 'views/layouts/header.php';
<div class="flex-grow-1">
<div class="d-flex justify-content-between align-items-start">
<h5 class="card-title mb-1">
<?php if (!empty($book['is_short_story'])): ?>
<i class="bi bi-file-text text-success me-1"></i>
<?php else: ?>
<i class="bi bi-book text-primary me-1"></i>
<?php endif; ?>
<a href="<?= SITE_URL ?>/books/<?= $book['id'] ?>/edit" class="text-decoration-none">
<?= e($book['title']) ?>
</a>
@ -70,6 +75,17 @@ include 'views/layouts/header.php';
<?php endif; ?>
<div class="row text-center mb-3">
<?php if (!empty($book['is_short_story'])): ?>
<div class="col-6">
<div class="border-end">
<div class="fw-bold text-success"><?= number_format($book['total_words'] ?? 0) ?></div>
<small class="text-muted">слов</small>
</div>
</div>
<div class="col-6">
<small class="text-success">(рассказ)</small>
</div>
<?php else: ?>
<div class="col-4">
<div class="border-end">
<div class="fw-bold text-primary"><?= $book['chapter_count'] ?? 0 ?></div>
@ -90,6 +106,7 @@ include 'views/layouts/header.php';
<small class="text-muted">слов/глава</small>
</div>
</div>
<?php endif; ?>
</div>
<div class="d-grid gap-2">
@ -97,9 +114,11 @@ include 'views/layouts/header.php';
<i class="bi bi-pencil"></i> Редактировать
</a>
<div class="btn-group" role="group">
<?php if (empty($book['is_short_story'])): ?>
<a href="<?= SITE_URL ?>/books/<?= $book['id'] ?>/chapters" class="btn btn-outline-secondary btn-sm">
<i class="bi bi-file-text"></i> Главы
</a>
<?php endif; ?>
<a href="<?= SITE_URL ?>/book/<?= $book['share_token'] ?>" class="btn btn-outline-success btn-sm" target="_blank">
<i class="bi bi-eye"></i> Просмотр
</a>

View File

@ -40,12 +40,22 @@ include 'views/layouts/header.php';
<?php endif; ?>
<div class="d-flex justify-content-center gap-4 flex-wrap mb-4">
<?php if (!$is_short_story): ?>
<div class="text-center">
<div class="h4 text-primary mb-0"><?= count($chapters) ?></div>
<small class="text-muted">Глав</small>
</div>
<?php endif; ?>
<div class="text-center">
<div class="h4 text-success mb-0"><?= array_sum(array_column($chapters, 'word_count')) ?></div>
<div class="h4 text-success mb-0">
<?php if ($is_short_story && !empty($book['content'])): ?>
<?= number_format(mb_strlen($book['content']) - mb_strlen(str_replace(' ', '', $book['content'])) + 1) ?>
<?php elseif (!$is_short_story): ?>
<?= array_sum(array_column($chapters, 'word_count')) ?>
<?php else: ?>
0
<?php endif; ?>
</div>
<small class="text-muted">Слов</small>
</div>
</div>
@ -81,7 +91,13 @@ include 'views/layouts/header.php';
</div>
</header>
<?php if (empty($chapters)): ?>
<?php if ($is_short_story && !empty($book['content'])): ?>
<div class="chapter-content mb-5">
<div class="chapter-text" style="line-height: 1.8; font-size: 1.1em;">
<?= $book['content'] ?>
</div>
</div>
<?php elseif (empty($chapters)): ?>
<div class="text-center py-5">
<i class="bi bi-file-text fs-1 text-muted"></i>
<h3 class="h4 text-muted mt-3">В этой книге пока нет глав</h3>
@ -146,6 +162,7 @@ include 'views/layouts/header.php';
.chapter-text p {
margin-bottom: 1.5em;
text-align: justify;
text-indent: 2em;
}
.chapter-text .dialogue {

View File

@ -44,6 +44,7 @@
.chapter-content p {
margin-bottom: 1em;
text-align: justify;
text-indent: 2em;
}
.chapter-content code {
background: var(--bs-light);

View File

@ -117,16 +117,21 @@ include 'views/layouts/header.php';
<div class="d-flex justify-content-between align-items-center">
<small class="text-muted">
Глав: <?= $book['chapter_count'] ?? 0 ?> |
Слов: <?= $book['total_words'] ?? 0 ?>
<?php if (!empty($book['is_short_story'])): ?>
(рассказ) | <?= number_format($book['total_words'] ?? 0) ?> слов
<?php else: ?>
Глав: <?= $book['chapter_count'] ?? 0 ?> | <?= number_format($book['total_words'] ?? 0) ?> слов
<?php endif; ?>
</small>
<div class="btn-group btn-group-sm">
<a href="<?= SITE_URL ?>/books/<?= $book['id'] ?>/edit" class="btn btn-outline-primary btn-sm">
<i class="bi bi-pencil"></i>
</a>
<?php if (empty($book['is_short_story'])): ?>
<a href="<?= SITE_URL ?>/books/<?= $book['id'] ?>/chapters" class="btn btn-outline-secondary btn-sm">
<i class="bi bi-file-text"></i>
</a>
<?php endif; ?>
</div>
</div>
</div>

View File

@ -22,8 +22,9 @@
</div>
</div>
</footer>
<script src="<?= SITE_URL ?>/assets/js/quill.js"></script>
<?php if (!isset($GLOBALS['bootstrap_loaded'])): ?>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/js/bootstrap.bundle.min.js"></script>
<?php $GLOBALS['bootstrap_loaded'] = true; endif; ?>
<script>
// Функция для установки темы
function setTheme(themeName) {

View File

@ -49,6 +49,9 @@ $available_themes = [
<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap-icons@1.10.0/font/bootstrap-icons.css">
<link rel="stylesheet" href="<?= SITE_URL ?>/assets/css/style.css">
<link href="<?= SITE_URL ?>/assets/css/quill.snow.css" rel="stylesheet">
<script src="<?= SITE_URL ?>/assets/js/quill.min.js"></script>
<script src="https://cdn.jsdelivr.net/npm/bootstrap@5.3.0/dist/js/bootstrap.bundle.min.js"></script>
<?php $GLOBALS['bootstrap_loaded'] = true; ?>
<style>
.navbar-brand { font-weight: 600; }
.dropdown-menu { min-width: 200px; }

View File

@ -69,12 +69,20 @@ include 'views/layouts/header.php';
<div class="col-12">
<?php endif; ?>
<div class="d-flex justify-content-between align-items-start mb-2">
<h3 class="h4">
<h3 class="h4 mb-0">
<?php if ($book['sort_order_in_series']): ?>
<small class="text-muted">Книга <?= $book['sort_order_in_series'] ?></small><br>
<?php endif; ?>
<?php if (!empty($book['is_short_story'])): ?>
<i class="bi bi-file-text text-success me-1"></i>
<?php else: ?>
<i class="bi bi-book text-primary me-1"></i>
<?php endif; ?>
<?= e($book['title']) ?>
</h3>
<?php if (!empty($book['is_short_story'])): ?>
<span class="text-success">(рассказ)</span>
<?php endif; ?>
</div>
<?php if ($book['genre']): ?>
@ -96,8 +104,11 @@ include 'views/layouts/header.php';
$book_stats = $book_model->getBookStats($book['id'], true);
?>
<small class="text-muted">
Глав: <?= $book_stats['chapter_count'] ?? 0 ?> |
Слов: <?= $book_stats['total_words'] ?? 0 ?>
<?php if (!empty($book_stats['is_short_story'])): ?>
<?= number_format($book_stats['total_words'] ?? 0) ?> слов
<?php else: ?>
Глав: <?= $book_stats['chapter_count'] ?? 0 ?> | <?= number_format($book_stats['total_words'] ?? 0) ?> слов
<?php endif; ?>
</small>
</div>
</div>

View File

@ -81,7 +81,14 @@ include 'views/layouts/header.php';
</div>
<?php endif; ?>
<div class="flex-grow-1">
<h5 class="card-title"><?= e($book['title']) ?></h5>
<h5 class="card-title mb-0">
<?php if (!empty($book['is_short_story'])): ?>
<i class="bi bi-file-text text-success me-1"></i>
<?php else: ?>
<i class="bi bi-book text-primary me-1"></i>
<?php endif; ?>
<?= e($book['title']) ?>
</h5>
<?php if ($book['genre']): ?>
<p class="text-muted small mb-1"><em><?= e($book['genre']) ?></em></p>
<?php endif; ?>
@ -102,10 +109,18 @@ include 'views/layouts/header.php';
<div class="d-flex justify-content-between align-items-center">
<a href="<?= SITE_URL ?>/book/<?= e($book['share_token']) ?>" class="btn btn-primary btn-sm">
<?php if (!empty($book['is_short_story'])): ?>
<i class="bi bi-file-text"></i> Читать рассказ
<?php else: ?>
<i class="bi bi-book"></i> Читать книгу
<?php endif; ?>
</a>
<small class="text-muted">
Глав: <?= $chapter_count ?> | Слов: <?= $word_count ?>
<?php if (!empty($book['is_short_story'])): ?>
(рассказ) <?= number_format($word_count) ?> слов
<?php else: ?>
Глав: <?= $chapter_count ?> | <?= number_format($word_count) ?> слов
<?php endif; ?>
</small>
</div>
</div>