5 Commits

45 changed files with 740 additions and 1195 deletions

Binary file not shown.

View File

@@ -1,8 +1,8 @@
from flask import Blueprint, render_template, request, redirect, url_for # routes/gst_release_routes.py
from flask import Blueprint, render_template, request, redirect, url_for, flash
from flask_login import login_required from flask_login import login_required
from model.gst_release import GSTRelease from model.gst_release import GSTRelease
from model.Log import LogHelper from model.Log import LogHelper
from flask import flash, current_app
gst_release_bp = Blueprint('gst_release_bp', __name__) gst_release_bp = Blueprint('gst_release_bp', __name__)
gst_service = GSTRelease() gst_service = GSTRelease()
@@ -13,7 +13,7 @@ gst_service = GSTRelease()
def add_gst_release(): def add_gst_release():
if request.method == 'POST': if request.method == 'POST':
gst_service.AddGSTRelease(request) gst_service.AddGSTRelease(request)
LogHelper.log_action("Add GST Release", f"User added GST release") LogHelper.log_action("Add GST Release", "User added GST release")
flash(gst_service.resultMessage, 'success' if gst_service.isSuccess else 'error') flash(gst_service.resultMessage, 'success' if gst_service.isSuccess else 'error')
return redirect(url_for('gst_release_bp.add_gst_release')) return redirect(url_for('gst_release_bp.add_gst_release'))
@@ -30,7 +30,7 @@ def edit_gst_release(gst_release_id):
if request.method == 'POST': if request.method == 'POST':
gst_service.EditGSTRelease(request, gst_release_id) gst_service.EditGSTRelease(request, gst_release_id)
LogHelper.log_action("Edit GST Release", f"User edited GST release") LogHelper.log_action("Edit GST Release", "User edited GST release")
flash(gst_service.resultMessage, 'success' if gst_service.isSuccess else 'error') flash(gst_service.resultMessage, 'success' if gst_service.isSuccess else 'error')
return redirect(url_for('gst_release_bp.add_gst_release')) return redirect(url_for('gst_release_bp.add_gst_release'))
@@ -40,7 +40,7 @@ def edit_gst_release(gst_release_id):
@gst_release_bp.route('/delete_gst_release/<int:gst_release_id>', methods=['GET', 'POST']) @gst_release_bp.route('/delete_gst_release/<int:gst_release_id>', methods=['GET', 'POST'])
@login_required @login_required
def delete_gst_release(gst_release_id): def delete_gst_release(gst_release_id):
gst_service.DeleteGSTRelease(gst_release_id) # remove request gst_service.DeleteGSTRelease(gst_release_id)
LogHelper.log_action("Delete GST Release", f"User deleted GST release") LogHelper.log_action("Delete GST Release", "User deleted GST release")
flash(gst_service.resultMessage, 'success' if gst_service.isSuccess else 'error') flash(gst_service.resultMessage, 'success' if gst_service.isSuccess else 'error')
return redirect(url_for('gst_release_bp.add_gst_release')) return redirect(url_for('gst_release_bp.add_gst_release'))

View File

@@ -86,12 +86,12 @@ def edit_payment(payment_id):
return render_template('edit_payment.html', payment_data=payment_data) return render_template('edit_payment.html', payment_data=payment_data)
# ------------------- Delete Payment ------------------- # ------------------- Delete Payment -------------------
@payment_bp.route('/delete_payment/<int:payment_id>', methods=['POST']) @payment_bp.route('/delete_payment/<int:payment_id>', methods=['GET'])
@login_required @login_required
def delete_payment(payment_id): def delete_payment(payment_id):
success, pmc_no, invoice_no = Paymentmodel.delete_payment(payment_id) success, pmc_no, invoice_no = Paymentmodel.delete_payment(payment_id)
if not success: if not success:
flash("Payment not found or failed to delete", "error") flash("Payment not found or failed to delete", "error")
else: else:

View File

@@ -83,29 +83,17 @@ def check_village():
return village.CheckVillage(request=request) return village.CheckVillage(request=request)
# ------------------------- Delete Village -------------------------
@village_bp.route('/delete_village/<int:village_id>') @village_bp.route('/delete_village/<int:village_id>')
@login_required @login_required
def delete_village(village_id): def delete_village(village_id):
village = Village() village = Village()
village.DeleteVillage(request=request, village_id=village_id) village.DeleteVillage(request=request, village_id=village_id)
# ✅ Convert resultMessage to string if it's a Response or tuple flash(village.resultMessage, "success" if village.isSuccess else "error")
raw_msg = village.resultMessage return redirect(url_for('village.add_village'))
if isinstance(raw_msg, tuple):
# e.g., (<Response ...>, 200)
msg = "Village deleted successfully!"
elif hasattr(raw_msg, 'get_data'):
# Flask Response object
msg = raw_msg.get_data(as_text=True) # get raw text
else:
# fallback
msg = str(raw_msg) if raw_msg else "Village deleted successfully!"
return jsonify({
"status": "success" if village.isSuccess else "error",
"message": msg
})
# ------------------------- Edit Village ------------------------- # ------------------------- Edit Village -------------------------
@village_bp.route('/edit_village/<int:village_id>', methods=['GET', 'POST']) @village_bp.route('/edit_village/<int:village_id>', methods=['GET', 'POST'])

View File

