This commit is contained in:
2025-12-21 15:12:09 +03:00
parent e149d7380d
commit 865224ba5a
10 changed files with 145 additions and 156 deletions
BIN
View File
Binary file not shown.
+17 -14
View File
@@ -332,6 +332,7 @@ def api_vk():
@app.route("/api/posts", methods=["POST", "GET"])
def api_posts():
response = {"status": "ok"}
match request.method:
case "POST":
requestData = request.json
@@ -362,32 +363,34 @@ def api_posts():
logger.info("Обновление расписания публикации")
try:
scheduler = PostScheduler.query.first()
startTime = schedulerData.get("startTime", None)
endTime = schedulerData.get("endTime", None)
interval_minutes = schedulerData.get("interval_minutes", None)
hour = schedulerData.get("hour", None)
minute = schedulerData.get("minute", None)
enabled = schedulerData.get("enabled", None)
if scheduler:
if startTime:
scheduler.start_hour = int(startTime)
if endTime:
scheduler.end_hour = int(endTime)
if interval_minutes:
scheduler.interval_minutes = int(interval_minutes)
if hour is not None:
if scheduler.hour != hour:
scheduler.hour = hour
if minute is not None:
if scheduler.minute != minute:
scheduler.minute = minute
if enabled is not None:
scheduler.enabled = enabled
if scheduler.enabled != enabled:
scheduler.enabled = enabled
else:
db.session.merge(
PostScheduler(
start_hour=int(startTime),
end_hour=int(endTime),
interval_minutes=int(interval_minutes),
hour=int(hour),
minute=int(minute),
enabled=enabled,
)
)
db.session.commit()
enable_publish_job()
scheduleInfo = get_scheduler_status()
response["next_run_time"] = scheduleInfo.get("next_run_time")
except Exception as e:
logger.error(f"Ошибка при обновлении расписания публикации: {e}")
return jsonify({"status": "ok"})
return jsonify(response)
case "GET":
queryParams = request.args.to_dict()
+4 -6
View File
@@ -123,9 +123,8 @@ class PostScheduler(db.Model):
__tablename__ = "post_scheduler"
id = db.Column(db.Integer, primary_key=True)
start_hour = db.Column(db.Integer)
end_hour = db.Column(db.Integer)
interval_minutes = db.Column(db.Integer)
hour = db.Column(db.Integer)
minute = db.Column(db.Integer)
enabled = db.Column(db.Boolean)
created_at = db.Column(db.DateTime, default=datetime.now)
updated_at = db.Column(db.DateTime, default=datetime.now, onupdate=datetime.now)
@@ -133,9 +132,8 @@ class PostScheduler(db.Model):
def toDict(self):
return {
"id": self.id,
"start_hour": self.start_hour,
"end_hour": self.end_hour,
"interval_minutes": self.interval_minutes,
"hour": self.hour,
"minute": self.minute,
"enabled": self.enabled,
"created_at": self.created_at.strftime("%Y-%m-%d %H:%M:%S"),
"updated_at": self.updated_at.strftime("%Y-%m-%d %H:%M:%S"),
+2 -10
View File
@@ -75,19 +75,11 @@ def enable_publish_job():
disable_publish_job()
return
start_hour = scheduleData.start_hour
end_hour = scheduleData.end_hour
interval_minutes = scheduleData.interval_minutes
trigger = CronTrigger(
hour="12",
minute="0",
hour=f"{scheduleData.hour}",
minute=f"{scheduleData.minute}",
day="*",
)
# trigger = CronTrigger(
# hour=f"{start_hour}-{end_hour - 1}",
# minute=f"*/{interval_minutes}",
# )
scheduler.add_job(
vk_publish_job,
+1 -1
View File
@@ -17,7 +17,7 @@
/* Таблица сотрудников */
.table-responsive {
max-height: 800px;
max-height: 575px;
overflow-y: auto;
}
+18 -30
View File
@@ -20,9 +20,8 @@ function saveOriginalSettings() {
selected_users: getSelectedUsers(),
static_text: document.getElementById('static_text').value,
full_name: document.getElementById('full_name').checked,
start_hour: parseInt(document.getElementById('start_hour').value),
end_hour: parseInt(document.getElementById('end_hour').value),
interval_minutes: parseInt(document.getElementById('interval_minutes').value),
hour: parseInt(document.getElementById('hour').value),
minute: parseInt(document.getElementById('minute').value),
enabled: document.getElementById('scheduler_enabled').checked
};
}
@@ -38,7 +37,7 @@ function setupChangeListeners() {
});
// Поля формы
const formFields = ['static_text', 'start_hour', 'end_hour', 'interval_minutes'];
const formFields = ['static_text', 'hour', 'minute'];
formFields.forEach(field => {
const element = document.getElementById(field);
if (element) {
@@ -58,9 +57,8 @@ function checkForChanges() {
selected_users: getSelectedUsers(),
static_text: document.getElementById('static_text').value,
full_name: document.getElementById('full_name').checked,
start_hour: parseInt(document.getElementById('start_hour').value),
end_hour: parseInt(document.getElementById('end_hour').value),
interval_minutes: parseInt(document.getElementById('interval_minutes').value),
hour: parseInt(document.getElementById('hour').value),
minute: parseInt(document.getElementById('minute').value),
enabled: document.getElementById('scheduler_enabled').checked
};
@@ -194,9 +192,8 @@ async function saveSettings() {
const selectedUsers = getSelectedUsers();
const staticText = document.getElementById('static_text').value.trim();
const fullName = document.getElementById('full_name').checked;
const startHour = parseInt(document.getElementById('start_hour').value);
const endHour = parseInt(document.getElementById('end_hour').value);
const intervalMinutes = parseInt(document.getElementById('interval_minutes').value);
const hour = parseInt(document.getElementById('hour').value);
const minute = parseInt(document.getElementById('minute').value);
const schedulerEnabled = document.getElementById('scheduler_enabled').checked;
// Валидация
@@ -210,23 +207,13 @@ async function saveSettings() {
return;
}
if (startHour < 0 || startHour > 23) {
showAlert('warning', 'Время начала должно быть от 0 до 23 часов');
if (hour < 0 || hour > 23) {
showAlert('warning', 'Час должен быть от 0 до 23 часов');
return;
}
if (endHour < 0 || endHour > 23) {
showAlert('warning', 'Время окончания должно быть от 0 до 23 часов');
return;
}
if (startHour >= endHour) {
showAlert('warning', 'Время начала должно быть раньше времени окончания');
return;
}
if (intervalMinutes < 1 || intervalMinutes > 1440) {
showAlert('warning', 'Интервал должен быть от 1 до 1440 минут');
if (minute < 0 || minute > 59) {
showAlert('warning', 'Минута должна быть от 0 до 59 минут');
return;
}
@@ -248,14 +235,12 @@ async function saveSettings() {
}
// Только измененные данные для schedulerData
if (startHour !== originalSettings.start_hour ||
endHour !== originalSettings.end_hour ||
intervalMinutes !== originalSettings.interval_minutes ||
if (hour !== originalSettings.hour ||
minute !== originalSettings.minute ||
schedulerEnabled !== originalSettings.enabled) {
postData.schedulerData = {
startTime: startHour.toString(),
endTime: endHour.toString(),
interval_minutes: intervalMinutes.toString(),
hour: hour,
minute: minute,
enabled: schedulerEnabled
};
}
@@ -288,6 +273,9 @@ async function saveSettings() {
if (Object.keys(postData.schedulerData).length > 0) {
setTimeout(updateSchedulerStatus, 1000);
}
if (data.next_run_time) {
document.getElementById('nextRunTime').textContent = data.next_run_time;
}
} else {
const error = data.message || 'Ошибка сохранения';
showAlert('danger', error);
+39 -1
View File
@@ -151,4 +151,42 @@ document.getElementById('base_photo_url').addEventListener('focus', function ()
if (!this.value) {
showAlert('info', 'Формат ID фото: photo_id (например: 7236456789)');
}
});
});
function handleLink() {
const link = prompt('Введите ссылку на фотографию:');
if (!link) return;
try {
const url = new URL(link);
const zParam = url.searchParams.get('z');
if (!zParam || !zParam.startsWith('photo')) {
alert('Некорректная ссылка на фото ВК');
return;
}
const decoded = decodeURIComponent(zParam);
const match = decoded.match(/photo(-?\d+)_(\d+)/);
if (!match) {
alert('Не удалось разобрать ссылку');
return;
}
const groupId = Math.abs(Number(match[1]));
const photoId = Number(match[2]);
const isConfirmed = window.confirm(
`Применить новые данные?\nГруппа: ${groupId}\nID фото: ${photoId}`
);
if (isConfirmed) {
document.getElementById('group_id').value = groupId;
document.getElementById('base_photo_url').value = photoId;
}
} catch (e) {
alert('Ошибка обработки ссылки');
}
}
+8 -8
View File
@@ -228,12 +228,12 @@
</div>
</div>
<div class="col-6">
<div class="small text-muted">Следующий запуск</div>
<div class="small text-muted">Отображение имен</div>
<div class="fw-semibold">
{% if exitData.vkPost.scheduler.next_run_time %}
{{ exitData.vkPost.scheduler.next_run_time }}
{% if exitData.vkPost.full_name %}
Полные имена
{% else %}
Не запланирован
Короткие имена
{% endif %}
</div>
</div>
@@ -365,12 +365,12 @@
<div class="col-6">
<div class="card bg-light">
<div class="card-body py-2">
<div class="small text-muted">Отображение имен</div>
<div class="small text-muted">Следующая публикация</div>
<div class="fw-semibold">
{% if exitData.vkPost.full_name %}
Полные имена
{% if exitData.vkPost.scheduler.next_run_time %}
{{ exitData.vkPost.scheduler.next_run_time }}
{% else %}
Короткие имена
Не запланирован
{% endif %}
</div>
</div>
+39 -81
View File
@@ -12,13 +12,16 @@
<h2 class="mb-1"><i class="bi bi-megaphone-fill text-primary me-2"></i>Управление постами</h2>
<p class="text-muted mb-0">Создание и планирование публикаций в VK</p>
</div>
<button type="button" class="btn btn-success btn-lg px-5" onclick="saveSettings()">
<i class="bi bi-save me-2"></i>Сохранить все настройки
</button>
<div class="badge bg-primary fs-6 px-3 py-2">
<i class="bi bi-calendar-week me-1"></i>Публикации
</div>
</div>
<!-- Основной контент -->
<div class="row g-4">
<div class="row g-4 mb-3">
<!-- Левая колонка: Сотрудники -->
<div class="col-lg-7">
<div class="card h-100">
@@ -164,95 +167,50 @@
</span>
</div>
<div class="card-body">
<div class="row g-3 align-items-start">
<div class="alert alert-danger">
<div class="d-flex">
<div class="me-3">
<i class="bi bi-exclamation-triangle fs-5"></i>
</div>
<div>
<h6 class="alert-heading mb-1">Временная настройка</h6>
<p class="mb-0">Посты будут опубликованы в 12:00 ежедневно.<br>Настройки из расписания будут
проигнорированы!</p>
<!-- Время запуска -->
<div class="col-12 col-md-4">
<label class="form-label fw-semibold mb-1">Время запуска</label>
<div class="input-group input-group-sm">
<span class="input-group-text"><i class="bi bi-alarm"></i></span>
<input type="number" class="form-control" id="hour" min="0" max="23"
value="{{ data.schedulerSettings.hour if data.schedulerSettings else 12 }}">
<span class="input-group-text">:</span>
<input type="number" class="form-control" id="minute" min="0" max="59"
value="{{ data.schedulerSettings.minute if data.schedulerSettings else 0 }}">
</div>
</div>
</div>
<!-- Время работы -->
<div class="row g-3 mb-3">
<div class="col-md-6">
<label class="form-label fw-semibold">Начало публикации</label>
<div class="input-group">
<span class="input-group-text"><i class="bi bi-sun"></i></span>
<input type="number" class="form-control" id="start_hour" min="0" max="23"
value="{{ data.schedulerSettings.start_hour if data.schedulerSettings else 9 }}">
<span class="input-group-text">:00</span>
<!-- Статус планировщика -->
<div class="col-12 col-md-8">
<label class="form-label fw-semibold mb-1">Статус планировщика</label>
<div class="form-check form-switch d-flex align-items-center gap-2 p-0">
<input class="form-check-input m-0" type="checkbox" role="switch" id="scheduler_enabled" {%
if data.schedulerSettings and data.schedulerSettings.enabled %}checked{% endif %}>
<label class="form-check-label mb-0" for="scheduler_enabled">
Автоматическая публикация
</label>
</div>
</div>
<div class="col-md-6">
<label class="form-label fw-semibold">Конец публикации</label>
<div class="input-group">
<span class="input-group-text"><i class="bi bi-moon"></i></span>
<input type="number" class="form-control" id="end_hour" min="0" max="23"
value="{{ data.schedulerSettings.end_hour if data.schedulerSettings else 21 }}">
<span class="input-group-text">:00</span>
</div>
</div>
</div>
<!-- Интервал -->
<div class="mb-3">
<label class="form-label fw-semibold">Интервал публикации (минут)</label>
<input type="number" class="form-control" id="interval_minutes" min="1" max="1440"
value="{{ data.schedulerSettings.interval_minutes if data.schedulerSettings else 60 }}">
<div class="form-text">
Через сколько минут публиковать следующий пост
</div>
</div>
<!-- Включить/выключить -->
<div class="mb-4">
<label class="form-label fw-semibold">Статус планировщика</label>
<div class="form-check form-switch d-flex justify-content-left gap-2 p-0">
<input class="form-check-input m-0" type="checkbox" role="switch" id="scheduler_enabled" {% if
data.schedulerSettings and data.schedulerSettings.enabled %}checked{% endif %}>
<label class="form-check-label mb-0" for="scheduler_enabled">
Автоматическая публикация по расписанию
</label>
</div>
</div>
<!-- Информация о следующем запуске -->
{% if data.schedulerStatus.next_run_time %}
<div class="alert alert-info">
<div class="d-flex">
<div class="me-3">
<i class="bi bi-info-circle fs-5"></i>
</div>
<div>
<h6 class="alert-heading mb-1">Следующая публикация</h6>
<p class="mb-0">{{ data.schedulerStatus.next_run_time }}</p>
</div>
</div>
</div>
{% endif %}
</div>
</div>
</div>
</div>
<!-- Кнопка сохранения -->
<div class="row mt-4">
<div class="col-12">
<div class="card">
<div class="card-body text-center py-3">
<button type="button" class="btn btn-success btn-lg px-5" onclick="saveSettings()">
<i class="bi bi-save me-2"></i>Сохранить все настройки
</button>
<div class="form-text mt-2">
Сохраняются: выбранные сотрудники, текст поста, настройки расписания
</div>
</div>
<!-- Информация о следующем запуске -->
{% if data.schedulerStatus.next_run_time %}
<div class="alert alert-success shadow mx-3">
<div class="d-flex">
<div class="me-3">
<i class="bi bi-info-circle fs-5"></i>
</div>
<div>
<h6 class="alert-heading mb-1">Следующая публикация</h6>
<p class="mb-0" id="nextRunTime">{{ data.schedulerStatus.next_run_time }}</p>
</div>
</div>
</div>
{% endif %}
</div>
</div>
</div>
+17 -5
View File
@@ -96,6 +96,9 @@
<button type="button" class="btn btn-primary" onclick="saveVkSettings()" id="saveButton">
<i class="bi bi-save me-1"></i>Сохранить настройки
</button>
<button type="button" class="btn btn-outline-success" onclick="handleLink()" id="linkButton">
<i class="bi bi-link me-1"></i>Получить данные из ссылки на фото
</button>
</div>
</form>
</div>
@@ -103,7 +106,7 @@
<!-- Информация о текущих настройках -->
{% if data.vk_settings %}
<div class="card mt-4 fade-in">
<div class="card my-4 fade-in">
<div class="card-header bg-success bg-opacity-10 text-success">
<h6 class="mb-0"><i class="bi bi-check-circle me-2"></i>Текущие настройки</h6>
</div>
@@ -123,16 +126,25 @@
<div class="card bg-light">
<div class="card-body">
<h6 class="card-title"><i class="bi bi-people me-2"></i>ID сообщества</h6>
<p class="card-text">-{{ data.vk_settings.group_id }}</p>
<p class="card-text">
<a href="https://vk.com/club{{ data.vk_settings.group_id }}" target="_blank">
<i class="bi bi-link me-1"></i>-{{ data.vk_settings.group_id }}
</a>
</p>
</div>
</div>
</div>
<div class="col-md-4">
<div class="card bg-light">
<div class="card-body">
<h6 class="card-title"><i class="bi bi-image me-2"></i>ID Базового фото</h6>
<p class="card-text">photo-{{ data.vk_settings.group_id }}_{{
data.vk_settings.base_photo_url }}</p>
<h6 class="card-title"><i class="bi bi-image me-2"></i>Базовое фото</h6>
<p class="card-text">
<a href="https://vk.com/photo-{{ data.vk_settings.group_id }}_{{ data.vk_settings.base_photo_url }}"
target="_blank">
<i class="bi bi-link me-1"></i>photo-{{ data.vk_settings.group_id }}_{{
data.vk_settings.base_photo_url }}
</a>
</p>
</div>
</div>
</div>