release
This commit is contained in:
@@ -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()
|
||||
|
||||
@@ -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
@@ -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,
|
||||
|
||||
@@ -17,7 +17,7 @@
|
||||
|
||||
/* Таблица сотрудников */
|
||||
.table-responsive {
|
||||
max-height: 800px;
|
||||
max-height: 575px;
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
|
||||
+18
-30
@@ -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);
|
||||
|
||||
@@ -152,3 +152,41 @@ document.getElementById('base_photo_url').addEventListener('focus', function ()
|
||||
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('Ошибка обработки ссылки');
|
||||
}
|
||||
}
|
||||
@@ -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
@@ -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
@@ -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>
|
||||
|
||||
Reference in New Issue
Block a user