@@ -6,7 +6,6 @@ import config
import re import re
import mysql.connector import mysql.connector
# ---------------------------------------------------------- # ----------------------------------------------------------
# Mapping Class # Mapping Class
# ---------------------------------------------------------- # ----------------------------------------------------------
@@ -23,10 +22,11 @@ class itemCRUDMapping:
self.name = "Hold Type" self.name = "Hold Type"
elif itemType is ItemCRUDType.Subcontractor: elif itemType is ItemCRUDType.Subcontractor:
self.name = "Subcontractor" self.name = "Subcontractor"
elif itemType.name == "GSTRelease":
self.name = "GSTRelease"
else: else:
self.name = "Item" self.name = "Item"
# ---------------------------------------------------------- # ----------------------------------------------------------
# Generic CRUD Class # Generic CRUD Class
# ---------------------------------------------------------- # ----------------------------------------------------------
@@ -93,13 +93,47 @@ class ItemCRUD:
try: try:
# ====================================================== # ======================================================
# SUBCONTRACTOR (MULTI-FIELD) # GSTRelease MULTI-FIELD
# ====================================================== # ======================================================
if data: if self.itemCRUDType.name == "GSTRelease" and data:
# Duplicate check (PMC_No + Invoice_No)
if storedprocfetch:
cursor.callproc(storedprocfetch, (data['PMC_No'], data['Invoice_No']))
existing_item = None
for rs in cursor.stored_results():
existing_item = rs.fetchone()
if existing_item:
self.isSuccess = False
self.resultMessage = HtmlHelper.json_response(
ResponseHandler.already_exists(self.itemCRUDMapping.name), 409
)
return
# Insert GSTRelease
cursor.callproc(storedprocadd, (
data['PMC_No'],
data['Invoice_No'],
data['Basic_Amount'],
data['Final_Amount'],
data['Total_Amount'],
data['UTR'],
data['Contractor_ID']
))
connection.commit()
self.isSuccess = True
self.resultMessage = HtmlHelper.json_response(
ResponseHandler.add_success(self.itemCRUDMapping.name), 200
)
return
# ======================================================
# SUBCONTRACTOR MULTI-FIELD
# ======================================================
if self.itemCRUDType.name == "Subcontractor" and data:
# Duplicate check
cursor.callproc(storedprocfetch, (data['Contractor_Name'],)) cursor.callproc(storedprocfetch, (data['Contractor_Name'],))
existing_item = None existing_item = None
for rs in cursor.stored_results(): for rs in cursor.stored_results():
existing_item = rs.fetchone() existing_item = rs.fetchone()
@@ -111,7 +145,6 @@ class ItemCRUD:
) )
return return
# Insert
cursor.callproc(storedprocadd, ( cursor.callproc(storedprocadd, (
data['Contractor_Name'], data['Contractor_Name'],
data['Address'], data['Address'],
@@ -123,17 +156,16 @@ class ItemCRUD:
data['GST_No'], data['GST_No'],
data['Contractor_password'] data['Contractor_password']
)) ))
connection.commit() connection.commit()
self.isSuccess = True self.isSuccess = True
self.resultMessage = HtmlHelper.json_response( self.resultMessage = HtmlHelper.json_response(
ResponseHandler.add_success(self.itemCRUDMapping.name), 200 ResponseHandler.add_success(self.itemCRUDMapping.name), 200
) )
return return
# ====================================================== # ======================================================
# NORMAL (Village / Block / State) # NORMAL SINGLE-FIELD (Village / Block / State)
# ====================================================== # ======================================================
if not re.match(RegEx.patternAlphabetOnly, childname): if not re.match(RegEx.patternAlphabetOnly, childname):
self.isSuccess = False self.isSuccess = False
@@ -142,7 +174,6 @@ class ItemCRUD:
) )
return return
# Duplicate check
if parentid is None: if parentid is None:
cursor.callproc(storedprocfetch, (childname,)) cursor.callproc(storedprocfetch, (childname,))
else: else:
@@ -159,17 +190,14 @@ class ItemCRUD:
) )
return return
# Insert
if parentid is None: if parentid is None:
cursor.callproc(storedprocadd, (childname,)) cursor.callproc(storedprocadd, (childname,))
else: else:
cursor.callproc(storedprocadd, (childname, parentid)) cursor.callproc(storedprocadd, (childname, parentid))
connection.commit() connection.commit()
self.isSuccess = True self.isSuccess = True
self.resultMessage = HtmlHelper.json_response( self.resultMessage = HtmlHelper.json_response(
ResponseHandler.add_success(self.itemCRUDMapping.name), 200 ResponseHandler.add_success(self.itemCRUDMapping.name), 200
) )
@@ -199,9 +227,33 @@ class ItemCRUD:
try: try:
# ====================================================== # ======================================================
# SUBCONTRACTOR (MULTI-FIELD) # GSTRelease MULTI-FIELD
# ====================================================== # ======================================================
if data: if self.itemCRUDType.name == "GSTRelease" and data:
cursor.callproc(storedprocupdate, (
childid,
data['PMC_No'],
data['Invoice_No'],
data['Basic_Amount'],
data['Final_Amount'],
data['Total_Amount'],
data['UTR'],
data['Contractor_ID']
))
connection.commit()
self.isSuccess = True
self.resultMessage = HtmlHelper.json_response(
ResponseHandler.update_success(self.itemCRUDMapping.name), 200
)
return
# ======================================================
# SUBCONTRACTOR MULTI-FIELD
# ======================================================
if self.itemCRUDType.name == "Subcontractor" and data:
cursor.callproc(storedprocupdate, ( cursor.callproc(storedprocupdate, (
childid, childid,
data['Contractor_Name'], data['Contractor_Name'],
@@ -214,9 +266,7 @@ class ItemCRUD:
data['GST_No'], data['GST_No'],
data['Contractor_password'] data['Contractor_password']
)) ))
connection.commit() connection.commit()
self.isSuccess = True self.isSuccess = True
self.resultMessage = HtmlHelper.json_response( self.resultMessage = HtmlHelper.json_response(
ResponseHandler.update_success(self.itemCRUDMapping.name), 200 ResponseHandler.update_success(self.itemCRUDMapping.name), 200
@@ -224,7 +274,7 @@ class ItemCRUD:
return return
# ====================================================== # ======================================================
# NORMAL # NORMAL SINGLE-FIELD
# ====================================================== # ======================================================
if not re.match(RegEx.patternAlphabetOnly, childname): if not re.match(RegEx.patternAlphabetOnly, childname):
self.isSuccess = False self.isSuccess = False
@@ -237,7 +287,6 @@ class ItemCRUD:
cursor.callproc(storedprocupdate, (childid, parentid, childname)) cursor.callproc(storedprocupdate, (childid, parentid, childname))
connection.commit() connection.commit()
self.isSuccess = True self.isSuccess = True
self.resultMessage = ResponseHandler.update_success(self.itemCRUDMapping.name)['message'] self.resultMessage = ResponseHandler.update_success(self.itemCRUDMapping.name)['message']
@@ -259,20 +308,15 @@ class ItemCRUD:
data = [] data = []
connection = config.get_db_connection() connection = config.get_db_connection()
if not connection: if not connection:
return [] return []
cursor = connection.cursor() cursor = connection.cursor()
try: try:
cursor.callproc(storedproc) cursor.callproc(storedproc)
for result in cursor.stored_results(): for result in cursor.stored_results():
data = result.fetchall() data = result.fetchall()
self.isSuccess = True self.isSuccess = True
except mysql.connector.Error as e: except mysql.connector.Error as e:
print(f"Error fetching {self.itemCRUDMapping.name}: {e}") print(f"Error fetching {self.itemCRUDMapping.name}: {e}")
self.isSuccess = False self.isSuccess = False
@@ -280,7 +324,6 @@ class ItemCRUD:
ResponseHandler.fetch_failure(self.itemCRUDMapping.name), 500 ResponseHandler.fetch_failure(self.itemCRUDMapping.name), 500
) )
return [] return []
finally: finally:
cursor.close() cursor.close()
connection.close() connection.close()
@@ -298,13 +341,10 @@ class ItemCRUD:
try: try:
cursor.callproc(storedproc, (id,)) cursor.callproc(storedproc, (id,))
for rs in cursor.stored_results(): for rs in cursor.stored_results():
data = rs.fetchone() data = rs.fetchone()
except mysql.connector.Error as e: except mysql.connector.Error as e:
print(f"Error fetching {self.itemCRUDMapping.name}: {e}") print(f"Error fetching {self.itemCRUDMapping.name}: {e}")
finally: finally:
cursor.close() cursor.close()
connection.close() connection.close()
@@ -353,7 +393,6 @@ class ItemCRUD:
return HtmlHelper.json_response( return HtmlHelper.json_response(
ResponseHandler.fetch_failure(self.itemCRUDMapping.name), 500 ResponseHandler.fetch_failure(self.itemCRUDMapping.name), 500
) )
finally: finally:
cursor.close() cursor.close()
connection.close() connection.close()

View File

@@ -8,7 +8,7 @@ class ItemCRUDType(Enum):
State = 4 State = 4
HoldType = 5 HoldType = 5
Subcontractor = 6 Subcontractor = 6
GSTRelease = 7
class RegEx: class RegEx:
patternAlphabetOnly = "^[A-Za-z ]+$" patternAlphabetOnly = "^[A-Za-z ]+$"

View File

