295 lines
13 KiB
HTML
295 lines
13 KiB
HTML
{% extends 'base.html' %}
|
||
{% load static %}
|
||
{% block title %}POS Terminal{% endblock %}
|
||
|
||
{% block extra_css %}
|
||
<link rel="stylesheet" href="{% static 'pos/css/terminal.css' %}">
|
||
{% endblock %}
|
||
|
||
{% block content %}
|
||
<!-- Main POS Container -->
|
||
<div class="pos-main-container">
|
||
<div class="pos-container">
|
||
<div class="row g-3" style="height: 100%;">
|
||
<!-- Products Grid (Left side - 8/12) -->
|
||
<div class="col-md-8" style="display: flex; flex-direction: column; height: 100%;">
|
||
<!-- Search Box -->
|
||
<div class="mb-3">
|
||
<input type="text" class="form-control" id="searchInput" placeholder="Поиск по товарам...">
|
||
</div>
|
||
|
||
<!-- Categories -->
|
||
<div class="mb-3">
|
||
<div class="row g-3" id="categoryGrid"></div>
|
||
</div>
|
||
|
||
<!-- Products Grid (Блок товаров) - Прокручиваемая область -->
|
||
<div class="products-scrollable">
|
||
<div class="row g-3" id="productGrid"></div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Right Panel (4/12) - Fixed -->
|
||
<div class="col-md-4">
|
||
<div class="right-panel-fixed d-flex flex-column">
|
||
<!-- Информация о складе -->
|
||
{% if current_warehouse %}
|
||
<div class="card mb-2">
|
||
<div class="card-body py-2 px-3 d-flex justify-content-between align-items-center">
|
||
<div>
|
||
<small class="text-muted d-block" style="font-size: 0.75rem;">Склад:</small>
|
||
<strong style="font-size: 0.95rem;">{{ current_warehouse.name }}</strong>
|
||
</div>
|
||
<button class="btn btn-sm btn-outline-secondary" id="changeWarehouseBtn" style="font-size: 0.75rem;">
|
||
<i class="bi bi-arrow-left-right"></i> Сменить
|
||
</button>
|
||
</div>
|
||
</div>
|
||
{% endif %}
|
||
|
||
<!-- Cart Panel -->
|
||
<div class="card mb-2 flex-grow-1" style="min-height: 0;">
|
||
<div class="card-header bg-white d-flex justify-content-between align-items-center">
|
||
<h6 class="mb-0">Корзина</h6>
|
||
<button class="btn btn-outline-primary btn-sm" id="customerSelectBtn">
|
||
<i class="bi bi-person"></i> Выбрать клиента
|
||
</button>
|
||
</div>
|
||
<div class="card-body d-flex flex-column" style="min-height: 0;">
|
||
<div id="cartList" class="flex-grow-1" style="overflow-y: auto;"></div>
|
||
|
||
<div class="mt-auto">
|
||
<div class="d-flex justify-content-between align-items-center py-1 border-top">
|
||
<strong class="mb-0">Итого:</strong>
|
||
<strong class="mb-0 text-primary fs-5" id="cartTotal">0.00</strong>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Action Buttons Panel -->
|
||
<div class="card action-buttons-panel">
|
||
<div class="card-body p-2">
|
||
<div class="row g-2">
|
||
<div class="col-4">
|
||
<button class="btn btn-outline-warning rounded-3 w-100" id="addToShowcaseBtn" style="height: 60px;">
|
||
<i class="bi bi-flower1"></i><br>На витрину
|
||
</button>
|
||
</div>
|
||
<div class="col-4">
|
||
<button class="btn btn-outline-secondary rounded-3 w-100" style="height: 60px;">
|
||
</button>
|
||
</div>
|
||
<div class="col-4">
|
||
<button class="btn btn-outline-secondary rounded-3 w-100" style="height: 60px;">
|
||
</button>
|
||
</div>
|
||
<div class="col-4">
|
||
<button class="btn btn-outline-secondary rounded-3 w-100" style="height: 60px;">
|
||
</button>
|
||
</div>
|
||
<div class="col-4">
|
||
<button class="btn btn-outline-secondary rounded-3 w-100" style="height: 60px;">
|
||
</button>
|
||
</div>
|
||
<div class="col-4">
|
||
<button class="btn btn-outline-primary rounded-3 w-100" id="scheduleLater" style="height: 60px;">
|
||
<i class="bi bi-calendar2"></i><br>Отложенный заказ
|
||
</button>
|
||
</div>
|
||
<div class="col-4">
|
||
<button class="btn btn-outline-secondary rounded-3 w-100" id="clearCart" style="height: 60px;">
|
||
<i class="bi bi-trash"></i><br>Очистить
|
||
</button>
|
||
</div>
|
||
<div class="col-4">
|
||
<button class="btn btn-outline-secondary rounded-3 w-100" style="height: 60px;">
|
||
</button>
|
||
</div>
|
||
<div class="col-4">
|
||
<button class="btn btn-success rounded-3 w-100" id="checkoutNow" style="height: 60px;">
|
||
<i class="bi bi-check2-circle"></i><br>Продать
|
||
</button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Modal: Создание временного комплекта на витрину -->
|
||
<div class="modal fade" id="createTempKitModal" tabindex="-1" aria-labelledby="createTempKitModalLabel" aria-hidden="true">
|
||
<div class="modal-dialog modal-xl">
|
||
<div class="modal-content">
|
||
<div class="modal-header">
|
||
<h5 class="modal-title" id="createTempKitModalLabel">
|
||
<i class="bi bi-flower1"></i> Создать временный комплект на витрину
|
||
</h5>
|
||
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
|
||
</div>
|
||
<div class="modal-body">
|
||
<div class="row">
|
||
<!-- Левая колонка: основные поля -->
|
||
<div class="col-md-6">
|
||
<!-- Название комплекта -->
|
||
<div class="mb-3">
|
||
<label for="tempKitName" class="form-label">Название комплекта *</label>
|
||
<input type="text" class="form-control" id="tempKitName" placeholder="Введите название" required>
|
||
</div>
|
||
|
||
<!-- Выбор витрины -->
|
||
<div class="mb-3">
|
||
<label for="showcaseSelect" class="form-label">Витрина *</label>
|
||
<select class="form-select" id="showcaseSelect" required>
|
||
<option value="">Загрузка...</option>
|
||
</select>
|
||
</div>
|
||
|
||
<!-- Описание -->
|
||
<div class="mb-3">
|
||
<label for="tempKitDescription" class="form-label">Описание (опционально)</label>
|
||
<textarea class="form-control" id="tempKitDescription" rows="3" placeholder="Краткое описание комплекта"></textarea>
|
||
</div>
|
||
|
||
<!-- Загрузка фото -->
|
||
<div class="mb-3">
|
||
<label for="tempKitPhoto" class="form-label">Фото комплекта (опционально)</label>
|
||
<input type="file" class="form-control" id="tempKitPhoto" accept="image/*">
|
||
<div id="photoPreview" class="mt-2" style="display: none;">
|
||
<img id="photoPreviewImg" src="" alt="Preview" class="img-thumbnail" style="max-width: 200px; max-height: 200px;">
|
||
<button type="button" class="btn btn-sm btn-danger ms-2" id="removePhoto">
|
||
<i class="bi bi-x"></i> Удалить
|
||
</button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Правая колонка: цены и состав -->
|
||
<div class="col-md-6">
|
||
<!-- Список товаров в корзине -->
|
||
<div class="mb-3">
|
||
<label class="form-label">Товары в комплекте</label>
|
||
<div class="border rounded p-2" id="tempKitItemsList" style="max-height: 200px; overflow-y: auto; background: #f8f9fa;">
|
||
<!-- Динамически заполняется через JS -->
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Блок ценообразования -->
|
||
<div class="card">
|
||
<div class="card-header bg-light">
|
||
<strong>Ценообразование</strong>
|
||
</div>
|
||
<div class="card-body">
|
||
<!-- Базовая цена -->
|
||
<div class="mb-2">
|
||
<small class="text-muted">Базовая цена (сумма компонентов):</small>
|
||
<div class="fw-bold" id="tempKitBasePrice">0.00 руб.</div>
|
||
</div>
|
||
|
||
<!-- Корректировка цены -->
|
||
<div class="mb-2">
|
||
<label for="priceAdjustmentType" class="form-label small">Корректировка цены</label>
|
||
<select class="form-select form-select-sm" id="priceAdjustmentType">
|
||
<option value="none">Без изменения</option>
|
||
<option value="increase_percent">Увеличить на %</option>
|
||
<option value="increase_amount">Увеличить на сумму</option>
|
||
<option value="decrease_percent">Уменьшить на %</option>
|
||
<option value="decrease_amount">Уменьшить на сумму</option>
|
||
</select>
|
||
</div>
|
||
|
||
<div class="mb-2" id="adjustmentValueBlock" style="display: none;">
|
||
<label for="priceAdjustmentValue" class="form-label small">Значение</label>
|
||
<input type="number" class="form-control form-control-sm" id="priceAdjustmentValue"
|
||
min="0" step="0.01" value="0" placeholder="0.00">
|
||
</div>
|
||
|
||
<!-- Расчётная цена с корректировкой -->
|
||
<div class="mb-2 pb-2 border-bottom">
|
||
<small class="text-muted">Расчётная цена:</small>
|
||
<div class="fw-bold text-primary" id="tempKitCalculatedPrice">0.00 руб.</div>
|
||
</div>
|
||
|
||
<!-- Ручная финальная цена (sale_price) -->
|
||
<div class="mb-0">
|
||
<div class="form-check form-switch mb-2">
|
||
<input class="form-check-input" type="checkbox" id="useSalePrice">
|
||
<label class="form-check-label small" for="useSalePrice">
|
||
Установить свою цену (приоритет)
|
||
</label>
|
||
</div>
|
||
<div id="salePriceBlock" style="display: none;">
|
||
<input type="number" class="form-control form-control-sm" id="salePrice"
|
||
min="0" step="0.01" placeholder="Введите цену">
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Итоговая цена продажи -->
|
||
<div class="alert alert-success mt-3 mb-0">
|
||
<strong>Итоговая цена продажи:</strong><br>
|
||
<span class="fs-4" id="tempKitFinalPrice">0.00</span> руб.
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="modal-footer">
|
||
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Отмена</button>
|
||
<button type="button" class="btn btn-primary" id="confirmCreateTempKit">
|
||
<i class="bi bi-check-circle"></i> Создать и зарезервировать
|
||
</button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- Модалка: Выбор склада -->
|
||
<div class="modal fade" id="selectWarehouseModal" tabindex="-1" aria-labelledby="selectWarehouseModalLabel" aria-hidden="true">
|
||
<div class="modal-dialog">
|
||
<div class="modal-content">
|
||
<div class="modal-header">
|
||
<h5 class="modal-title" id="selectWarehouseModalLabel">
|
||
<i class="bi bi-building"></i> Выбор склада
|
||
</h5>
|
||
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
|
||
</div>
|
||
<div class="modal-body">
|
||
<div class="list-group" id="warehouseList">
|
||
{% for warehouse in warehouses %}
|
||
<button type="button" class="list-group-item list-group-item-action d-flex justify-content-between align-items-center warehouse-item"
|
||
data-warehouse-id="{{ warehouse.id }}"
|
||
data-warehouse-name="{{ warehouse.name }}">
|
||
<div>
|
||
<strong>{{ warehouse.name }}</strong>
|
||
{% if warehouse.is_default %}
|
||
<span class="badge bg-warning text-dark ms-2">По умолчанию</span>
|
||
{% endif %}
|
||
</div>
|
||
{% if warehouse.id == current_warehouse.id %}
|
||
<i class="bi bi-check-circle-fill text-success"></i>
|
||
{% endif %}
|
||
</button>
|
||
{% endfor %}
|
||
</div>
|
||
</div>
|
||
<div class="modal-footer">
|
||
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Отмена</button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
{% endblock %}
|
||
|
||
{% block extra_js %}
|
||
<!-- Hidden data containers for JavaScript -->
|
||
<script id="categoriesData" type="application/json">{{ categories_json|safe }}</script>
|
||
<script id="itemsData" type="application/json">{{ items_json|safe }}</script>
|
||
<script id="showcaseKitsData" type="application/json">{{ showcase_kits_json|safe }}</script>
|
||
|
||
<script src="{% static 'pos/js/terminal.js' %}"></script>
|
||
{% endblock %}
|