Autosave: 20260210-122318
This commit is contained in:
parent
46a59fc51c
commit
b66ef8649e
76
core/patch_views.py
Normal file
76
core/patch_views.py
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
@login_required
|
||||||
|
def send_invoice_whatsapp(request):
|
||||||
|
if request.method != 'POST':
|
||||||
|
return JsonResponse({'success': False, 'error': 'Method not allowed'})
|
||||||
|
|
||||||
|
try:
|
||||||
|
# Handle JSON payload
|
||||||
|
data = json.loads(request.body)
|
||||||
|
sale_id = data.get('sale_id')
|
||||||
|
phone = data.get('phone')
|
||||||
|
pdf_data = data.get('pdf_data') # Base64 string
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
# Fallback to Form Data
|
||||||
|
sale_id = request.POST.get('sale_id')
|
||||||
|
phone = request.POST.get('phone')
|
||||||
|
pdf_data = None
|
||||||
|
|
||||||
|
if not sale_id:
|
||||||
|
return JsonResponse({'success': False, 'error': 'Sale ID missing'})
|
||||||
|
|
||||||
|
sale = get_object_or_404(Sale, pk=sale_id)
|
||||||
|
|
||||||
|
if not phone:
|
||||||
|
if sale.customer and sale.customer.phone:
|
||||||
|
phone = sale.customer.phone
|
||||||
|
else:
|
||||||
|
return JsonResponse({'success': False, 'error': 'Phone number missing'})
|
||||||
|
|
||||||
|
try:
|
||||||
|
# If PDF data is present, save and send document
|
||||||
|
if pdf_data:
|
||||||
|
# Remove header if present (data:application/pdf;base64,)
|
||||||
|
if ',' in pdf_data:
|
||||||
|
pdf_data = pdf_data.split(',')[1]
|
||||||
|
|
||||||
|
file_data = base64.b64decode(pdf_data)
|
||||||
|
dir_path = os.path.join(django_settings.MEDIA_ROOT, 'temp_invoices')
|
||||||
|
os.makedirs(dir_path, exist_ok=True)
|
||||||
|
|
||||||
|
filename = f"invoice_{sale.id}_{int(timezone.now().timestamp())}.pdf"
|
||||||
|
file_path = os.path.join(dir_path, filename)
|
||||||
|
|
||||||
|
with open(file_path, 'wb') as f:
|
||||||
|
f.write(file_data)
|
||||||
|
|
||||||
|
# Construct URL
|
||||||
|
file_url = request.build_absolute_uri(django_settings.MEDIA_URL + 'temp_invoices/' + filename)
|
||||||
|
|
||||||
|
success, response_msg = send_whatsapp_document(phone, file_url, caption=f"Invoice #{sale.invoice_number or sale.id}")
|
||||||
|
|
||||||
|
else:
|
||||||
|
# Fallback to Text Link
|
||||||
|
receipt_url = request.build_absolute_uri(reverse('sale_receipt', args=[sale.pk]))
|
||||||
|
|
||||||
|
message = (
|
||||||
|
f"Hello {sale.customer.name if sale.customer else 'Guest'},
|
||||||
|
"
|
||||||
|
f"Here is your invoice #{sale.invoice_number or sale.id}.
|
||||||
|
"
|
||||||
|
f"Total: {sale.total_amount}
|
||||||
|
"
|
||||||
|
f"View Invoice: {receipt_url}
|
||||||
|
"
|
||||||
|
f"Thank you for your business!"
|
||||||
|
)
|
||||||
|
|
||||||
|
success, response_msg = send_whatsapp_message(phone, message)
|
||||||
|
|
||||||
|
if success:
|
||||||
|
return JsonResponse({'success': True, 'message': response_msg})
|
||||||
|
else:
|
||||||
|
return JsonResponse({'success': False, 'error': response_msg})
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(f"WhatsApp Error: {e}")
|
||||||
|
return JsonResponse({'success': False, 'error': str(e)}) # Changed to str(e) for clarity
|
||||||
@ -358,6 +358,36 @@
|
|||||||
</div>
|
</div>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
<!-- Edit Category Modal -->
|
||||||
|
<div class="modal fade" id="editCategoryModal{{ category.id }}" tabindex="-1" aria-hidden="true">
|
||||||
|
<div class="modal-dialog">
|
||||||
|
<div class="modal-content rounded-4 border-0 shadow">
|
||||||
|
<div class="modal-header border-0 pb-0">
|
||||||
|
<h5 class="modal-title fw-bold">{% trans "Edit Category" %}</h5>
|
||||||
|
<button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
|
||||||
|
</div>
|
||||||
|
<form action="{% url 'edit_category' category.id %}" method="POST">
|
||||||
|
{% csrf_token %}
|
||||||
|
<div class="modal-body">
|
||||||
|
<div class="row g-3">
|
||||||
|
<div class="col-12">
|
||||||
|
<label class="form-label small fw-bold">{% trans "Name (English)" %}</label>
|
||||||
|
<input type="text" name="name_en" class="form-control rounded-3" value="{{ category.name_en }}" required>
|
||||||
|
</div>
|
||||||
|
<div class="col-12">
|
||||||
|
<label class="form-label small fw-bold">{% trans "Name (Arabic)" %}</label>
|
||||||
|
<input type="text" name="name_ar" class="form-control rounded-3" dir="rtl" value="{{ category.name_ar }}" required>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="modal-footer border-0">
|
||||||
|
<button type="button" class="btn btn-light rounded-3" data-bs-dismiss="modal">{% trans "Cancel" %}</button>
|
||||||
|
<button type="submit" class="btn btn-primary rounded-3 px-4">{% trans "Save Changes" %}</button>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
{% empty %}
|
{% empty %}
|
||||||
<tr>
|
<tr>
|
||||||
<td colspan="4" class="text-center py-4 text-muted">{% trans "No categories found." %}</td>
|
<td colspan="4" class="text-center py-4 text-muted">{% trans "No categories found." %}</td>
|
||||||
|
|||||||
@ -30,6 +30,13 @@
|
|||||||
<div class="card border-0 shadow-sm rounded-4 mb-4">
|
<div class="card border-0 shadow-sm rounded-4 mb-4">
|
||||||
<div class="card-body p-3">
|
<div class="card-body p-3">
|
||||||
<form method="get" class="row g-3">
|
<form method="get" class="row g-3">
|
||||||
|
<div class="col-md-3">
|
||||||
|
<label class="form-label small text-muted">{% trans "Search" %}</label>
|
||||||
|
<div class="input-group">
|
||||||
|
<span class="input-group-text bg-white border-end-0"><i class="bi bi-search text-muted"></i></span>
|
||||||
|
<input type="text" name="q" class="form-control border-start-0 rounded-end-3" placeholder="{% trans 'Customer or Invoice #' %}" value="{{ query|default:'' }}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
<div class="col-md-2">
|
<div class="col-md-2">
|
||||||
<label class="form-label small text-muted">{% trans "Start Date" %}</label>
|
<label class="form-label small text-muted">{% trans "Start Date" %}</label>
|
||||||
<input type="date" name="start_date" class="form-control rounded-3" value="{{ request.GET.start_date }}">
|
<input type="date" name="start_date" class="form-control rounded-3" value="{{ request.GET.start_date }}">
|
||||||
@ -49,16 +56,7 @@
|
|||||||
{% endfor %}
|
{% endfor %}
|
||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
<div class="col-md-2">
|
<div class="col-md-2 d-flex align-items-end">
|
||||||
<label class="form-label small text-muted">{% trans "Status" %}</label>
|
|
||||||
<select name="status" class="form-select rounded-3">
|
|
||||||
<option value="">{% trans "All Statuses" %}</option>
|
|
||||||
<option value="paid" {% if request.GET.status == 'paid' %}selected{% endif %}>{% trans "Paid" %}</option>
|
|
||||||
<option value="partial" {% if request.GET.status == 'partial' %}selected{% endif %}>{% trans "Partial" %}</option>
|
|
||||||
<option value="unpaid" {% if request.GET.status == 'unpaid' %}selected{% endif %}>{% trans "Unpaid" %}</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
<div class="col-md-3 d-flex align-items-end">
|
|
||||||
<button type="submit" class="btn btn-light w-100 me-2 rounded-3 shadow-sm">{% trans "Filter" %}</button>
|
<button type="submit" class="btn btn-light w-100 me-2 rounded-3 shadow-sm">{% trans "Filter" %}</button>
|
||||||
<a href="{% url 'invoices' %}" class="btn btn-outline-secondary w-100 rounded-3">{% trans "Reset" %}</a>
|
<a href="{% url 'invoices' %}" class="btn btn-outline-secondary w-100 rounded-3">{% trans "Reset" %}</a>
|
||||||
</div>
|
</div>
|
||||||
@ -88,7 +86,10 @@
|
|||||||
{{ sale.invoice_number|default:sale.id }}
|
{{ sale.invoice_number|default:sale.id }}
|
||||||
</td>
|
</td>
|
||||||
<td>{{ sale.created_at|date:"Y-m-d" }}</td>
|
<td>{{ sale.created_at|date:"Y-m-d" }}</td>
|
||||||
<td>{{ sale.customer.name|default:_("Guest") }}</td>
|
<td>
|
||||||
|
<div>{{ sale.customer.name|default:_("Guest") }}</div>
|
||||||
|
<div class="small text-muted" style="font-size: 0.75rem;">{{ sale.customer.phone|default:"" }}</div>
|
||||||
|
</td>
|
||||||
<td class="fw-bold text-dark">{{ site_settings.currency_symbol }}{{ sale.total_amount|floatformat:3 }}</td>
|
<td class="fw-bold text-dark">{{ site_settings.currency_symbol }}{{ sale.total_amount|floatformat:3 }}</td>
|
||||||
<td>
|
<td>
|
||||||
<span class="text-muted small">
|
<span class="text-muted small">
|
||||||
@ -147,7 +148,7 @@
|
|||||||
</div>
|
</div>
|
||||||
<div class="mb-3">
|
<div class="mb-3">
|
||||||
<label class="form-label small fw-bold">{% trans "Payment Method" %}</label>
|
<label class="form-label small fw-bold">{% trans "Payment Method" %}</label>
|
||||||
<select name="payment_method_id" class="form-select rounded-3 shadow-none">
|
<select name="payment_method" class="form-select rounded-3 shadow-none">
|
||||||
{% for method in payment_methods %}
|
{% for method in payment_methods %}
|
||||||
<option value="{{ method.id }}">{% if LANGUAGE_CODE == 'ar' %}{{ method.name_ar }}{% else %}{{ method.name_en }}{% endif %}</option>
|
<option value="{{ method.id }}">{% if LANGUAGE_CODE == 'ar' %}{{ method.name_ar }}{% else %}{{ method.name_en }}{% endif %}</option>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
|
|||||||
174
core/views.py
174
core/views.py
@ -15,6 +15,9 @@ import json
|
|||||||
import decimal
|
import decimal
|
||||||
import datetime
|
import datetime
|
||||||
import logging
|
import logging
|
||||||
|
import base64
|
||||||
|
import os
|
||||||
|
from django.conf import settings as django_settings
|
||||||
|
|
||||||
from .models import (
|
from .models import (
|
||||||
SystemSetting, Customer, Supplier, Product, Category, Unit,
|
SystemSetting, Customer, Supplier, Product, Category, Unit,
|
||||||
@ -28,7 +31,7 @@ from .forms import (
|
|||||||
SystemSettingForm, CustomerForm, SupplierForm, ProductForm, CategoryForm,
|
SystemSettingForm, CustomerForm, SupplierForm, ProductForm, CategoryForm,
|
||||||
UnitForm, ExpenseForm, CashierSessionStartForm, CashierSessionCloseForm
|
UnitForm, ExpenseForm, CashierSessionStartForm, CashierSessionCloseForm
|
||||||
)
|
)
|
||||||
from .utils import number_to_words_en
|
from .utils import number_to_words_en, send_whatsapp_message, send_whatsapp_document
|
||||||
from .views_import import *
|
from .views_import import *
|
||||||
|
|
||||||
logger = logging.getLogger(__name__)
|
logger = logging.getLogger(__name__)
|
||||||
@ -67,7 +70,8 @@ def index(request):
|
|||||||
|
|
||||||
# 3. Charts: Monthly Sales (Last 12 months)
|
# 3. Charts: Monthly Sales (Last 12 months)
|
||||||
last_12_months = timezone.now() - datetime.timedelta(days=365)
|
last_12_months = timezone.now() - datetime.timedelta(days=365)
|
||||||
monthly_sales = (Sale.objects.filter(created_at__gte=last_12_months)
|
monthly_sales = (
|
||||||
|
Sale.objects.filter(created_at__gte=last_12_months)
|
||||||
.annotate(month=TruncMonth('created_at'))
|
.annotate(month=TruncMonth('created_at'))
|
||||||
.values('month')
|
.values('month')
|
||||||
.annotate(total=Sum('total_amount'))
|
.annotate(total=Sum('total_amount'))
|
||||||
@ -78,7 +82,8 @@ def index(request):
|
|||||||
|
|
||||||
# 4. Charts: Daily Sales (Last 7 days)
|
# 4. Charts: Daily Sales (Last 7 days)
|
||||||
last_7_days = timezone.now() - datetime.timedelta(days=7)
|
last_7_days = timezone.now() - datetime.timedelta(days=7)
|
||||||
daily_sales = (Sale.objects.filter(created_at__gte=last_7_days)
|
daily_sales = (
|
||||||
|
Sale.objects.filter(created_at__gte=last_7_days)
|
||||||
.annotate(day=TruncDay('created_at'))
|
.annotate(day=TruncDay('created_at'))
|
||||||
.values('day')
|
.values('day')
|
||||||
.annotate(total=Sum('total_amount'))
|
.annotate(total=Sum('total_amount'))
|
||||||
@ -88,7 +93,8 @@ def index(request):
|
|||||||
chart_data = [float(d['total']) for d in daily_sales]
|
chart_data = [float(d['total']) for d in daily_sales]
|
||||||
|
|
||||||
# 5. Category Distribution
|
# 5. Category Distribution
|
||||||
category_dist = (SaleItem.objects.values('product__category__name_en', 'product__category__name_ar')
|
category_dist = (
|
||||||
|
SaleItem.objects.values('product__category__name_en', 'product__category__name_ar')
|
||||||
.annotate(total=Sum('line_total'))
|
.annotate(total=Sum('line_total'))
|
||||||
.order_by('-total')[:5])
|
.order_by('-total')[:5])
|
||||||
|
|
||||||
@ -96,7 +102,8 @@ def index(request):
|
|||||||
category_data = [float(c['total']) for c in category_dist]
|
category_data = [float(c['total']) for c in category_dist]
|
||||||
|
|
||||||
# 6. Payment Methods
|
# 6. Payment Methods
|
||||||
payment_dist = (SalePayment.objects.values('payment_method__name_en')
|
payment_dist = (
|
||||||
|
SalePayment.objects.values('payment_method__name_en')
|
||||||
.annotate(total=Sum('amount'))
|
.annotate(total=Sum('amount'))
|
||||||
.order_by('-total'))
|
.order_by('-total'))
|
||||||
|
|
||||||
@ -382,6 +389,15 @@ def invoice_list(request):
|
|||||||
if status:
|
if status:
|
||||||
sales = sales.filter(status=status)
|
sales = sales.filter(status=status)
|
||||||
|
|
||||||
|
# NEW: Search functionality
|
||||||
|
query = request.GET.get('q')
|
||||||
|
if query:
|
||||||
|
sales = sales.filter(
|
||||||
|
Q(customer__name__icontains=query) |
|
||||||
|
Q(customer__phone__icontains=query) |
|
||||||
|
Q(invoice_number__icontains=query)
|
||||||
|
)
|
||||||
|
|
||||||
paginator = Paginator(sales, 25)
|
paginator = Paginator(sales, 25)
|
||||||
|
|
||||||
settings = None
|
settings = None
|
||||||
@ -395,6 +411,7 @@ def invoice_list(request):
|
|||||||
'customers': Customer.objects.all(),
|
'customers': Customer.objects.all(),
|
||||||
'payment_methods': PaymentMethod.objects.filter(is_active=True),
|
'payment_methods': PaymentMethod.objects.filter(is_active=True),
|
||||||
'site_settings': settings,
|
'site_settings': settings,
|
||||||
|
'query': query, # Pass query back to template
|
||||||
}
|
}
|
||||||
return render(request, 'core/invoices.html', context)
|
return render(request, 'core/invoices.html', context)
|
||||||
|
|
||||||
@ -518,7 +535,7 @@ def sale_receipt(request, pk):
|
|||||||
'settings': settings
|
'settings': settings
|
||||||
})
|
})
|
||||||
|
|
||||||
# --- Quotations ---
|
# ---Quotations ---
|
||||||
|
|
||||||
@login_required
|
@login_required
|
||||||
def quotations(request):
|
def quotations(request):
|
||||||
@ -871,10 +888,10 @@ def cashflow_report(request):
|
|||||||
@login_required
|
@login_required
|
||||||
def add_product(request):
|
def add_product(request):
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
form = ProductForm(request.POST, request.FILES)
|
form = ProductForm(request.POST, request.FILES, instance=product)
|
||||||
if form.is_valid():
|
if form.is_valid():
|
||||||
form.save()
|
form.save()
|
||||||
messages.success(request, _("Product added."))
|
messages.success(request, _("Product updated."))
|
||||||
return redirect(reverse('inventory') + '#items')
|
return redirect(reverse('inventory') + '#items')
|
||||||
return redirect('inventory')
|
return redirect('inventory')
|
||||||
|
|
||||||
@ -916,11 +933,20 @@ def add_category(request):
|
|||||||
|
|
||||||
@login_required
|
@login_required
|
||||||
def edit_category(request, pk):
|
def edit_category(request, pk):
|
||||||
return redirect('inventory')
|
category = get_object_or_404(Category, pk=pk)
|
||||||
|
if request.method == 'POST':
|
||||||
|
category.name_en = request.POST.get('name_en')
|
||||||
|
category.name_ar = request.POST.get('name_ar')
|
||||||
|
category.save()
|
||||||
|
messages.success(request, _("Category updated."))
|
||||||
|
return redirect(reverse('inventory') + '#categories-list')
|
||||||
|
|
||||||
@login_required
|
@login_required
|
||||||
def delete_category(request, pk):
|
def delete_category(request, pk):
|
||||||
return redirect('inventory')
|
category = get_object_or_404(Category, pk=pk)
|
||||||
|
category.delete()
|
||||||
|
messages.success(request, _("Category deleted."))
|
||||||
|
return redirect(reverse('inventory') + '#categories-list')
|
||||||
|
|
||||||
@login_required
|
@login_required
|
||||||
def add_unit(request):
|
def add_unit(request):
|
||||||
@ -1046,7 +1072,76 @@ def test_whatsapp_connection(request):
|
|||||||
|
|
||||||
@login_required
|
@login_required
|
||||||
def send_invoice_whatsapp(request):
|
def send_invoice_whatsapp(request):
|
||||||
return JsonResponse({'success': True})
|
if request.method != 'POST':
|
||||||
|
return JsonResponse({'success': False, 'error': 'Method not allowed'})
|
||||||
|
|
||||||
|
try:
|
||||||
|
# Handle JSON payload
|
||||||
|
data = json.loads(request.body)
|
||||||
|
sale_id = data.get('sale_id')
|
||||||
|
phone = data.get('phone')
|
||||||
|
pdf_data = data.get('pdf_data') # Base64 string
|
||||||
|
except json.JSONDecodeError:
|
||||||
|
# Fallback to Form Data
|
||||||
|
sale_id = request.POST.get('sale_id')
|
||||||
|
phone = request.POST.get('phone')
|
||||||
|
pdf_data = None
|
||||||
|
|
||||||
|
if not sale_id:
|
||||||
|
return JsonResponse({'success': False, 'error': 'Sale ID missing'})
|
||||||
|
|
||||||
|
sale = get_object_or_404(Sale, pk=sale_id)
|
||||||
|
|
||||||
|
if not phone:
|
||||||
|
if sale.customer and sale.customer.phone:
|
||||||
|
phone = sale.customer.phone
|
||||||
|
else:
|
||||||
|
return JsonResponse({'success': False, 'error': 'Phone number missing'})
|
||||||
|
|
||||||
|
try:
|
||||||
|
# If PDF data is present, save and send document
|
||||||
|
if pdf_data:
|
||||||
|
# Remove header if present (data:application/pdf;base64,)
|
||||||
|
if ',' in pdf_data:
|
||||||
|
pdf_data = pdf_data.split(',')[1]
|
||||||
|
|
||||||
|
file_data = base64.b64decode(pdf_data)
|
||||||
|
dir_path = os.path.join(django_settings.MEDIA_ROOT, 'temp_invoices')
|
||||||
|
os.makedirs(dir_path, exist_ok=True)
|
||||||
|
|
||||||
|
filename = f"invoice_{sale.id}_{int(timezone.now().timestamp())}.pdf"
|
||||||
|
file_path = os.path.join(dir_path, filename)
|
||||||
|
|
||||||
|
with open(file_path, 'wb') as f:
|
||||||
|
f.write(file_data)
|
||||||
|
|
||||||
|
# Construct URL
|
||||||
|
file_url = request.build_absolute_uri(django_settings.MEDIA_URL + 'temp_invoices/' + filename)
|
||||||
|
|
||||||
|
success, response_msg = send_whatsapp_document(phone, file_url, caption=f"Invoice #{sale.invoice_number or sale.id}")
|
||||||
|
|
||||||
|
else:
|
||||||
|
# Fallback to Text Link
|
||||||
|
receipt_url = request.build_absolute_uri(reverse('sale_receipt', args=[sale.pk]))
|
||||||
|
|
||||||
|
message = (
|
||||||
|
f"Hello {sale.customer.name if sale.customer else 'Guest'},
|
||||||
|
f"Here is your invoice #{sale.invoice_number or sale.id}.\n"
|
||||||
|
f"Total: {sale.total_amount}\n"
|
||||||
|
f"View Invoice: {receipt_url}\n"
|
||||||
|
f"Thank you for your business!"
|
||||||
|
)
|
||||||
|
|
||||||
|
success, response_msg = send_whatsapp_message(phone, message)
|
||||||
|
|
||||||
|
if success:
|
||||||
|
return JsonResponse({'success': True, 'message': response_msg})
|
||||||
|
else:
|
||||||
|
return JsonResponse({'success': False, 'error': response_msg})
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(f"WhatsApp Error: {e}")
|
||||||
|
return JsonResponse({'success': False, 'error': str(e)}) # Changed to str(e) for clarity
|
||||||
|
|
||||||
# --- LPO ---
|
# --- LPO ---
|
||||||
@login_required
|
@login_required
|
||||||
@ -1212,8 +1307,57 @@ def create_sale_api(request):
|
|||||||
@csrf_exempt
|
@csrf_exempt
|
||||||
@login_required
|
@login_required
|
||||||
def update_sale_api(request, pk):
|
def update_sale_api(request, pk):
|
||||||
# Simplified update stub
|
if request.method != 'POST':
|
||||||
return JsonResponse({'success': True})
|
return JsonResponse({'success': False, 'error': 'Invalid request'})
|
||||||
|
|
||||||
|
try:
|
||||||
|
data = json.loads(request.body)
|
||||||
|
sale = get_object_or_404(Sale, pk=pk)
|
||||||
|
|
||||||
|
with transaction.atomic():
|
||||||
|
# 1. Restore Stock from OLD Items
|
||||||
|
for item in sale.items.all():
|
||||||
|
Product.objects.filter(pk=item.product_id).update(stock_quantity=F('stock_quantity') + item.quantity)
|
||||||
|
|
||||||
|
# 2. Delete OLD Items
|
||||||
|
sale.items.all().delete()
|
||||||
|
|
||||||
|
# 3. Update Sale Details
|
||||||
|
sale.customer_id = data.get('customer_id') or None
|
||||||
|
sale.total_amount = data.get('total_amount', 0)
|
||||||
|
sale.discount = data.get('discount', 0)
|
||||||
|
sale.notes = data.get('notes', '')
|
||||||
|
sale.invoice_number = data.get('invoice_number', sale.invoice_number)
|
||||||
|
sale.payment_type = data.get('payment_type', sale.payment_type)
|
||||||
|
sale.paid_amount = data.get('paid_amount', sale.paid_amount)
|
||||||
|
if data.get('due_date'):
|
||||||
|
sale.due_date = data.get('due_date')
|
||||||
|
sale.save()
|
||||||
|
|
||||||
|
# 4. Create NEW Items and Deduct Stock
|
||||||
|
for item_data in data.get('items', []):
|
||||||
|
qty = decimal.Decimal(str(item_data['quantity']))
|
||||||
|
price = decimal.Decimal(str(item_data['price']))
|
||||||
|
product_id = item_data['id']
|
||||||
|
|
||||||
|
SaleItem.objects.create(
|
||||||
|
sale=sale,
|
||||||
|
product_id=product_id,
|
||||||
|
quantity=qty,
|
||||||
|
unit_price=price,
|
||||||
|
line_total=qty * price
|
||||||
|
)
|
||||||
|
|
||||||
|
# Deduct Stock
|
||||||
|
Product.objects.filter(pk=product_id).update(stock_quantity=F('stock_quantity') - qty)
|
||||||
|
|
||||||
|
# 5. Handle Payment Update
|
||||||
|
sale.update_balance()
|
||||||
|
|
||||||
|
return JsonResponse({'success': True})
|
||||||
|
except Exception as e:
|
||||||
|
logger.error(f"Update Sale Error: {e}")
|
||||||
|
return JsonResponse({'success': False, 'error': str(e)})
|
||||||
|
|
||||||
@csrf_exempt
|
@csrf_exempt
|
||||||
@login_required
|
@login_required
|
||||||
@ -1333,7 +1477,9 @@ def add_customer_ajax(request):
|
|||||||
@login_required
|
@login_required
|
||||||
def search_customers_api(request):
|
def search_customers_api(request):
|
||||||
query = request.GET.get('q', '')
|
query = request.GET.get('q', '')
|
||||||
customers = Customer.objects.filter(name__icontains=query).values('id', 'name', 'phone')[:10]
|
customers = Customer.objects.filter(
|
||||||
|
Q(name__icontains=query) | Q(phone__icontains=query)
|
||||||
|
).values('id', 'name', 'phone')[:10]
|
||||||
return JsonResponse({'results': list(customers)})
|
return JsonResponse({'results': list(customers)})
|
||||||
|
|
||||||
@csrf_exempt
|
@csrf_exempt
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user