@@ -1,109 +1,8 @@
# from flask import request # model/gst_release.py
# from model.ItemCRUD import ItemCRUD
# from model.Utilities import ItemCRUDType
# class GSTRelease:
# """CRUD operations for GST Release using ItemCRUD"""
# def __init__(self):
# self.isSuccess = False
# self.resultMessage = ""
# # ------------------- Add GST Release -------------------
# def AddGSTRelease(self, request):
# pmc_no = request.form.get('PMC_No', '').strip()
# invoice_no = request.form.get('invoice_No', '').strip()
# gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease)
# gst.AddItem(
# request=request,
# parentid=None,
# childname=f"{pmc_no}-{invoice_no}",
# storedprocfetch="CheckGSTReleaseExists",
# storedprocadd="AddGSTReleaseFromExcel" # your stored procedure handles extra fields
# )
# self.isSuccess = gst.isSuccess
# self.resultMessage = str(gst.resultMessage)
# # ------------------- Get All GST Releases -------------------
# def GetAllGSTReleases(self):
# gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease)
# # Pass request=None for fetch
# rows = gst.GetAllData(request=None, storedproc="GetAllGSTReleases")
# self.isSuccess = gst.isSuccess
# self.resultMessage = str(gst.resultMessage)
# data = []
# for row in rows:
# data.append({
# "gst_release_id": row[0],
# "pmc_no": row[1],
# "invoice_no": row[2],
# "basic_amount": row[3],
# "final_amount": row[4],
# "total_amount": row[5],
# "utr": row[6],
# "contractor_id": row[7]
# })
# return data
# # ------------------- Get GST Release By ID -------------------
# def GetGSTReleaseByID(self, gst_release_id):
# gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease)
# row = gst.GetDataByID(gst_release_id, request=None, storedproc="GetGSTReleaseById")
# self.isSuccess = gst.isSuccess
# self.resultMessage = str(gst.resultMessage)
# if row:
# return {
# "gst_release_id": row[0],
# "pmc_no": row[1],
# "invoice_no": row[2],
# "basic_amount": row[3],
# "final_amount": row[4],
# "total_amount": row[5],
# "utr": row[6],
# "contractor_id": row[7]
# }
# return None
# # ------------------- Edit GST Release -------------------
# def EditGSTRelease(self, request, gst_release_id):
# pmc_no = request.form.get('PMC_No', '').strip()
# invoice_no = request.form.get('invoice_No', '').strip()
# gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease)
# gst.EditItem(
# request=request,
# childid=gst_release_id,
# parentid=None,
# childname=f"{pmc_no}-{invoice_no}",
# storedprocupdate="UpdateGSTRelease" # stored procedure handles extra fields
# )
# self.isSuccess = gst.isSuccess
# self.resultMessage = str(gst.resultMessage)
# # ------------------- Delete GST Release -------------------
# def DeleteGSTRelease(self, gst_release_id):
# gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease)
# gst.DeleteItem(
# itemID=gst_release_id,
# request=None,
# storedprocDelete="DeleteGSTReleaseById"
# )
# self.isSuccess = gst.isSuccess
# self.resultMessage = str(gst.resultMessage)
from flask import request, jsonify from flask import request, jsonify
from model.ItemCRUD import ItemCRUD from model.ItemCRUD import ItemCRUD
from model.Utilities import ItemCRUDType from model.Utilities import ItemCRUDType
class GSTRelease: class GSTRelease:
def __init__(self): def __init__(self):
@@ -125,6 +24,7 @@ class GSTRelease:
"Contractor_ID": int(request.form.get("Contractor_ID", 0) or 0) "Contractor_ID": int(request.form.get("Contractor_ID", 0) or 0)
} }
# Add GST Release
gst.AddItem( gst.AddItem(
request=request, request=request,
data=data, data=data,
@@ -132,6 +32,12 @@ class GSTRelease:
storedprocadd="AddGSTReleaseFromExcel" storedprocadd="AddGSTReleaseFromExcel"
) )
# Check if addition was successful
if gst.isSuccess:
print(f"GST Release Added: {data}")
else:
print(f"Failed to add GST Release: {gst.resultMessage}")
self.isSuccess = gst.isSuccess self.isSuccess = gst.isSuccess
self.resultMessage = str(gst.resultMessage) self.resultMessage = str(gst.resultMessage)
@@ -198,7 +104,6 @@ class GSTRelease:
def GetAllGSTReleases(self): def GetAllGSTReleases(self):
try: try:
gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease) gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease)
rows = gst.GetAllData(None, "GetAllGSTReleases") rows = gst.GetAllData(None, "GetAllGSTReleases")
data = [] data = []
@@ -224,7 +129,6 @@ class GSTRelease:
def GetGSTReleaseByID(self, gst_release_id): def GetGSTReleaseByID(self, gst_release_id):
try: try:
gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease) gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease)
row = gst.GetDataByID(gst_release_id, "GetGSTReleaseById") row = gst.GetDataByID(gst_release_id, "GetGSTReleaseById")
if row: if row:

View File

@@ -1,160 +1,5 @@
// $(document).ready(function () { // Subcontractor autocomplete functionality
// // =============================== $(document).ready(function () {
// // FORM / TABLE TOGGLE
// // ===============================
// if ($('#addForm').length && $('#addTable').length) {
// $('#addForm').show();
// $('#addTable').hide();
// $('#addButton').click(function () {
// $('#addForm').show();
// $('#addTable').hide();
// });
// $('#displayButton').click(function () {
// $('#addForm').hide();
// $('#addTable').show();
// });
// }
// // ===============================
// // Subcontractor autocomplete
// // ===============================
// $("#subcontractor").keyup(function () {
// let query = $(this).val();
// if (query !== "") {
// $.ajax({
// url: "/search_subcontractor",
// method: "POST",
// data: { query: query },
// success: function (data) {
// $("#subcontractor_list").fadeIn().html(data);
// }
// });
// } else {
// $("#subcontractor_list").fadeOut();
// }
// });
// $(document).on("click", "li", function () {
// $("#subcontractor").val($(this).text());
// $("#subcontractor_id").val($(this).attr("data-id"));
// $("#subcontractor_list").fadeOut();
// });
// // Focus
// if (document.getElementById('subcontractor')) {
// document.getElementById('subcontractor').focus();
// }
// // ===============================
// // ADD INVOICE
// // ===============================
// if ($('#invoiceForm').length && window.location.href.includes('add_invoice')) {
// $("#invoiceForm").on("submit", function (e) {
// e.preventDefault();
// let formData = $(this).serialize();
// $.ajax({
// url: '/add_invoice',
// method: 'POST',
// data: formData,
// dataType: 'json',
// success: function (response) {
// if (response.status === "success") {
// alert(response.message || "Invoice added successfully!");
// $('#invoiceForm')[0].reset(); // clear form
// $('#addForm').hide();
// $('#addTable').show(); // switch to table
// location.reload(); // optional refresh
// } else {
// alert(response.message || "Error adding invoice.");
// }
// },
// error: function (xhr) {
// alert(xhr.responseJSON?.message || "Submission failed. Please try again.");
// }
// });
// });
// }
// // Example AJAX update function
// function updateInvoice(invoiceId, formElement) {
// const formData = $(formElement).serialize();
// $.ajax({
// url: '/update_invoice/' + invoiceId,
// method: 'POST',
// data: formData,
// dataType: 'json',
// success: function(response) {
// if(response.status === "success") {
// alert(response.message || "Invoice updated successfully!");
// // ✅ Hide Add Form, Show Table
// $('#addForm').hide();
// $('#addTable').show();
// // Optional: reload table or refresh page
// location.reload();
// } else {
// alert(response.message || "Update failed. Please try again.");
// }
// },
// error: function(xhr) {
// alert(xhr.responseJSON?.message || "Error updating invoice.");
// }
// });
// }
// // ===============================
// // DELETE INVOICE
// // ===============================
// function deleteInvoice(invoiceId, element) {
// if (!confirm("Are you sure you want to delete this invoice?")) return;
// $.ajax({
// url: '/delete_invoice/' + invoiceId,
// method: 'GET',
// dataType: 'json',
// success: function (response) {
// alert(response.message || "Invoice deleted successfully!");
// if (element) $(element).closest("tr").remove();
// },
// error: function (xhr) {
// alert(xhr.responseJSON?.message || "Error deleting invoice. Please try again.");
// }
// });
// }
$(document).ready(function () {
// ===============================
// FORM / TABLE TOGGLE
// ===============================
if ($('#addForm').length && $('#addTable').length) {
// Default: show form, hide table
$('#addForm').show();
$('#addTable').hide();
// ✅ Check URL hash to show table instead
if (window.location.hash === "#addTable") {
$('#addForm').hide();
$('#addTable').show();
}
$('#addButton').click(function () {
$('#addForm').show();
$('#addTable').hide();
});
$('#displayButton').click(function () {
$('#addForm').hide();
$('#addTable').show();
});
}
// ===============================
// Subcontractor autocomplete
// ===============================
$("#subcontractor").keyup(function () { $("#subcontractor").keyup(function () {
let query = $(this).val(); let query = $(this).val();
if (query !== "") { if (query !== "") {
@@ -176,96 +21,42 @@ $(document).ready(function () {
$("#subcontractor_id").val($(this).attr("data-id")); $("#subcontractor_id").val($(this).attr("data-id"));
$("#subcontractor_list").fadeOut(); $("#subcontractor_list").fadeOut();
}); });
});
// Focus // Success Alert: show alert and reload after 3 seconds
if (document.getElementById('subcontractor')) { function showSuccessAlert() {
document.getElementById('subcontractor').focus(); const alertBox = document.getElementById("invoiceSuccessAlert");
alertBox.classList.add("show");
setTimeout(() => {
alertBox.classList.remove("show");
// Reload page or redirect after alert hides
window.location.href = '/add_invoice';
}, 3000);
} }
// =============================== // Submit form via AJAX
// ADD INVOICE
// ===============================
if ($('#invoiceForm').length && window.location.href.includes('add_invoice')) {
$("#invoiceForm").on("submit", function (e) { $("#invoiceForm").on("submit", function (e) {
e.preventDefault(); e.preventDefault();
let formData = $(this).serialize(); let formData = $(this).serialize();
$.ajax({ $.ajax({
url: '/add_invoice', url: '/add_invoice',
method: 'POST', method: 'POST',
data: formData, data: formData,
dataType: 'json',
success: function (response) { success: function (response) {
if (response.status === "success") {
alert(response.message || "Invoice added successfully!");
$('#invoiceForm')[0].reset(); // clear form
$('#addForm').hide();
$('#addTable').show(); // switch to table
location.reload(); // optional refresh
} else {
alert(response.message || "Error adding invoice.");
}
},
error: function (xhr) {
let msg = xhr.responseJSON?.message || "Submission failed. Please try again.";
alert(msg);
}
});
});
}
// ===============================
// UPDATE INVOICE
// ===============================
function updateInvoice(invoiceId, formElement) {
const formData = $(formElement).serialize();
$.ajax({
url: '/update_invoice/' + invoiceId,
method: 'POST',
data: formData,
dataType: 'json',
success: function(response) {
if(response.status === "success") { if(response.status === "success") {
alert(response.message || "Invoice updated successfully!"); showSuccessAlert();
// Redirect to Add Invoice page's table part
window.location.href = "/add_invoice#addTable";
} else { } else {
alert(response.message || "Update failed. Please try again."); alert(response.message);
} }
}, },
error: function(xhr) { error: function (xhr, status, error) {
let msg = xhr.responseJSON?.message || "Error updating invoice."; alert("Submission failed: " + error);
alert(msg);
} }
}); });
}
window.updateInvoice = updateInvoice; // make globally accessible
// ===============================
// DELETE INVOICE
// ===============================
function deleteInvoice(invoiceId, element) {
if (!confirm("Are you sure you want to delete this invoice?")) return;
$.ajax({
url: '/delete_invoice/' + invoiceId,
method: 'GET',
dataType: 'json',
success: function (response) {
if (response.status === "success") {
alert(response.message || "Invoice deleted successfully!");
if (element) $(element).closest("tr").remove();
} else {
alert(response.message || "Error deleting invoice.");
}
},
error: function (xhr) {
let msg = xhr.responseJSON?.message || "Error deleting invoice. Please try again.";
alert(msg);
}
}); });
}
window.deleteInvoice = deleteInvoice; // make globally accessible
});
window.onload = function () {
document.getElementById('subcontractor').focus();
};

View File

@@ -1,37 +1,41 @@
window.onload = function () { window.onload = function () {
if (document.getElementById('Village_Name')) {
document.getElementById('Village_Name').focus(); document.getElementById('Village_Name').focus();
}
}; };
$(document).ready(function () { $(document).ready(function () {
// RUN ONLY IF THIS PAGE HAS FORM/TABLE // 🔥 RESTORE VIEW MODE AFTER RELOAD
if ($('#addForm').length && $('#addTable').length) { var viewMode = localStorage.getItem("viewMode");
// ✅ DEFAULT VIEW → SHOW FORM if (viewMode === "table") {
$('#addForm').hide();
$('#addTable').show();
} else {
$('#addForm').show(); $('#addForm').show();
$('#addTable').hide(); $('#addTable').hide();
}
// 🔥 BUTTON TOGGLE // 🔥 BUTTON TOGGLE LOGIC
$('#addButton').click(function () { $('#addButton').click(function () {
$('#addForm').show(); $('#addForm').show();
$('#addTable').hide(); $('#addTable').hide();
localStorage.setItem("viewMode", "form");
}); });
$('#displayButton').click(function () { $('#displayButton').click(function () {
$('#addForm').hide(); $('#addForm').hide();
$('#addTable').show(); $('#addTable').show();
localStorage.setItem("viewMode", "table");
}); });
}
// ===============================
// STATE → DISTRICT // STATE → DISTRICT
// ===============================
if ($('#state_Id').length) {
$('#state_Id').change(function () { $('#state_Id').change(function () {
var stateId = $(this).val(); var stateId = $(this).val();
@@ -50,24 +54,25 @@ $(document).ready(function () {
districtDropdown.append('<option value="" disabled selected>Select District</option>'); districtDropdown.append('<option value="" disabled selected>Select District</option>');
data.forEach(function (district) { data.forEach(function (district) {
districtDropdown.append( districtDropdown.append(
'<option value="' + district.id + '">' + district.name + '</option>' '<option value="' + district.id + '">' + district.name + '</option>'
); );
}); });
districtDropdown.prop('disabled', false); districtDropdown.prop('disabled', false);
}
});
}
});
} }
});
}
});
// ===============================
// DISTRICT → BLOCK // DISTRICT → BLOCK
// ===============================
if ($('#district_Id').length) {
$('#district_Id').change(function () { $('#district_Id').change(function () {
var districtId = $(this).val(); var districtId = $(this).val();
@@ -86,24 +91,25 @@ $(document).ready(function () {
blockDropdown.append('<option value="" disabled selected>Select Block</option>'); blockDropdown.append('<option value="" disabled selected>Select Block</option>');
data.forEach(function (block) { data.forEach(function (block) {
blockDropdown.append( blockDropdown.append(
'<option value="' + block.id + '">' + block.name + '</option>' '<option value="' + block.id + '">' + block.name + '</option>'
); );
}); });
blockDropdown.prop('disabled', false); blockDropdown.prop('disabled', false);
}
});
}
});
} }
});
}
});
// ===============================
// VILLAGE NAME VALIDATION // VILLAGE NAME VALIDATION
// ===============================
if ($('#Village_Name').length) {
$('#Village_Name').on('input', function () { $('#Village_Name').on('input', function () {
var villageName = $(this).val(); var villageName = $(this).val();
@@ -121,16 +127,13 @@ $(document).ready(function () {
$('#villageMessage').text(''); $('#villageMessage').text('');
$('#submitVillage').prop('disabled', false); $('#submitVillage').prop('disabled', false);
} }
}); });
}
// ===============================
// CHECK DUPLICATE VILLAGE // CHECK DUPLICATE VILLAGE
// ===============================
if ($('#Village_Name').length && $('#block_Id').length) {
$('#Village_Name, #block_Id').on('change keyup', function () { $('#Village_Name, #block_Id').on('change keyup', function () {
var blockId = $('#block_Id').val(); var blockId = $('#block_Id').val();
@@ -139,6 +142,7 @@ $(document).ready(function () {
if (blockId && villageName) { if (blockId && villageName) {
$.ajax({ $.ajax({
url: '/check_village', url: '/check_village',
type: 'POST', type: 'POST',
@@ -164,7 +168,9 @@ $(document).ready(function () {
.css('color', 'green'); .css('color', 'green');
$('#submitVillage').prop('disabled', false); $('#submitVillage').prop('disabled', false);
} }
}, },
error: function () { error: function () {
@@ -174,77 +180,85 @@ $(document).ready(function () {
.css('color', 'red'); .css('color', 'red');
$('#submitVillage').prop('disabled', true); $('#submitVillage').prop('disabled', true);
} }
}); });
} }
}); });
}
// =============================== // ADD VILLAGE
// ADD VILLAGE (SAFE SCOPE FIX)
// ===============================
if ($('#villageForm').length) {
$('#villageForm').submit(function (event) { $('#villageForm').submit(function (event) {
event.preventDefault(); event.preventDefault();
$.ajax({ $.ajax({
url: '/add_village', url: '/add_village',
type: 'POST', type: 'POST',
data: $(this).serialize(), data: $(this).serialize(),
success: function (response) { success: function (response) {
if (response && response.status === 'success') { if (response.status === 'success') {
alert(response.message || 'Village added successfully!'); alert('Village added successfully!');
// ✅ clear form
$('#villageForm')[0].reset();
// ✅ switch to table
$('#addForm').hide();
$('#addTable').show();
// optional refresh
location.reload(); location.reload();
} else { } else {
alert(response.message || 'Error adding village. Please try again.'); alert(response.message || 'Error adding village. Please try again.');
} }
}, },
error: function () { error: function () {
alert('An error occurred. Please try again.'); alert('An error occurred. Please try again.');
} }
}); });
}); });
}
}); });
// 🔥 DELETE FUNCTION (UPDATED)
function deleteVillage(villageId) {
if (!confirm("Are you sure you want to delete this village?")) {
return;
}
// =============================== // ✅ save that user is on table
// DELETE FUNCTION (SAFE) localStorage.setItem("viewMode", "table");
// ===============================
function deleteVillage(villageId, element) {
if (!confirm("Are you sure you want to delete this village?")) return;
$.ajax({ $.ajax({
url: '/delete_village/' + villageId, url: '/delete_village/' + villageId,
type: 'GET', type: 'GET',
dataType: 'json',
success: function (response) { success: function () {
alert(response.message); // ✅ now shows "Village deleted successfully!"
if (element) $(element).closest("tr").remove(); setTimeout(function () {
alert("Village deleted successfully!");
// reload but stay on table
location.reload();
}, 1000);
}, },
error: function () { error: function () {
alert("Error deleting village. Please try again."); alert("Error deleting village. Please try again.");
} }
}); });
} }

View File

@@ -68,10 +68,11 @@
{% endwith %} {% endwith %}
</div> </div>
<!-- GST Release History Table -->
<div id="addTable" style="display: none;"> <div id="addTable" style="display: none;">
<div class="search-container"> <div class="search-container">
<h2>GST Release History</h2> <h2>GST Release History</h2>
<input type="text" id="searchBar" placeholder="Searching..." onkeyup="searchTable()"> <input type="text" id="searchBar" placeholder="Search..." onkeyup="searchTable()">
</div> </div>
<table id="sortableTable" border="1"> <table id="sortableTable" border="1">
<thead> <thead>
@@ -90,24 +91,22 @@
<tbody> <tbody>
{% for gst_rel in gst_releases %} {% for gst_rel in gst_releases %}
<tr> <tr>
<td>{{ gst_rel[0] }}</td> <td>{{ gst_rel.gst_release_id }}</td>
<td>{{ gst_rel[1] }}</td> <td>{{ gst_rel.pmc_no }}</td>
<td>{{ gst_rel[2] }}</td> <td>{{ gst_rel.invoice_no }}</td>
<td>{{ gst_rel[3] }}</td> <td>{{ gst_rel.basic_amount }}</td>
<td>{{ gst_rel[4] }}</td> <td>{{ gst_rel.final_amount }}</td>
<td>{{ gst_rel[5] }}</td> <td>{{ gst_rel.total_amount }}</td>
<td>{{ gst_rel[6] }}</td> <td>{{ gst_rel.utr }}</td>
<td> <td>
<a href="/edit_gst_release/{{ gst_rel[0] }}"> <a href="{{ url_for('gst_release_bp.edit_gst_release', gst_release_id=gst_rel.gst_release_id) }}">
<img src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit" <img src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit" class="icon">
class="icon">
</a> </a>
</td> </td>
<td> <td>
<a href="/delete_gst_release/{{ gst_rel[0] }}" <a href="{{ url_for('gst_release_bp.delete_gst_release', gst_release_id=gst_rel.gst_release_id) }}"
onclick="return confirm('Are you sure you want to delete this GST Release?')"> onclick="return confirm('Are you sure you want to delete this GST Release?')">
<img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" alt="Delete" <img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" alt="Delete" class="icon">
class="icon">
</a> </a>
</td> </td>
</tr> </tr>

View File

@@ -1,67 +1,48 @@
{% extends 'base.html' %} {% block content %} {% extends 'base.html' %}
{% block content %}
<head xmlns="http://www.w3.org/1999/html"> <head xmlns="http://www.w3.org/1999/html">
<meta charset="UTF-8" /> <meta charset="UTF-8"/>
<meta name="viewport" content="width=device-width, initial-scale=1.0" /> <meta name="viewport" content="width=device-width, initial-scale=1.0"/>
<title>Add Invoice</title> <title>Add Invoice</title>
<link <link rel="stylesheet" type="text/css" href="{{ url_for('static', filename='css/invoice.css') }}">
rel="stylesheet"
type="text/css"
href="{{ url_for('static', filename='css/invoice.css') }}"
/>
<script src="{{ url_for('static', filename='js/invoice.js') }}"></script> <script src="{{ url_for('static', filename='js/invoice.js') }}"></script>
<script src="{{ url_for('static', filename='js/holdAmount.js') }}"></script> <script src="{{ url_for('static', filename='js/holdAmount.js') }}"></script>
<script src="{{ url_for('static', filename='js/search_on_table.js') }}"></script> <script src="{{ url_for('static', filename='js/search_on_table.js') }}"></script>
</head> </head>
<body> <body>
{% if success == 'true' %} {% if success == 'true' %}
<div <div class="alert alert-success alert-dismissible fade show mt-3" role="alert">
class="alert alert-success alert-dismissible fade show mt-3"
role="alert"
>
✅ Invoice added successfully! ✅ Invoice added successfully!
<button <button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button>
type="button" </div>
class="btn-close" {% endif %}
data-bs-dismiss="alert"
aria-label="Close"
></button>
</div>
{% endif %}
<!-- Flash Messages -->
{% with messages = get_flashed_messages(with_categories=true) %} {% if <!-- Flash Messages -->
messages %} {% with messages = get_flashed_messages(with_categories=true) %}
{% if messages %}
<div class="flash-messages"> <div class="flash-messages">
{% for category, message in messages %} {% for category, message in messages %}
<div class="alert {{ category }}">{{ message }}</div> <div class="alert {{ category }}">{{ message }}</div>
{% endfor %} {% endfor %}
</div> </div>
{% endif %} {% endwith %} {% endif %}
{% endwith %}
<div class="button-container"> <div class="button-container">
<button id="addButton" class="action-button">Add</button> <button id="addButton" class="action-button">Add</button>
<button id="displayButton" class="action-button">Display</button> <button id="displayButton" class="action-button">Display</button>
</div> </div>
<div id="addForm" style="display: none"> <div id="addForm" style="display: none;">
<h2>Add Invoice</h2> <h2>Add Invoice</h2>
<form <form id="invoiceForm" action="{{ url_for('invoice.add_invoice') }}" method="POST">
id="invoiceForm"
action="{{ url_for('invoice.add_invoice') }}"
method="POST"
>
<div class="row1"> <div class="row1">
<div> <div>
<label for="subcontractor">Subcontractor Name:</label> <label for="subcontractor">Subcontractor Name:</label>
<input <input type="text" id="subcontractor" name="subcontractor" required autocomplete="off"/>
type="text" <input type="hidden" id="subcontractor_id" name="subcontractor_id"/>
id="subcontractor"
name="subcontractor"
required
autocomplete="off"
/>
<input type="hidden" id="subcontractor_id" name="subcontractor_id" />
<div id="subcontractor_list"></div> <div id="subcontractor_list"></div>
</div> </div>
</div> </div>
@@ -72,15 +53,13 @@
<select id="village" name="village" required> <select id="village" name="village" required>
<option value="">-- Select Village --</option> <option value="">-- Select Village --</option>
{% for village in villages %} {% for village in villages %}
<option value="{{ village.Village_Name }}"> <option value="{{ village.Village_Name }}">{{ village.Village_Name }}</option>
{{ village.Village_Name }}
</option>
{% endfor %} {% endfor %}
</select> </select>
</div> </div>
<div> <div>
<label for="pmc_no">PMC No:</label> <label for="pmc_no">PMC No:</label>
<input type="text" id="pmc_no" name="pmc_no" required /> <input type="text" id="pmc_no" name="pmc_no" required/>
<div id="pmc_no_list" class="autocomplete-list"></div> <div id="pmc_no_list" class="autocomplete-list"></div>
</div> </div>
</div> </div>
@@ -88,187 +67,81 @@
<div class="row2"> <div class="row2">
<div> <div>
<label for="work_type">Work Type:</label> <label for="work_type">Work Type:</label>
<input type="text" id="work_type" name="work_type" required /> <input type="text" id="work_type" name="work_type" required/>
</div> </div>
<div> <div>
<label for="invoice_details">Invoice Details:</label> <label for="invoice_details">Invoice Details:</label>
<textarea <textarea id="invoice_details" name="invoice_details" required></textarea>
id="invoice_details"
name="invoice_details"
required
></textarea>
</div> </div>
</div> </div>
<div class="row2"> <div class="row2">
<div> <div>
<label for="invoice_no">Invoice No:</label> <label for="invoice_no">Invoice No:</label>
<input type="text" id="invoice_no" name="invoice_no" required /> <input type="text" id="invoice_no" name="invoice_no" required/>
</div> </div>
<div> <div>
<label for="invoice_date">Invoice Date:</label> <label for="invoice_date">Invoice Date:</label>
<input type="date" id="invoice_date" name="invoice_date" required /> <input type="date" id="invoice_date" name="invoice_date" required/>
</div> </div>
</div> </div>
<div class="row3"> <div class="row3">
<div> <div>
<label for="basic_amount">Basic Amount:</label> <label for="basic_amount">Basic Amount:</label>
<input <input type="number" step="0.01" id="basic_amount" name="basic_amount" placeholder="₹ - 00.00" required/>
type="number"
step="0.01"
id="basic_amount"
name="basic_amount"
placeholder="₹ - 00.00"
required
/>
</div> </div>
<div> <div>
<label for="debit_amount">Debit Amount:</label> <label for="debit_amount">Debit Amount:</label>
<input <input type="number" step="0.01" id="debit_amount" name="debit_amount" placeholder="₹ - 00.00" required/>
type="number"
step="0.01"
id="debit_amount"
name="debit_amount"
placeholder="₹ - 00.00"
required
/>
</div> </div>
<div> <div>
<label for="after_debit_amount">After Debit Amount:</label> <label for="after_debit_amount">After Debit Amount:</label>
<input <input type="number" step="0.01" id="after_debit_amount" name="after_debit_amount" placeholder="₹ - 00.00" readonly required/>
type="number"
step="0.01"
id="after_debit_amount"
name="after_debit_amount"
placeholder="₹ - 00.00"
readonly
required
/>
</div>
</div> </div>
</div>
<div class="row3"> <div class="row3">
<div class="percentage-field"> <div class="percentage-field">
<label for="gst_percentage">GST %:</label> <label for="gst_percentage">GST %:</label>
<input <input type="number" step="0.01" id="gst_percentage" name="gst_percentage" placeholder="%"/>
type="number"
step="0.01"
id="gst_percentage"
name="gst_percentage"
placeholder="%"
/>
<label for="gst_amount">GST Amount:</label> <label for="gst_amount">GST Amount:</label>
<input <input type="number" step="0.01" id="gst_amount" name="gst_amount" placeholder="₹ - 00.00" readonly required/>
type="number"
step="0.01"
id="gst_amount"
name="gst_amount"
placeholder="₹ - 00.00"
readonly
required
/>
</div> </div>
<div> <div>
<label for="amount">Amount:</label> <label for="amount">Amount:</label>
<input <input type="number" step="0.01" id="amount" name="amount" placeholder="₹ - 00.00" readonly required/>
type="number"
step="0.01"
id="amount"
name="amount"
placeholder="₹ - 00.00"
readonly
required
/>
</div> </div>
<div class="percentage-field"> <div class="percentage-field">
<label for="tds_percentage">TDS %:</label> <label for="tds_percentage">TDS %:</label>
<input <input type="number" step="0.01" id="tds_percentage" name="tds_percentage" placeholder="%"/>
type="number"
step="0.01"
id="tds_percentage"
name="tds_percentage"
placeholder="%"
/>
<label for="tds_amount">TDS Amount:</label> <label for="tds_amount">TDS Amount:</label>
<input <input type="number" step="0.01" id="tds_amount" name="tds_amount" placeholder="₹ - 00.00" readonly required/>
type="number"
step="0.01"
id="tds_amount"
name="tds_amount"
placeholder="₹ - 00.00"
readonly
required
/>
</div>
</div> </div>
</div>
<div class="row3"> <div class="row3">
<div class="percentage-field"> <div class="percentage-field">
<label for="sd_percentage">SD %:</label> <label for="sd_percentage">SD %:</label>
<input <input type="number" step="0.01" id="sd_percentage" name="sd_percentage" placeholder="%"/>
type="number"
step="0.01"
id="sd_percentage"
name="sd_percentage"
placeholder="%"
/>
<label for="sd_amount">SD Amount:</label> <label for="sd_amount">SD Amount:</label>
<input <input type="number" step="0.01" id="sd_amount" name="sd_amount" placeholder="₹ - 00.00" readonly required>
type="number"
step="0.01"
id="sd_amount"
name="sd_amount"
placeholder="₹ - 00.00"
readonly
required
/>
</div> </div>
<div class="percentage-field"> <div class="percentage-field">
<label for="commission_percentage">On Commission %:</label> <label for="commission_percentage">On Commission %:</label>
<input <input type="number" step="0.01" id="commission_percentage" name="commission_percentage" placeholder="%"/>
type="number"
step="0.01"
id="commission_percentage"
name="commission_percentage"
placeholder="%"
/>
<label for="on_commission">On Commission:</label> <label for="on_commission">On Commission:</label>
<input <input type="number" step="0.01" id="on_commission" name="on_commission" placeholder="₹ - 00.00" readonly required>
type="number"
step="0.01"
id="on_commission"
name="on_commission"
placeholder="₹ - 00.00"
readonly
required
/>
</div> </div>
<div class="percentage-field"> <div class="percentage-field">
<label for="hydro_percentage">Hydro Testing %:</label> <label for="hydro_percentage">Hydro Testing %:</label>
<input <input type="number" step="0.01" id="hydro_percentage" name="hydro_percentage" placeholder="%"/>
type="number"
step="0.01"
id="hydro_percentage"
name="hydro_percentage"
placeholder="%"
/>
<label for="hydro_testing">Hydro Testing:</label> <label for="hydro_testing">Hydro Testing:</label>
<input <input type="number" step="0.01" id="hydro_testing" name="hydro_testing" placeholder="₹ - 00.00" readonly required>
type="number"
step="0.01"
id="hydro_testing"
name="hydro_testing"
placeholder="₹ - 00.00"
readonly
required
/>
</div>
</div> </div>
</div>
<div class="hold-row"> <div class="hold-row">
<button type="button" id="add_hold_amount" class="button"> <button type="button" id="add_hold_amount" class="button">+ Add Hold Amount</button>
+ Add Hold Amount
</button>
</div> </div>
<!-- Dynamically added hold amount fields --> <!-- Dynamically added hold amount fields -->
@@ -277,42 +150,23 @@
<div class="row2"> <div class="row2">
<div> <div>
<label for="gst_sd_amount">GST SD Amount:</label> <label for="gst_sd_amount">GST SD Amount:</label>
<input <input type="number" step="0.01" id="gst_sd_amount" name="gst_sd_amount" placeholder="₹ - 00.00" required/>
type="number"
step="0.01"
id="gst_sd_amount"
name="gst_sd_amount"
placeholder="₹ - 00.00"
required
/>
</div> </div>
<div> <div>
<label for="final_amount">Final Amount:</label> <label for="final_amount">Final Amount:</label>
<input <input type="number" step="0.01" id="final_amount" name="final_amount" placeholder="₹ - 00.00" required/>
type="number"
step="0.01"
id="final_amount"
name="final_amount"
placeholder="₹ - 00.00"
required
/>
</div> </div>
</div> </div>
<button type="submit" class="button">Submit</button> <button type="submit" class="button">Submit</button>
</form> </form>
</div> </div>
<div id="addTable" style="display: none"> <div id="addTable" style="display: none;">
<!-- Invoice Table Section --> <!-- Invoice Table Section -->
<div class="search-container"> <div class="search-container">
<h2>Invoice List</h2> <h2>Invoice List</h2>
<input <input type="text" id="searchBar" placeholder="Searching..." onkeyup="searchTable()">
type="text"
id="searchBar"
placeholder="Searching..."
onkeyup="searchTable()"
/>
{% if invoices %} {% if invoices %}
<table class="invoice-table"> <table class="invoice-table">
<thead> <thead>
@@ -364,27 +218,14 @@
<td>{{ invoice.Final_Amount }}</td> <td>{{ invoice.Final_Amount }}</td>
<td> <td>
<!-- Edit --> <!-- Edit -->
<a <a href="{{ url_for('invoice.edit_invoice', invoice_id=invoice.Invoice_Id) }}">
href="{{ url_for('invoice.edit_invoice', invoice_id=invoice.Invoice_Id) }}" <img src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit" class="icon">
>
<img
src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}"
alt="Edit"
class="icon edit-btn"
data-id="{{ invoice.Invoice_Id }}"
/>
</a> </a>
</td> </td>
<td> <td>
<a <!-- Delete -->
href="javascript:void(0);" <a href="{{ url_for('invoice.delete_invoice_route', invoice_id=invoice.Invoice_Id) }}" onclick="return confirm('Are you sure?')">
onclick="deleteInvoice({{ invoice.Invoice_Id }}, this)" <img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" alt="Delete" class="icon">
>
<img
src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}"
alt="Delete"
class="icon"
/>
</a> </a>
</td> </td>
</tr> </tr>
@@ -394,31 +235,29 @@
{% else %} {% else %}
<p>No invoices found.</p> <p>No invoices found.</p>
{% endif %} {% endif %}
</div> </div>
<script> <script>
document.addEventListener("DOMContentLoaded", function () { document.addEventListener('DOMContentLoaded', function() {
// Get all the input fields // Get all the input fields
const basicAmount = document.getElementById("basic_amount"); const basicAmount = document.getElementById('basic_amount');
const debitAmount = document.getElementById("debit_amount"); const debitAmount = document.getElementById('debit_amount');
const afterDebitAmount = document.getElementById("after_debit_amount"); const afterDebitAmount = document.getElementById('after_debit_amount');
const amount = document.getElementById("amount"); const amount = document.getElementById('amount');
// Percentage fields // Percentage fields
const gstPercentage = document.getElementById("gst_percentage"); const gstPercentage = document.getElementById('gst_percentage');
const gstAmount = document.getElementById("gst_amount"); const gstAmount = document.getElementById('gst_amount');
const tdsPercentage = document.getElementById("tds_percentage"); const tdsPercentage = document.getElementById('tds_percentage');
const tdsAmount = document.getElementById("tds_amount"); const tdsAmount = document.getElementById('tds_amount');
const sdPercentage = document.getElementById("sd_percentage"); const sdPercentage = document.getElementById('sd_percentage');
const sdAmountInput = document.getElementById("sd_amount"); const sdAmountInput = document.getElementById('sd_amount');
const commissionPercentage = document.getElementById( const commissionPercentage = document.getElementById('commission_percentage');
"commission_percentage", const onCommission = document.getElementById('on_commission');
); const hydroPercentage = document.getElementById('hydro_percentage');
const onCommission = document.getElementById("on_commission"); const hydroTesting = document.getElementById('hydro_testing');
const hydroPercentage = document.getElementById("hydro_percentage"); const gstSdAmount = document.getElementById('gst_sd_amount');
const hydroTesting = document.getElementById("hydro_testing"); const finalAmount = document.getElementById('final_amount');
const gstSdAmount = document.getElementById("gst_sd_amount");
const finalAmount = document.getElementById("final_amount");
// Calculate after debit amount when basic or debit amount changes // Calculate after debit amount when basic or debit amount changes
function calculateAfterDebitAmount() { function calculateAfterDebitAmount() {
@@ -494,9 +333,7 @@
// Get hold amounts // Get hold amounts
let totalHold = 0; let totalHold = 0;
document document.querySelectorAll('input[name="hold_amount[]"]').forEach(input => {
.querySelectorAll('input[name="hold_amount[]"]')
.forEach((input) => {
totalHold += parseFloat(input.value) || 0; totalHold += parseFloat(input.value) || 0;
}); });
@@ -506,31 +343,28 @@
} }
// Add event listeners // Add event listeners
basicAmount.addEventListener("input", calculateAfterDebitAmount); basicAmount.addEventListener('input', calculateAfterDebitAmount);
debitAmount.addEventListener("input", calculateAfterDebitAmount); debitAmount.addEventListener('input', calculateAfterDebitAmount);
// Percentage fields // Percentage fields
gstPercentage.addEventListener("input", calculateGST); gstPercentage.addEventListener('input', calculateGST);
tdsPercentage.addEventListener("input", calculateOtherDeductions); tdsPercentage.addEventListener('input', calculateOtherDeductions);
sdPercentage.addEventListener("input", calculateOtherDeductions); sdPercentage.addEventListener('input', calculateOtherDeductions);
commissionPercentage.addEventListener( commissionPercentage.addEventListener('input', calculateOtherDeductions);
"input", hydroPercentage.addEventListener('input', calculateOtherDeductions);
calculateOtherDeductions,
);
hydroPercentage.addEventListener("input", calculateOtherDeductions);
// Listen for changes in hold amounts // Listen for changes in hold amounts
document.addEventListener("holdAmountChanged", calculateFinalAmount); document.addEventListener('holdAmountChanged', calculateFinalAmount);
}); });
// Optional JS for auto-hiding flash // Optional JS for auto-hiding flash
setTimeout(() => { setTimeout(() => {
document document.querySelectorAll('.alert').forEach(el => el.style.display = 'none');
.querySelectorAll(".alert") }, 5000);
.forEach((el) => (el.style.display = "none")); </script>
}, 5000);
</script>
</div> </div>
</body> </body>
{% endblock %} {% endblock %}

View File

@@ -97,15 +97,13 @@
<td><a href="/edit_payment/{{ payment[0] }}"><img <td><a href="/edit_payment/{{ payment[0] }}"><img
src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit" src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit"
class="icon"></a></td> class="icon"></a></td>
<td> <td>
<form action="{{ url_for('payment_bp.delete_payment', payment_id=payment[0]) }}" method="POST" <a href="{{ url_for('payment_bp.delete_payment', payment_id=payment[0]) }}"
style="display:inline;"> onclick="return confirm('Are you sure you want to delete this Payment?')">
<button type="submit" <img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" alt="Delete"
onclick="return confirm('Are you sure you want to delete this payment?')"> class="icon">
<img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" </a>
alt="Delete" class="icon">
</button>
</form>
</td> </td>
</tr> </tr>
<!-- <tr> <!-- <tr>

View File

@@ -1,24 +1,22 @@
{% extends 'base.html' %} {% block content %} {% extends 'base.html' %}
{% block content %}
<head> <head>
<meta charset="UTF-8" /> <meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0" /> <meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Village Management</title> <title>Village Management</title>
<link <link rel="stylesheet" type="text/css" href="{{ url_for('static', filename='css/style.css') }}">
rel="stylesheet"
type="text/css"
href="{{ url_for('static', filename='css/style.css') }}"
/>
<script src="{{ url_for('static', filename='js/village.js') }}"></script> <script src="{{ url_for('static', filename='js/village.js') }}"></script>
<script src="{{ url_for('static', filename='js/search_on_table.js') }}"></script> <script src="{{ url_for('static', filename='js/search_on_table.js') }}"></script>
</head> </head>
<body> <body>
<!-- Button Container to Center Buttons -->
<div class="button-container"> <!-- Button Container to Center Buttons -->
<div class="button-container">
<button id="addButton" class="action-button">Add</button> <button id="addButton" class="action-button">Add</button>
<button id="displayButton" class="action-button">Display</button> <button id="displayButton" class="action-button">Display</button>
</div> </div>
<div id="addForm" style="display: none"> <div id="addForm" style="display: none;">
<div class="container"> <div class="container">
<div class="form-block"> <div class="form-block">
<h2>Add a New Village</h2> <h2>Add a New Village</h2>
@@ -42,30 +40,19 @@
</select> </select>
<label for="Village_Name">Village Name:</label> <label for="Village_Name">Village Name:</label>
<input <input type="text" id="Village_Name" name="Village_Name" placeholder="Enter Village Name" required>
type="text"
id="Village_Name"
name="Village_Name"
placeholder="Enter Village Name"
required
/>
<span id="villageMessage"></span> <span id="villageMessage"></span>
<button type="submit" id="submitVillage" disabled>Add Village</button> <button type="submit" id="submitVillage" disabled>Add Village</button>
</form> </form>
</div> </div>
</div> </div>
</div> </div>
<div id="addTable" style="display: none"> <div id="addTable" style="display: none;">
<div class="search-container"> <div class="search-container">
<h2>Display Villages</h2> <h2>Display Villages</h2>
<input <input type="text" id="searchBar" placeholder="Searching..." onkeyup="searchTable()">
type="text"
id="searchBar"
placeholder="Searching..."
onkeyup="searchTable()"
/>
</div> </div>
<table id="sortableTable" border="1"> <table id="sortableTable" border="1">
@@ -76,15 +63,12 @@
<span class="sort-buttons"> <span class="sort-buttons">
<span class="sort-asc">⬆️</span> <span class="sort-asc">⬆️</span>
<span class="sort-desc">⬇️</span> <span class="sort-desc">⬇️</span>
</span> </span></th>
</th> <th class="sortable-header">Block Name
<th class="sortable-header">
Block Name
<span class="sort-buttons"> <span class="sort-buttons">
<span class="sort-asc">⬆️</span> <span class="sort-asc">⬆️</span>
<span class="sort-desc">⬇️</span> <span class="sort-desc">⬇️</span>
</span> </span></th>
</th>
<th>Update</th> <th>Update</th>
<th>Delete</th> <th>Delete</th>
</tr> </tr>
@@ -94,36 +78,22 @@
<td>{{ village[1] }}</td> <td>{{ village[1] }}</td>
<td>{{ village[2] }}</td> <td>{{ village[2] }}</td>
<td> <td>
<a <a href="{{ url_for('village.edit_village', village_id=village[0]) }}">
href="{{ url_for('village.edit_village', village_id=village[0]) }}" <img src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit"
> class="icon">
<img
src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}"
alt="Edit"
class="icon"
/>
</a> </a>
</td> </td>
<td> <td>
<a href="javascript:void(0);" <a href="#"
onclick="deleteVillage({{ village[0] }}, this)"> onclick="deleteVillage({{ village[0] }}); return false;">
<img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" <img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}"
class="icon"> alt="Delete" class="icon">
</a> </a>
</td> </td>
</tr> </tr>
{% endfor %} {% endfor %}
</table> </table>
</div> </div>
<!-- Flash Alerts -->
{% with messages = get_flashed_messages(with_categories=true) %}
{% if messages %}
<script>
{% for category, message in messages %}
alert("{{ message }}");
{% endfor %}
</script>
{% endif %}
{% endwith %}
</body> </body>
{% endblock %} {% endblock %}

View File

@@ -28,17 +28,25 @@
<button type="submit">Update Village</button> <button type="submit">Update Village</button>
</form> </form>
<!-- Flash Messages (hidden, used for JS popup) --> </div>
<div id="flash-messages-container" style="display:none;">
<!-- Flash Message (Hidden, used for JS popup) -->
{% with messages = get_flashed_messages(with_categories=true) %} {% with messages = get_flashed_messages(with_categories=true) %}
{% if messages %} {% if messages %}
{% for category, message in messages %} {% for category, message in messages %}
<div class="flash-message" data-category="{{ category }}">{{ message }}</div> <div id="flash-message" data-category="{{ category }}" style="display:none;">{{ message }}</div>
{% endfor %} {% endfor %}
{% endif %} {% endif %}
{% endwith %} {% endwith %}
</div> </div>
<script>
window.onload = function () {
const flash = document.getElementById('flash-message');
if (flash && flash.innerText.trim() !== "") {
alert(flash.innerText);
}
};
</script>
</body> </body>
{% endblock %} {% endblock %}