Compare commits
23 Commits
e7646eee76
...
swapnil-de
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d84ba520cf | ||
| 0aeaf775dd | |||
| 88e8771b51 | |||
| 6c74b5d3bf | |||
| 47ba78d72c | |||
| 1946a98d59 | |||
| 8f35e08429 | |||
| 82bedc3117 | |||
| cb68e454bc | |||
|
|
91b078a0c3 | ||
|
|
eb46929893 | ||
| 675301df7f | |||
|
|
14e799a1d4 | ||
| 8ab1b69033 | |||
| dbeec9962d | |||
| 8750f268db | |||
|
|
b78526ad9f | ||
| 7146391c18 | |||
| 94b5563d15 | |||
| 937018dc16 | |||
| eda238c235 | |||
| f184d6cecc | |||
| 64ca39944b |
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -1,5 +1,5 @@
|
|||||||
import config
|
import config
|
||||||
from flask import Blueprint, render_template, request, redirect, url_for, jsonify
|
from flask import Blueprint, render_template, request, redirect, url_for, jsonify, flash
|
||||||
from flask_login import login_required
|
from flask_login import login_required
|
||||||
|
|
||||||
from model.State import State
|
from model.State import State
|
||||||
@@ -74,7 +74,14 @@ def edit_block(block_id):
|
|||||||
|
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
block.EditBlock(request, block_id)
|
block.EditBlock(request, block_id)
|
||||||
return block.resultMessage
|
block.resultMessage
|
||||||
|
|
||||||
|
if block.resultMessage:
|
||||||
|
flash("Block updated successfully!", "success")
|
||||||
|
return redirect(url_for('block.add_block'))
|
||||||
|
else:
|
||||||
|
flash(block.resultMessage, "error")
|
||||||
|
|
||||||
|
|
||||||
connection = config.get_db_connection()
|
connection = config.get_db_connection()
|
||||||
cursor = connection.cursor()
|
cursor = connection.cursor()
|
||||||
|
|||||||
@@ -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'))
|
||||||
@@ -10,21 +10,21 @@ hold_bp = Blueprint("hold_types", __name__)
|
|||||||
@hold_bp.route('/add_hold_type', methods=['GET','POST'])
|
@hold_bp.route('/add_hold_type', methods=['GET','POST'])
|
||||||
@login_required
|
@login_required
|
||||||
def add_hold_type():
|
def add_hold_type():
|
||||||
|
|
||||||
hold = HoldTypes()
|
hold = HoldTypes()
|
||||||
|
|
||||||
if request.method == 'POST':
|
if request.method == 'POST':
|
||||||
hold.AddHoldType(request) # ✅
|
hold.AddHoldType(request)
|
||||||
return hold.resultMessage
|
# ✅ Always redirect to same page (NO JSON)
|
||||||
|
return redirect(url_for("hold_types.add_hold_type"))
|
||||||
|
|
||||||
hold_types = hold.GetAllHoldTypes() # ✅
|
# GET request → show data
|
||||||
|
hold_types = hold.GetAllHoldTypes()
|
||||||
|
|
||||||
return render_template(
|
return render_template(
|
||||||
"add_hold_type.html",
|
"add_hold_type.html",
|
||||||
Hold_Types_data=hold_types
|
Hold_Types_data=hold_types
|
||||||
)
|
)
|
||||||
|
|
||||||
|
|
||||||
# ---------------- CHECK HOLD TYPE (OPTIONAL LIKE BLOCK) ----------------
|
# ---------------- CHECK HOLD TYPE (OPTIONAL LIKE BLOCK) ----------------
|
||||||
@hold_bp.route('/check_hold_type', methods=['POST'])
|
@hold_bp.route('/check_hold_type', methods=['POST'])
|
||||||
@login_required
|
@login_required
|
||||||
|
|||||||
@@ -21,7 +21,6 @@ def activity_log():
|
|||||||
end_date,
|
end_date,
|
||||||
user_name
|
user_name
|
||||||
)
|
)
|
||||||
|
|
||||||
return render_template(
|
return render_template(
|
||||||
"activity_log.html",
|
"activity_log.html",
|
||||||
logs=filtered_logs,
|
logs=filtered_logs,
|
||||||
|
|||||||
@@ -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:
|
||||||
|
|||||||
@@ -83,17 +83,29 @@ 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)
|
||||||
|
|
||||||
flash(village.resultMessage, "success" if village.isSuccess else "error")
|
# ✅ Convert resultMessage to string if it's a Response or tuple
|
||||||
return redirect(url_for('village.add_village'))
|
raw_msg = village.resultMessage
|
||||||
|
|
||||||
|
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'])
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
from flask_login import LoginManager, UserMixin, login_user, logout_user, login_required, current_user
|
from flask import Blueprint, render_template, request, redirect, url_for, jsonify
|
||||||
|
|
||||||
from model.Utilities import RegEx, ResponseHandler, HtmlHelper, ItemCRUDType
|
from model.Utilities import RegEx, ResponseHandler, HtmlHelper, ItemCRUDType
|
||||||
from model.Log import LogData, LogHelper
|
from model.Log import LogData, LogHelper
|
||||||
@@ -147,7 +147,7 @@ class Block:
|
|||||||
|
|
||||||
self.isSuccess = block.isSuccess
|
self.isSuccess = block.isSuccess
|
||||||
self.resultMessage = block.resultMessage
|
self.resultMessage = block.resultMessage
|
||||||
return render_template('add_block.html')
|
return
|
||||||
|
|
||||||
# ----------------------------------------------------------
|
# ----------------------------------------------------------
|
||||||
# Delete Block
|
# Delete Block
|
||||||
|
|||||||
@@ -27,13 +27,27 @@ class FolderAndFile:
|
|||||||
os.makedirs(folder, exist_ok=True)
|
os.makedirs(folder, exist_ok=True)
|
||||||
return folder
|
return folder
|
||||||
|
|
||||||
# -----------------------------
|
@staticmethod
|
||||||
# FILE PATH METHODS
|
def get_logs_folder():
|
||||||
# -----------------------------
|
folder = os.path.join(current_app.root_path, "logs")
|
||||||
|
|
||||||
|
if not os.path.exists(folder):
|
||||||
|
os.makedirs(folder)
|
||||||
|
|
||||||
|
os.makedirs(folder, exist_ok=True)
|
||||||
|
return folder
|
||||||
|
|
||||||
|
# FILE PATH METHODS - download
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def get_download_path(filename):
|
def get_download_path(filename):
|
||||||
return os.path.join(FolderAndFile.get_download_folder(), filename)
|
return os.path.join(FolderAndFile.get_download_folder(), filename)
|
||||||
|
|
||||||
|
# FILE PATH METHODS - upload file
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def get_upload_path(filename):
|
def get_upload_path(filename):
|
||||||
return os.path.join(FolderAndFile.get_upload_folder(), filename)
|
return os.path.join(FolderAndFile.get_upload_folder(), filename)
|
||||||
|
|
||||||
|
# FILE PATH METHODS - activity log file
|
||||||
|
@staticmethod
|
||||||
|
def get_activity_log_path(filename):
|
||||||
|
return os.path.join(FolderAndFile.get_logs_folder(), filename)
|
||||||
@@ -1,6 +1,3 @@
|
|||||||
|
|
||||||
|
|
||||||
|
|
||||||
from flask_login import current_user
|
from flask_login import current_user
|
||||||
from model.Utilities import RegEx, ResponseHandler, HtmlHelper, ItemCRUDType
|
from model.Utilities import RegEx, ResponseHandler, HtmlHelper, ItemCRUDType
|
||||||
from model.Log import LogHelper
|
from model.Log import LogHelper
|
||||||
@@ -9,7 +6,6 @@ import config
|
|||||||
import re
|
import re
|
||||||
import mysql.connector
|
import mysql.connector
|
||||||
|
|
||||||
|
|
||||||
# ----------------------------------------------------------
|
# ----------------------------------------------------------
|
||||||
# Mapping Class
|
# Mapping Class
|
||||||
# ----------------------------------------------------------
|
# ----------------------------------------------------------
|
||||||
@@ -26,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
|
||||||
# ----------------------------------------------------------
|
# ----------------------------------------------------------
|
||||||
@@ -38,7 +35,6 @@ class ItemCRUD:
|
|||||||
def __init__(self, itemType):
|
def __init__(self, itemType):
|
||||||
self.isSuccess = False
|
self.isSuccess = False
|
||||||
self.resultMessage = ""
|
self.resultMessage = ""
|
||||||
self.response = {} # ✅ ADDED
|
|
||||||
self.itemCRUDType = itemType
|
self.itemCRUDType = itemType
|
||||||
self.itemCRUDMapping = itemCRUDMapping(itemType)
|
self.itemCRUDMapping = itemCRUDMapping(itemType)
|
||||||
|
|
||||||
@@ -60,14 +56,16 @@ class ItemCRUD:
|
|||||||
connection.commit()
|
connection.commit()
|
||||||
|
|
||||||
self.isSuccess = True
|
self.isSuccess = True
|
||||||
self.response = ResponseHandler.delete_success(self.itemCRUDMapping.name)
|
self.resultMessage = HtmlHelper.json_response(
|
||||||
self.resultMessage = self.response["message"]
|
ResponseHandler.delete_success(self.itemCRUDMapping.name), 200
|
||||||
|
)
|
||||||
|
|
||||||
except mysql.connector.Error as e:
|
except mysql.connector.Error as e:
|
||||||
print(f"Error deleting {self.itemCRUDMapping.name}: {e}")
|
print(f"Error deleting {self.itemCRUDMapping.name}: {e}")
|
||||||
self.isSuccess = False
|
self.isSuccess = False
|
||||||
self.response = ResponseHandler.delete_failure(self.itemCRUDMapping.name)
|
self.resultMessage = HtmlHelper.json_response(
|
||||||
self.resultMessage = self.response["message"]
|
ResponseHandler.delete_failure(self.itemCRUDMapping.name), 500
|
||||||
|
)
|
||||||
|
|
||||||
finally:
|
finally:
|
||||||
cursor.close()
|
cursor.close()
|
||||||
@@ -81,8 +79,9 @@ class ItemCRUD:
|
|||||||
connection = config.get_db_connection()
|
connection = config.get_db_connection()
|
||||||
if not connection:
|
if not connection:
|
||||||
self.isSuccess = False
|
self.isSuccess = False
|
||||||
self.response = ResponseHandler.add_failure(self.itemCRUDMapping.name)
|
self.resultMessage = HtmlHelper.json_response(
|
||||||
self.resultMessage = self.response["message"]
|
ResponseHandler.db_connection_failure(), 500
|
||||||
|
)
|
||||||
return
|
return
|
||||||
|
|
||||||
cursor = connection.cursor()
|
cursor = connection.cursor()
|
||||||
@@ -93,18 +92,57 @@ class ItemCRUD:
|
|||||||
)
|
)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
# SUBCONTRACTOR
|
# ======================================================
|
||||||
if data:
|
# GSTRelease MULTI-FIELD
|
||||||
cursor.callproc(storedprocfetch, (data['Contractor_Name'],))
|
# ======================================================
|
||||||
|
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:
|
||||||
|
|
||||||
|
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()
|
||||||
|
|
||||||
if existing_item:
|
if existing_item:
|
||||||
self.isSuccess = False
|
self.isSuccess = False
|
||||||
self.response = ResponseHandler.already_exists(self.itemCRUDMapping.name)
|
self.resultMessage = HtmlHelper.json_response(
|
||||||
self.resultMessage = self.response["message"]
|
ResponseHandler.already_exists(self.itemCRUDMapping.name), 409
|
||||||
|
)
|
||||||
return
|
return
|
||||||
|
|
||||||
cursor.callproc(storedprocadd, (
|
cursor.callproc(storedprocadd, (
|
||||||
@@ -118,19 +156,22 @@ 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.response = ResponseHandler.add_success(self.itemCRUDMapping.name)
|
self.resultMessage = HtmlHelper.json_response(
|
||||||
self.resultMessage = self.response["message"]
|
ResponseHandler.add_success(self.itemCRUDMapping.name), 200
|
||||||
|
)
|
||||||
return
|
return
|
||||||
|
|
||||||
# NORMAL
|
|
||||||
if not re.match(RegEx.patternAlphabetOnly, childname):
|
# ======================================================
|
||||||
|
# NORMAL SINGLE-FIELD (Village / Block / State)
|
||||||
|
# ======================================================
|
||||||
|
if not re.match(RegEx.allPattern, childname):
|
||||||
self.isSuccess = False
|
self.isSuccess = False
|
||||||
self.response = ResponseHandler.invalid_name(self.itemCRUDMapping.name)
|
self.resultMessage = HtmlHelper.json_response(
|
||||||
self.resultMessage = self.response["message"]
|
ResponseHandler.invalid_name(self.itemCRUDMapping.name), 400
|
||||||
|
)
|
||||||
return
|
return
|
||||||
|
|
||||||
if parentid is None:
|
if parentid is None:
|
||||||
@@ -144,8 +185,9 @@ class ItemCRUD:
|
|||||||
|
|
||||||
if existing_item:
|
if existing_item:
|
||||||
self.isSuccess = False
|
self.isSuccess = False
|
||||||
self.response = ResponseHandler.already_exists(self.itemCRUDMapping.name)
|
self.resultMessage = HtmlHelper.json_response(
|
||||||
self.resultMessage = self.response["message"]
|
ResponseHandler.already_exists(self.itemCRUDMapping.name), 409
|
||||||
|
)
|
||||||
return
|
return
|
||||||
|
|
||||||
if parentid is None:
|
if parentid is None:
|
||||||
@@ -154,16 +196,17 @@ class ItemCRUD:
|
|||||||
cursor.callproc(storedprocadd, (childname, parentid))
|
cursor.callproc(storedprocadd, (childname, parentid))
|
||||||
|
|
||||||
connection.commit()
|
connection.commit()
|
||||||
|
|
||||||
self.isSuccess = True
|
self.isSuccess = True
|
||||||
self.response = ResponseHandler.add_success(self.itemCRUDMapping.name)
|
self.resultMessage = HtmlHelper.json_response(
|
||||||
self.resultMessage = self.response["message"]
|
ResponseHandler.add_success(self.itemCRUDMapping.name), 200
|
||||||
|
)
|
||||||
|
|
||||||
except mysql.connector.Error as e:
|
except mysql.connector.Error as e:
|
||||||
print(f"Database Error: {e}")
|
print(f"Database Error: {e}")
|
||||||
self.isSuccess = False
|
self.isSuccess = False
|
||||||
self.response = ResponseHandler.add_failure(self.itemCRUDMapping.name)
|
self.resultMessage = HtmlHelper.json_response(
|
||||||
self.resultMessage = self.response["message"]
|
ResponseHandler.add_failure(self.itemCRUDMapping.name), 500
|
||||||
|
)
|
||||||
|
|
||||||
finally:
|
finally:
|
||||||
cursor.close()
|
cursor.close()
|
||||||
@@ -183,7 +226,33 @@ class ItemCRUD:
|
|||||||
)
|
)
|
||||||
|
|
||||||
try:
|
try:
|
||||||
if data:
|
# ======================================================
|
||||||
|
# GSTRelease MULTI-FIELD
|
||||||
|
# ======================================================
|
||||||
|
if self.itemCRUDType.name == "GSTRelease" and data:
|
||||||
|
|
||||||
|
cursor.callproc(storedprocupdate, (
|
||||||
|
data['p_pmc_no'], # PMC_No
|
||||||
|
data['p_invoice_no'], # Invoice_No
|
||||||
|
data['p_basic_amount'], # Basic_Amount
|
||||||
|
data['p_final_amount'], # Final_Amount
|
||||||
|
data['p_total_amount'], # Total_Amount
|
||||||
|
data['p_utr'], # UTR
|
||||||
|
data['p_gst_release_id']# GST_Release_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'],
|
||||||
@@ -196,18 +265,19 @@ 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.response = ResponseHandler.update_success(self.itemCRUDMapping.name)
|
self.resultMessage = HtmlHelper.json_response(
|
||||||
self.resultMessage = self.response["message"]
|
ResponseHandler.update_success(self.itemCRUDMapping.name), 200
|
||||||
|
)
|
||||||
return
|
return
|
||||||
|
|
||||||
if not re.match(RegEx.patternAlphabetOnly, childname):
|
# ======================================================
|
||||||
|
# NORMAL SINGLE-FIELD
|
||||||
|
# ======================================================
|
||||||
|
if not re.match(RegEx.allPattern, childname):
|
||||||
self.isSuccess = False
|
self.isSuccess = False
|
||||||
self.response = ResponseHandler.update_failure(self.itemCRUDMapping.name)
|
self.resultMessage = ResponseHandler.update_failure(self.itemCRUDMapping.name)['message']
|
||||||
self.resultMessage = self.response["message"]
|
|
||||||
return
|
return
|
||||||
|
|
||||||
if parentid is None:
|
if parentid is None:
|
||||||
@@ -216,16 +286,15 @@ 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.response = ResponseHandler.update_success(self.itemCRUDMapping.name)
|
self.resultMessage = ResponseHandler.update_success(self.itemCRUDMapping.name)['message']
|
||||||
self.resultMessage = self.response["message"]
|
|
||||||
|
|
||||||
except mysql.connector.Error as e:
|
except mysql.connector.Error as e:
|
||||||
print(f"Error updating {self.itemCRUDMapping.name}: {e}")
|
print(f"Error updating {self.itemCRUDMapping.name}: {e}")
|
||||||
self.isSuccess = False
|
self.isSuccess = False
|
||||||
self.response = ResponseHandler.update_failure(self.itemCRUDMapping.name)
|
self.resultMessage = HtmlHelper.json_response(
|
||||||
self.resultMessage = self.response["message"]
|
ResponseHandler.update_failure(self.itemCRUDMapping.name), 500
|
||||||
|
)
|
||||||
|
|
||||||
finally:
|
finally:
|
||||||
cursor.close()
|
cursor.close()
|
||||||
@@ -238,27 +307,22 @@ 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
|
||||||
self.response = ResponseHandler.fetch_failure(self.itemCRUDMapping.name)
|
self.resultMessage = HtmlHelper.json_response(
|
||||||
self.resultMessage = self.response["message"]
|
ResponseHandler.fetch_failure(self.itemCRUDMapping.name), 500
|
||||||
|
)
|
||||||
return []
|
return []
|
||||||
|
|
||||||
finally:
|
finally:
|
||||||
cursor.close()
|
cursor.close()
|
||||||
connection.close()
|
connection.close()
|
||||||
@@ -276,13 +340,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()
|
||||||
@@ -290,7 +351,7 @@ class ItemCRUD:
|
|||||||
return data
|
return data
|
||||||
|
|
||||||
# ----------------------------------------------------------
|
# ----------------------------------------------------------
|
||||||
# CHECK ITEM (KEEP AS IS - API USE)
|
# CHECK ITEM
|
||||||
# ----------------------------------------------------------
|
# ----------------------------------------------------------
|
||||||
def CheckItem(self, request, parentid, childname, storedprocfetch):
|
def CheckItem(self, request, parentid, childname, storedprocfetch):
|
||||||
|
|
||||||
@@ -302,7 +363,7 @@ class ItemCRUD:
|
|||||||
f"User {current_user.id} checked '{childname}'"
|
f"User {current_user.id} checked '{childname}'"
|
||||||
)
|
)
|
||||||
|
|
||||||
if not re.match(RegEx.patternAlphabetOnly, childname):
|
if not re.match(RegEx.allPattern, childname):
|
||||||
return HtmlHelper.json_response(
|
return HtmlHelper.json_response(
|
||||||
ResponseHandler.invalid_name(self.itemCRUDMapping.name), 400
|
ResponseHandler.invalid_name(self.itemCRUDMapping.name), 400
|
||||||
)
|
)
|
||||||
@@ -331,7 +392,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()
|
||||||
116
model/Log.py
116
model/Log.py
@@ -1,29 +1,41 @@
|
|||||||
import os
|
import os
|
||||||
from datetime import datetime
|
|
||||||
from flask import current_app
|
from flask import current_app
|
||||||
from flask_login import current_user
|
from flask_login import LoginManager, UserMixin, login_user, logout_user, login_required, current_user
|
||||||
|
|
||||||
|
from datetime import datetime
|
||||||
|
|
||||||
|
from model.FolderAndFile import FolderAndFile
|
||||||
|
|
||||||
class LogHelper:
|
class LogHelper:
|
||||||
@staticmethod
|
@staticmethod
|
||||||
def log_action(action, details=""):
|
def log_action(action, details=""):
|
||||||
"""Add a log entry."""
|
"""Log user actions with timestamp, user, action, and details."""
|
||||||
log_data = LogData()
|
logData = LogData()
|
||||||
log_data.add_log(action, details)
|
logData.WriteLog(action, details="")
|
||||||
|
|
||||||
|
|
||||||
class LogData:
|
class LogData:
|
||||||
|
filepath = ""
|
||||||
|
timestamp = None
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.filepath = os.path.join(current_app.root_path, 'activity.log')
|
self.filepath = FolderAndFile.get_activity_log_path('activity.log')
|
||||||
self.timestamp = datetime.now().strftime("%Y-%m-%d %H:%M:%S")
|
self.timestamp = datetime.now().strftime("%Y-%m-%d %H:%M:%S")
|
||||||
self.user = getattr(current_user, "cn", None) \
|
self.user = LogData.get_current_user()
|
||||||
or getattr(current_user, "username", None) \
|
|
||||||
or getattr(current_user, "sAMAccountName", "Unknown")
|
|
||||||
|
|
||||||
|
|
||||||
def add_log(self, action, details=""):
|
@staticmethod
|
||||||
"""Create/Add a log entry."""
|
def get_current_user():
|
||||||
|
if hasattr(current_user, "cn") and current_user.cn:
|
||||||
|
return current_user.cn
|
||||||
|
elif hasattr(current_user, "username") and current_user.username:
|
||||||
|
return current_user.username
|
||||||
|
elif hasattr(current_user, "sAMAccountName") and current_user.sAMAccountName:
|
||||||
|
return current_user.sAMAccountName
|
||||||
|
return "Unknown"
|
||||||
|
|
||||||
|
def WriteLog(self, action, details=""):
|
||||||
|
"""Log user actions with timestamp, user, action, and details."""
|
||||||
|
|
||||||
with open(self.filepath, "a", encoding="utf-8") as f:
|
with open(self.filepath, "a", encoding="utf-8") as f:
|
||||||
f.write(
|
f.write(
|
||||||
f"Timestamp: {self.timestamp} | "
|
f"Timestamp: {self.timestamp} | "
|
||||||
@@ -32,73 +44,41 @@ class LogData:
|
|||||||
f"Details: {details}\n"
|
f"Details: {details}\n"
|
||||||
)
|
)
|
||||||
|
|
||||||
def get_all_logs(self):
|
def GetActivitiesLog(self):
|
||||||
"""Read all logs."""
|
|
||||||
logs = []
|
logs = []
|
||||||
|
|
||||||
if os.path.exists(self.filepath):
|
if os.path.exists(self.filepath):
|
||||||
with open(self.filepath, 'r', encoding="utf-8") as f:
|
with open(self.filepath, 'r') as f:
|
||||||
for line in f:
|
for line in f:
|
||||||
parts = line.strip().split(" | ")
|
parts = line.strip().split(" | ")
|
||||||
if len(parts) == 4:
|
if len(parts) == 4:
|
||||||
logs.append({
|
logs.append({
|
||||||
"timestamp": parts[0].split(":", 1)[1].strip(),
|
"timestamp": parts[0].replace("Timestamp:", "").strip(),
|
||||||
"user": parts[1].split(":", 1)[1].strip(),
|
"user": parts[1].replace("User:", "").strip(),
|
||||||
"action": parts[2].split(":", 1)[1].strip(),
|
"action": parts[2].replace("Action:", "").strip(),
|
||||||
"details": parts[3].split(":", 1)[1].strip()
|
"details": parts[3].replace("Details:", "").strip()
|
||||||
})
|
})
|
||||||
return logs
|
return logs
|
||||||
|
|
||||||
def get_filtered_logs(self, start_date=None, end_date=None, user_name=None):
|
def GetFilteredActivitiesLog(self, startDate, endDate, userName):
|
||||||
"""Filter logs by date and/or user."""
|
filtered_logs = self.GetActivitiesLog()
|
||||||
logs = self.get_all_logs()
|
|
||||||
|
|
||||||
# Filter by date
|
# Date filter
|
||||||
if start_date or end_date:
|
if startDate or endDate:
|
||||||
start_dt = datetime.strptime(start_date, "%Y-%m-%d") if start_date else datetime.min
|
try:
|
||||||
end_dt = datetime.strptime(end_date, "%Y-%m-%d") if end_date else datetime.max
|
start_dt = datetime.strptime(startDate, "%Y-%m-%d") if startDate else datetime.min
|
||||||
logs = [
|
end_dt = datetime.strptime(endDate, "%Y-%m-%d") if endDate else datetime.max
|
||||||
log for log in logs
|
|
||||||
if start_dt <= datetime.strptime(log["timestamp"], "%Y-%m-%d %H:%M:%S") <= end_dt
|
|
||||||
]
|
|
||||||
|
|
||||||
# Filter by username
|
filtered_logs = [
|
||||||
if user_name:
|
log for log in filtered_logs
|
||||||
logs = [log for log in logs if user_name.lower() in log.get("user", "").lower()]
|
if start_dt <= datetime.strptime(log["timestamp"], "%Y-%m-%d %H:%M:%S") <= end_dt
|
||||||
|
]
|
||||||
|
|
||||||
return logs
|
except Exception as e:
|
||||||
|
print("Date filter error:", e)
|
||||||
def update_log(self, index, action=None, details=None):
|
|
||||||
"""Update a specific log entry by index (0-based)."""
|
|
||||||
logs = self.get_all_logs()
|
|
||||||
if 0 <= index < len(logs):
|
|
||||||
if action:
|
|
||||||
logs[index]["action"] = action
|
|
||||||
if details:
|
|
||||||
logs[index]["details"] = details
|
|
||||||
self._rewrite_logs_file(logs)
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
def delete_log(self, index):
|
|
||||||
"""Delete a specific log entry by index (0-based)."""
|
|
||||||
logs = self.get_all_logs()
|
|
||||||
if 0 <= index < len(logs):
|
|
||||||
logs.pop(index)
|
|
||||||
self._rewrite_logs_file(logs)
|
|
||||||
return True
|
|
||||||
return False
|
|
||||||
|
|
||||||
# ------------------- INTERNAL HELPER -------------------
|
|
||||||
|
|
||||||
def _rewrite_logs_file(self, logs):
|
|
||||||
"""Overwrite the log file with current logs."""
|
|
||||||
with open(self.filepath, "w", encoding="utf-8") as f:
|
|
||||||
for log in logs:
|
|
||||||
f.write(
|
|
||||||
f"Timestamp: {log['timestamp']} | "
|
|
||||||
f"User: {log['user']} | "
|
|
||||||
f"Action: {log['action']} | "
|
|
||||||
f"Details: {log['details']}\n"
|
|
||||||
)
|
|
||||||
|
|
||||||
|
# Username filter
|
||||||
|
if userName:
|
||||||
|
filtered_logs = [log for log in filtered_logs if userName.lower() in log["user"].lower()]
|
||||||
|
|
||||||
|
return filtered_logs
|
||||||
@@ -8,10 +8,11 @@ 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 ]+$"
|
||||||
|
allPattern = "^(?!\s*$).+"
|
||||||
|
|
||||||
|
|
||||||
class ResponseHandler:
|
class ResponseHandler:
|
||||||
|
|||||||
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@@ -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):
|
||||||
@@ -115,6 +14,12 @@ class GSTRelease:
|
|||||||
try:
|
try:
|
||||||
gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease)
|
gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease)
|
||||||
|
|
||||||
|
# Print the full form data
|
||||||
|
print("===== DEBUG: FORM DATA =====")
|
||||||
|
for key, value in request.form.items():
|
||||||
|
print(f"{key} : {value}")
|
||||||
|
print("=============================")
|
||||||
|
|
||||||
data = {
|
data = {
|
||||||
"PMC_No": request.form.get("PMC_No", "").strip(),
|
"PMC_No": request.form.get("PMC_No", "").strip(),
|
||||||
"Invoice_No": request.form.get("Invoice_No", "").strip(),
|
"Invoice_No": request.form.get("Invoice_No", "").strip(),
|
||||||
@@ -125,6 +30,11 @@ class GSTRelease:
|
|||||||
"Contractor_ID": int(request.form.get("Contractor_ID", 0) or 0)
|
"Contractor_ID": int(request.form.get("Contractor_ID", 0) or 0)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
print("===== DEBUG: PARSED DATA =====")
|
||||||
|
print(data)
|
||||||
|
print("==============================")
|
||||||
|
|
||||||
|
# Add GST Release
|
||||||
gst.AddItem(
|
gst.AddItem(
|
||||||
request=request,
|
request=request,
|
||||||
data=data,
|
data=data,
|
||||||
@@ -132,6 +42,8 @@ class GSTRelease:
|
|||||||
storedprocadd="AddGSTReleaseFromExcel"
|
storedprocadd="AddGSTReleaseFromExcel"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
print(f"AddItem result: isSuccess={gst.isSuccess}, message={gst.resultMessage}")
|
||||||
|
|
||||||
self.isSuccess = gst.isSuccess
|
self.isSuccess = gst.isSuccess
|
||||||
self.resultMessage = str(gst.resultMessage)
|
self.resultMessage = str(gst.resultMessage)
|
||||||
|
|
||||||
@@ -142,20 +54,26 @@ class GSTRelease:
|
|||||||
|
|
||||||
return jsonify({"success": self.isSuccess, "message": self.resultMessage})
|
return jsonify({"success": self.isSuccess, "message": self.resultMessage})
|
||||||
|
|
||||||
# ------------------- Edit GST Release -------------------
|
|
||||||
def EditGSTRelease(self, request, gst_release_id):
|
def EditGSTRelease(self, request, gst_release_id):
|
||||||
try:
|
try:
|
||||||
gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease)
|
gst = ItemCRUD(itemType=ItemCRUDType.GSTRelease)
|
||||||
|
|
||||||
|
# Map form inputs to stored procedure parameters
|
||||||
data = {
|
data = {
|
||||||
"PMC_No": request.form.get("PMC_No", "").strip(),
|
"p_pmc_no": request.form.get("PMC_No", "").strip(),
|
||||||
"Invoice_No": request.form.get("Invoice_No", "").strip(),
|
"p_invoice_no": request.form.get("invoice_no", "").strip(),
|
||||||
"Basic_Amount": float(request.form.get("Basic_Amount", 0) or 0),
|
"p_basic_amount": float(request.form.get("Basic_Amount", 0) or 0),
|
||||||
"Final_Amount": float(request.form.get("Final_Amount", 0) or 0),
|
"p_final_amount": float(request.form.get("Final_Amount", 0) or 0),
|
||||||
"Total_Amount": float(request.form.get("Total_Amount", 0) or 0),
|
"p_total_amount": float(request.form.get("Total_Amount", 0) or 0),
|
||||||
"UTR": request.form.get("UTR", "").strip()
|
"p_utr": request.form.get("UTR", "").strip(),
|
||||||
|
"p_gst_release_id": gst_release_id
|
||||||
}
|
}
|
||||||
|
|
||||||
|
print("===== DEBUG: UPDATE DATA =====")
|
||||||
|
print(data)
|
||||||
|
print("==============================")
|
||||||
|
|
||||||
|
# Call your stored procedure
|
||||||
gst.EditItem(
|
gst.EditItem(
|
||||||
request=request,
|
request=request,
|
||||||
childid=gst_release_id,
|
childid=gst_release_id,
|
||||||
@@ -171,8 +89,6 @@ class GSTRelease:
|
|||||||
self.isSuccess = False
|
self.isSuccess = False
|
||||||
self.resultMessage = str(e)
|
self.resultMessage = str(e)
|
||||||
|
|
||||||
return jsonify({"success": self.isSuccess, "message": self.resultMessage})
|
|
||||||
|
|
||||||
# ------------------- Delete GST Release -------------------
|
# ------------------- Delete GST Release -------------------
|
||||||
def DeleteGSTRelease(self, gst_release_id):
|
def DeleteGSTRelease(self, gst_release_id):
|
||||||
try:
|
try:
|
||||||
@@ -198,7 +114,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 +139,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:
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ $(document).ready(function () {
|
|||||||
let holdType = $(this).val().replace(/^\s+/, "");
|
let holdType = $(this).val().replace(/^\s+/, "");
|
||||||
$(this).val(holdType);
|
$(this).val(holdType);
|
||||||
|
|
||||||
let reg = /^[A-Za-z]/;
|
let reg = /^.+$/; // all pattern allow
|
||||||
|
|
||||||
if (!reg.test(holdType)) {
|
if (!reg.test(holdType)) {
|
||||||
$("#holdTypeMessage").text("Hold Type must start with a letter.").css("color", "red");
|
$("#holdTypeMessage").text("Hold Type must start with a letter.").css("color", "red");
|
||||||
|
|||||||
@@ -1,62 +1,271 @@
|
|||||||
// Subcontractor autocomplete functionality
|
// $(document).ready(function () {
|
||||||
$(document).ready(function () {
|
// // ===============================
|
||||||
$("#subcontractor").keyup(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 () {
|
||||||
let query = $(this).val();
|
let query = $(this).val();
|
||||||
if (query !== "") {
|
if (query !== "") {
|
||||||
$.ajax({
|
$.ajax({
|
||||||
url: "/search_subcontractor",
|
url: "/search_subcontractor",
|
||||||
method: "POST",
|
method: "POST",
|
||||||
data: { query: query },
|
data: { query: query },
|
||||||
success: function (data) {
|
success: function (data) {
|
||||||
$("#subcontractor_list").fadeIn().html(data);
|
$("#subcontractor_list").fadeIn().html(data);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
$("#subcontractor_list").fadeOut();
|
$("#subcontractor_list").fadeOut();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
$(document).on("click", "li", function () {
|
$(document).on("click", "li", function () {
|
||||||
$("#subcontractor").val($(this).text());
|
$("#subcontractor").val($(this).text());
|
||||||
$("#subcontractor_id").val($(this).attr("data-id"));
|
$("#subcontractor_id").val($(this).attr("data-id"));
|
||||||
$("#subcontractor_list").fadeOut();
|
$("#subcontractor_list").fadeOut();
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|
||||||
// Success Alert: show alert and reload after 3 seconds
|
// Focus
|
||||||
function showSuccessAlert() {
|
if (document.getElementById('subcontractor')) {
|
||||||
const alertBox = document.getElementById("invoiceSuccessAlert");
|
document.getElementById('subcontractor').focus();
|
||||||
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
|
// ===============================
|
||||||
$("#invoiceForm").on("submit", function (e) {
|
// ADD INVOICE
|
||||||
e.preventDefault();
|
// ===============================
|
||||||
let formData = $(this).serialize();
|
if ($('#invoiceForm').length && window.location.href.includes('add_invoice')) {
|
||||||
$.ajax({
|
$("#invoiceForm").on("submit", function (e) {
|
||||||
url: '/add_invoice',
|
e.preventDefault();
|
||||||
method: 'POST',
|
let formData = $(this).serialize();
|
||||||
data: formData,
|
|
||||||
success: function (response) {
|
|
||||||
if(response.status === "success") {
|
|
||||||
showSuccessAlert();
|
|
||||||
} else {
|
|
||||||
alert(response.message);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
error: function (xhr, status, error) {
|
|
||||||
alert("Submission failed: " + error);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
|
$.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) {
|
||||||
|
let msg = xhr.responseJSON?.message || "Submission failed. Please try again.";
|
||||||
|
alert(msg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// ===============================
|
||||||
|
// UPDATE INVOICE
|
||||||
|
// ===============================
|
||||||
|
function updateInvoice(invoiceId, formElement) {
|
||||||
|
const formData = $(formElement).serialize();
|
||||||
|
|
||||||
window.onload = function () {
|
$.ajax({
|
||||||
document.getElementById('subcontractor').focus();
|
url: '/update_invoice/' + invoiceId,
|
||||||
};
|
method: 'POST',
|
||||||
|
data: formData,
|
||||||
|
dataType: 'json',
|
||||||
|
success: function(response) {
|
||||||
|
if(response.status === "success") {
|
||||||
|
alert(response.message || "Invoice updated successfully!");
|
||||||
|
|
||||||
|
// Redirect to Add Invoice page's table part
|
||||||
|
window.location.href = "/add_invoice#addTable";
|
||||||
|
} else {
|
||||||
|
alert(response.message || "Update failed. Please try again.");
|
||||||
|
}
|
||||||
|
},
|
||||||
|
error: function(xhr) {
|
||||||
|
let msg = xhr.responseJSON?.message || "Error updating invoice.";
|
||||||
|
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
|
||||||
|
});
|
||||||
@@ -1,26 +1,23 @@
|
|||||||
|
|
||||||
// Search on table using search inpute options
|
// Search on table using search inpute options
|
||||||
function searchTable() {
|
function searchTable() {
|
||||||
let input = document.getElementById("searchBar").value.toLowerCase();
|
let input = document.getElementById("searchBar").value.toLowerCase();
|
||||||
let rows = document.querySelectorAll("table tbody tr");
|
let tables = document.querySelectorAll("table");
|
||||||
|
|
||||||
rows.forEach(row => {
|
tables.forEach(table => {
|
||||||
let blockName = row.cells[1].textContent.toLowerCase();
|
let rows = table.querySelectorAll("tr");
|
||||||
let districtName = row.cells[2].textContent.toLowerCase();
|
|
||||||
let villageName = row.cells[3].textContent.toLowerCase();
|
|
||||||
|
|
||||||
if (blockName.includes(input) || districtName.includes(input)|| villageName.includes(input)) {
|
rows.forEach((row, index) => {
|
||||||
row.style.display = "";
|
if (index === 0) return; // header skip
|
||||||
} else {
|
|
||||||
row.style.display = "none";
|
let text = row.textContent.toLowerCase();
|
||||||
}
|
|
||||||
|
row.style.display = text.includes(input) ? "" : "none";
|
||||||
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// Common Sorting Script for Tables
|
// Common Sorting Script for Tables
|
||||||
function sortTable(n, dir) {
|
function sortTable(n, dir) {
|
||||||
var table, rows, switching, i, x, y, shouldSwitch;
|
var table, rows, switching, i, x, y, shouldSwitch;
|
||||||
@@ -57,14 +54,14 @@ function sortTable(n, dir) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Attach sorting functionality to all sortable tables
|
// Attach sorting functionality to all sortable tables
|
||||||
document.addEventListener("DOMContentLoaded", function() {
|
document.addEventListener("DOMContentLoaded", function () {
|
||||||
// Find all elements with the class "sortable-header"
|
// Find all elements with the class "sortable-header"
|
||||||
var sortableHeaders = document.querySelectorAll(".sortable-header");
|
var sortableHeaders = document.querySelectorAll(".sortable-header");
|
||||||
|
|
||||||
sortableHeaders.forEach(function(header) {
|
sortableHeaders.forEach(function (header) {
|
||||||
// Attach click event for ascending sort
|
// Attach click event for ascending sort
|
||||||
if (header.querySelector(".sort-asc")) {
|
if (header.querySelector(".sort-asc")) {
|
||||||
header.querySelector(".sort-asc").addEventListener("click", function() {
|
header.querySelector(".sort-asc").addEventListener("click", function () {
|
||||||
var columnIndex = Array.from(header.parentNode.children).indexOf(header);
|
var columnIndex = Array.from(header.parentNode.children).indexOf(header);
|
||||||
sortTable(columnIndex, "asc");
|
sortTable(columnIndex, "asc");
|
||||||
});
|
});
|
||||||
@@ -72,7 +69,7 @@ document.addEventListener("DOMContentLoaded", function() {
|
|||||||
|
|
||||||
// Attach click event for descending sort
|
// Attach click event for descending sort
|
||||||
if (header.querySelector(".sort-desc")) {
|
if (header.querySelector(".sort-desc")) {
|
||||||
header.querySelector(".sort-desc").addEventListener("click", function() {
|
header.querySelector(".sort-desc").addEventListener("click", function () {
|
||||||
var columnIndex = Array.from(header.parentNode.children).indexOf(header);
|
var columnIndex = Array.from(header.parentNode.children).indexOf(header);
|
||||||
sortTable(columnIndex, "desc");
|
sortTable(columnIndex, "desc");
|
||||||
});
|
});
|
||||||
@@ -106,3 +103,30 @@ document.addEventListener("DOMContentLoaded", function () {
|
|||||||
addButton.classList.remove("active-button");
|
addButton.classList.remove("active-button");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", function () {
|
||||||
|
|
||||||
|
let tables = document.querySelectorAll("table");
|
||||||
|
|
||||||
|
tables.forEach(table => {
|
||||||
|
let header = table.querySelector("tr:first-child");
|
||||||
|
|
||||||
|
if (header) {
|
||||||
|
header.style.position = "sticky";
|
||||||
|
header.style.top = "0";
|
||||||
|
header.style.background = "#fff";
|
||||||
|
header.style.zIndex = "2";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!table.parentElement.classList.contains("table-wrapper")) {
|
||||||
|
let wrapper = document.createElement("div");
|
||||||
|
wrapper.classList.add("table-wrapper");
|
||||||
|
wrapper.style.maxHeight = "65vh"
|
||||||
|
wrapper.style.overflowY = "auto";
|
||||||
|
|
||||||
|
table.parentNode.insertBefore(wrapper, table);
|
||||||
|
wrapper.appendChild(table);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
||||||
@@ -1,264 +1,250 @@
|
|||||||
|
|
||||||
|
|
||||||
window.onload = function () {
|
window.onload = function () {
|
||||||
document.getElementById('Village_Name').focus();
|
if (document.getElementById('Village_Name')) {
|
||||||
|
document.getElementById('Village_Name').focus();
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
$(document).ready(function () {
|
$(document).ready(function () {
|
||||||
|
|
||||||
// 🔥 RESTORE VIEW MODE AFTER RELOAD
|
// ✅ RUN ONLY IF THIS PAGE HAS FORM/TABLE
|
||||||
var viewMode = localStorage.getItem("viewMode");
|
if ($('#addForm').length && $('#addTable').length) {
|
||||||
|
|
||||||
if (viewMode === "table") {
|
// ✅ DEFAULT VIEW → SHOW FORM
|
||||||
$('#addForm').hide();
|
|
||||||
$('#addTable').show();
|
|
||||||
} else {
|
|
||||||
$('#addForm').show();
|
$('#addForm').show();
|
||||||
$('#addTable').hide();
|
$('#addTable').hide();
|
||||||
|
|
||||||
|
|
||||||
|
// 🔥 BUTTON TOGGLE
|
||||||
|
$('#addButton').click(function () {
|
||||||
|
$('#addForm').show();
|
||||||
|
$('#addTable').hide();
|
||||||
|
});
|
||||||
|
|
||||||
|
$('#displayButton').click(function () {
|
||||||
|
$('#addForm').hide();
|
||||||
|
$('#addTable').show();
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// 🔥 BUTTON TOGGLE LOGIC
|
// ===============================
|
||||||
|
|
||||||
$('#addButton').click(function () {
|
|
||||||
$('#addForm').show();
|
|
||||||
$('#addTable').hide();
|
|
||||||
|
|
||||||
localStorage.setItem("viewMode", "form");
|
|
||||||
});
|
|
||||||
|
|
||||||
$('#displayButton').click(function () {
|
|
||||||
$('#addForm').hide();
|
|
||||||
$('#addTable').show();
|
|
||||||
|
|
||||||
localStorage.setItem("viewMode", "table");
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
// STATE → DISTRICT
|
// STATE → DISTRICT
|
||||||
$('#state_Id').change(function () {
|
// ===============================
|
||||||
|
if ($('#state_Id').length) {
|
||||||
|
|
||||||
var stateId = $(this).val();
|
$('#state_Id').change(function () {
|
||||||
|
|
||||||
if (stateId) {
|
var stateId = $(this).val();
|
||||||
|
|
||||||
$.ajax({
|
if (stateId) {
|
||||||
url: '/get_districts/' + stateId,
|
|
||||||
type: 'GET',
|
|
||||||
|
|
||||||
success: function (data) {
|
$.ajax({
|
||||||
|
url: '/get_districts/' + stateId,
|
||||||
|
type: 'GET',
|
||||||
|
|
||||||
var districtDropdown = $('#district_Id');
|
success: function (data) {
|
||||||
|
|
||||||
districtDropdown.empty();
|
var districtDropdown = $('#district_Id');
|
||||||
districtDropdown.append('<option value="" disabled selected>Select District</option>');
|
|
||||||
|
|
||||||
data.forEach(function (district) {
|
districtDropdown.empty();
|
||||||
|
districtDropdown.append('<option value="" disabled selected>Select District</option>');
|
||||||
|
|
||||||
districtDropdown.append(
|
data.forEach(function (district) {
|
||||||
'<option value="' + district.id + '">' + district.name + '</option>'
|
districtDropdown.append(
|
||||||
);
|
'<option value="' + district.id + '">' + district.name + '</option>'
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
});
|
districtDropdown.prop('disabled', false);
|
||||||
|
}
|
||||||
districtDropdown.prop('disabled', false);
|
});
|
||||||
|
}
|
||||||
}
|
});
|
||||||
|
}
|
||||||
});
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
|
// ===============================
|
||||||
// DISTRICT → BLOCK
|
// DISTRICT → BLOCK
|
||||||
$('#district_Id').change(function () {
|
// ===============================
|
||||||
|
if ($('#district_Id').length) {
|
||||||
|
|
||||||
var districtId = $(this).val();
|
$('#district_Id').change(function () {
|
||||||
|
|
||||||
if (districtId) {
|
var districtId = $(this).val();
|
||||||
|
|
||||||
$.ajax({
|
if (districtId) {
|
||||||
url: '/get_blocks/' + districtId,
|
|
||||||
type: 'GET',
|
|
||||||
|
|
||||||
success: function (data) {
|
$.ajax({
|
||||||
|
url: '/get_blocks/' + districtId,
|
||||||
|
type: 'GET',
|
||||||
|
|
||||||
var blockDropdown = $('#block_Id');
|
success: function (data) {
|
||||||
|
|
||||||
blockDropdown.empty();
|
var blockDropdown = $('#block_Id');
|
||||||
blockDropdown.append('<option value="" disabled selected>Select Block</option>');
|
|
||||||
|
|
||||||
data.forEach(function (block) {
|
blockDropdown.empty();
|
||||||
|
blockDropdown.append('<option value="" disabled selected>Select Block</option>');
|
||||||
|
|
||||||
blockDropdown.append(
|
data.forEach(function (block) {
|
||||||
'<option value="' + block.id + '">' + block.name + '</option>'
|
blockDropdown.append(
|
||||||
);
|
'<option value="' + block.id + '">' + block.name + '</option>'
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
});
|
blockDropdown.prop('disabled', false);
|
||||||
|
}
|
||||||
blockDropdown.prop('disabled', false);
|
});
|
||||||
|
}
|
||||||
}
|
});
|
||||||
|
}
|
||||||
});
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
|
// ===============================
|
||||||
// VILLAGE NAME VALIDATION
|
// VILLAGE NAME VALIDATION
|
||||||
$('#Village_Name').on('input', function () {
|
// ===============================
|
||||||
|
if ($('#Village_Name').length) {
|
||||||
|
|
||||||
var villageName = $(this).val();
|
$('#Village_Name').on('input', function () {
|
||||||
var validPattern = /^[A-Za-z ]*$/;
|
|
||||||
|
|
||||||
if (!validPattern.test(villageName)) {
|
var villageName = $(this).val();
|
||||||
|
var validPattern = /^[A-Za-z ]*$/;
|
||||||
|
|
||||||
$('#villageMessage')
|
if (!validPattern.test(villageName)) {
|
||||||
.text('Only letters and spaces are allowed!')
|
|
||||||
.css('color', 'red');
|
|
||||||
|
|
||||||
$('#submitVillage').prop('disabled', true);
|
$('#villageMessage')
|
||||||
|
.text('Only letters and spaces are allowed!')
|
||||||
|
.css('color', 'red');
|
||||||
|
|
||||||
} else {
|
$('#submitVillage').prop('disabled', true);
|
||||||
|
|
||||||
$('#villageMessage').text('');
|
} else {
|
||||||
$('#submitVillage').prop('disabled', false);
|
|
||||||
|
|
||||||
}
|
$('#villageMessage').text('');
|
||||||
|
$('#submitVillage').prop('disabled', false);
|
||||||
});
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// ===============================
|
||||||
// CHECK DUPLICATE VILLAGE
|
// CHECK DUPLICATE VILLAGE
|
||||||
$('#Village_Name, #block_Id').on('change keyup', function () {
|
// ===============================
|
||||||
|
if ($('#Village_Name').length && $('#block_Id').length) {
|
||||||
|
|
||||||
var blockId = $('#block_Id').val();
|
$('#Village_Name, #block_Id').on('change keyup', function () {
|
||||||
var villageName = $('#Village_Name').val().trim();
|
|
||||||
|
|
||||||
if (blockId && villageName) {
|
var blockId = $('#block_Id').val();
|
||||||
|
var villageName = $('#Village_Name').val().trim();
|
||||||
|
|
||||||
$.ajax({
|
if (blockId && villageName) {
|
||||||
|
|
||||||
url: '/check_village',
|
$.ajax({
|
||||||
type: 'POST',
|
url: '/check_village',
|
||||||
|
type: 'POST',
|
||||||
|
|
||||||
data: {
|
data: {
|
||||||
block_Id: blockId,
|
block_Id: blockId,
|
||||||
Village_Name: villageName
|
Village_Name: villageName
|
||||||
},
|
},
|
||||||
|
|
||||||
success: function (response) {
|
success: function (response) {
|
||||||
|
|
||||||
if (response.status === 'exists') {
|
if (response.status === 'exists') {
|
||||||
|
|
||||||
|
$('#villageMessage')
|
||||||
|
.text(response.message)
|
||||||
|
.css('color', 'red');
|
||||||
|
|
||||||
|
$('#submitVillage').prop('disabled', true);
|
||||||
|
|
||||||
|
} else {
|
||||||
|
|
||||||
|
$('#villageMessage')
|
||||||
|
.text(response.message)
|
||||||
|
.css('color', 'green');
|
||||||
|
|
||||||
|
$('#submitVillage').prop('disabled', false);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
error: function () {
|
||||||
|
|
||||||
$('#villageMessage')
|
$('#villageMessage')
|
||||||
.text(response.message)
|
.text('Error checking village name')
|
||||||
.css('color', 'red');
|
.css('color', 'red');
|
||||||
|
|
||||||
$('#submitVillage').prop('disabled', true);
|
$('#submitVillage').prop('disabled', true);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// ===============================
|
||||||
|
// ADD VILLAGE (SAFE SCOPE FIX)
|
||||||
|
// ===============================
|
||||||
|
if ($('#villageForm').length) {
|
||||||
|
|
||||||
|
$('#villageForm').submit(function (event) {
|
||||||
|
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
$.ajax({
|
||||||
|
url: '/add_village',
|
||||||
|
type: 'POST',
|
||||||
|
data: $(this).serialize(),
|
||||||
|
|
||||||
|
success: function (response) {
|
||||||
|
|
||||||
|
if (response && response.status === 'success') {
|
||||||
|
|
||||||
|
alert(response.message || 'Village added successfully!');
|
||||||
|
|
||||||
|
// ✅ clear form
|
||||||
|
$('#villageForm')[0].reset();
|
||||||
|
|
||||||
|
// ✅ switch to table
|
||||||
|
$('#addForm').hide();
|
||||||
|
$('#addTable').show();
|
||||||
|
|
||||||
|
// optional refresh
|
||||||
|
location.reload();
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
|
alert(response.message || 'Error adding village. Please try again.');
|
||||||
$('#villageMessage')
|
|
||||||
.text(response.message)
|
|
||||||
.css('color', 'green');
|
|
||||||
|
|
||||||
$('#submitVillage').prop('disabled', false);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
},
|
},
|
||||||
|
|
||||||
error: function () {
|
error: function () {
|
||||||
|
alert('An error occurred. Please try again.');
|
||||||
$('#villageMessage')
|
|
||||||
.text('Error checking village name')
|
|
||||||
.css('color', 'red');
|
|
||||||
|
|
||||||
$('#submitVillage').prop('disabled', true);
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
});
|
});
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
// ADD VILLAGE
|
|
||||||
$('#villageForm').submit(function (event) {
|
|
||||||
|
|
||||||
event.preventDefault();
|
|
||||||
|
|
||||||
$.ajax({
|
|
||||||
|
|
||||||
url: '/add_village',
|
|
||||||
type: 'POST',
|
|
||||||
data: $(this).serialize(),
|
|
||||||
|
|
||||||
success: function (response) {
|
|
||||||
|
|
||||||
if (response.status === 'success') {
|
|
||||||
|
|
||||||
alert('Village added successfully!');
|
|
||||||
location.reload();
|
|
||||||
|
|
||||||
} else {
|
|
||||||
|
|
||||||
alert(response.message || 'Error adding village. Please try again.');
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
},
|
|
||||||
|
|
||||||
error: function () {
|
|
||||||
|
|
||||||
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
|
// ===============================
|
||||||
localStorage.setItem("viewMode", "table");
|
// DELETE FUNCTION (SAFE)
|
||||||
|
// ===============================
|
||||||
|
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 () {
|
success: function (response) {
|
||||||
|
alert(response.message); // ✅ now shows "Village deleted successfully!"
|
||||||
setTimeout(function () {
|
if (element) $(element).closest("tr").remove();
|
||||||
|
|
||||||
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.");
|
||||||
}
|
}
|
||||||
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -20,12 +20,16 @@
|
|||||||
<h2>Add GST Release</h2>
|
<h2>Add GST Release</h2>
|
||||||
<form action="/add_gst_release" method="POST" onsubmit="showSuccessAlert(event)">
|
<form action="/add_gst_release" method="POST" onsubmit="showSuccessAlert(event)">
|
||||||
<div class="row1">
|
<div class="row1">
|
||||||
<div>
|
<div>
|
||||||
<label for="subcontractor">Subcontractor Name:</label>
|
<label for="subcontractor">Subcontractor Name:</label>
|
||||||
<input type="text" id="subcontractor" name="subcontractor" required autocomplete="off"/>
|
<!-- Text input for user-friendly autocomplete -->
|
||||||
<input type="hidden" id="subcontractor_id" name="subcontractor_id"/>
|
<input type="text" id="subcontractor" name="subcontractor" required autocomplete="off"/>
|
||||||
<div id="subcontractor_list" class="autocomplete-items"></div>
|
|
||||||
</div>
|
<!-- Hidden input for backend; must match model's Contractor_ID -->
|
||||||
|
<input type="hidden" id="subcontractor_id" name="Contractor_ID"/>
|
||||||
|
|
||||||
|
<div id="subcontractor_list" class="autocomplete-items"></div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<label for="PMC_No">PMC No:</label><br>
|
<label for="PMC_No">PMC No:</label><br>
|
||||||
@@ -37,19 +41,19 @@
|
|||||||
</select><br><br>
|
</select><br><br>
|
||||||
|
|
||||||
<label for="invoice_No">Invoice No:</label><br>
|
<label for="invoice_No">Invoice No:</label><br>
|
||||||
<input type="text" id="invoice_No" name="invoice_No" required><br><br>
|
<input type="text" id="invoice_No" name="Invoice_No" required><br><br>
|
||||||
|
|
||||||
<label for="basic_amount">Basic Amount:</label><br>
|
<label for="basic_amount">Basic Amount:</label><br>
|
||||||
<input type="number" step="0.01" id="basic_amount" name="basic_amount" placeholder="₹ - 00.00" required><br><br>
|
<input type="number" step="0.01" id="basic_amount" name="Basic_Amount" placeholder="₹ - 00.00" required><br><br>
|
||||||
|
|
||||||
<label for="final_amount">Final Amount:</label><br>
|
<label for="final_amount">Final Amount:</label><br>
|
||||||
<input type="number" step="0.01" id="final_amount" name="final_amount" placeholder="₹ - 00.00" required><br><br>
|
<input type="number" step="0.01" id="final_amount" name="Final_Amount" placeholder="₹ - 00.00" required><br><br>
|
||||||
|
|
||||||
<label for="total_amount">Total Amount:</label><br>
|
<label for="total_amount">Total Amount:</label><br>
|
||||||
<input type="number" step="0.01" id="total_amount" name="total_amount" placeholder="₹ - 00.00" required><br><br>
|
<input type="number" step="0.01" id="total_amount" name="Total_Amount" placeholder="₹ - 00.00" required><br><br>
|
||||||
|
|
||||||
<label for="utr">UTR:</label><br>
|
<label for="utr">UTR:</label><br>
|
||||||
<input type="text" id="utr" name="utr" required><br><br>
|
<input type="text" id="utr" name="UTR" required><br><br>
|
||||||
|
|
||||||
<button type="submit">Submit GST Release</button>
|
<button type="submit">Submit GST Release</button>
|
||||||
</form>
|
</form>
|
||||||
@@ -68,96 +72,110 @@
|
|||||||
{% 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>
|
||||||
<tr>
|
<tr>
|
||||||
<th class="sortable-header">GST_Release_Id</th>
|
<th class="sortable-header">GST_Release_Id</th>
|
||||||
<th class="sortable-header">PMC_No</th>
|
<th class="sortable-header">PMC_No</th>
|
||||||
<th>Invoice_No</th>
|
<th>Invoice_No</th>
|
||||||
<th>Basic_Amount</th>
|
<th>Basic_Amount</th>
|
||||||
<th>Final_Amount</th>
|
<th>Final_Amount</th>
|
||||||
<th>Total_Amount</th>
|
<th>Total_Amount</th>
|
||||||
<th>UTR</th>
|
<th>UTR</th>
|
||||||
<th>Update</th>
|
<th>Update</th>
|
||||||
<th>Delete</th>
|
<th>Delete</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<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="{{ url_for('gst_release_bp.delete_gst_release', gst_release_id=gst_rel.gst_release_id) }}"
|
||||||
<a href="/delete_gst_release/{{ gst_rel[0] }}"
|
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" class="icon">
|
||||||
<img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" alt="Delete"
|
</a>
|
||||||
class="icon">
|
</td>
|
||||||
</a>
|
</tr>
|
||||||
</td>
|
{% endfor %}
|
||||||
</tr>
|
|
||||||
{% endfor %}
|
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
document.addEventListener('DOMContentLoaded', function() {
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
// Handle subcontractor autocomplete
|
|
||||||
document.getElementById("subcontractor").addEventListener("input", function () {
|
const subcontractorInput = document.getElementById("subcontractor");
|
||||||
|
const subcontractorIdInput = document.getElementById("subcontractor_id");
|
||||||
|
const subcontractorList = document.getElementById("subcontractor_list");
|
||||||
|
const pmcDropdown = document.getElementById("PMC_No");
|
||||||
|
const form = document.querySelector('form');
|
||||||
|
|
||||||
|
// --------------------------
|
||||||
|
// Subcontractor autocomplete
|
||||||
|
// --------------------------
|
||||||
|
subcontractorInput.addEventListener("input", function () {
|
||||||
const query = this.value;
|
const query = this.value;
|
||||||
const list = document.getElementById("subcontractor_list");
|
|
||||||
|
|
||||||
if (query.length < 2) {
|
if (query.length < 2) {
|
||||||
list.innerHTML = '';
|
subcontractorList.innerHTML = '';
|
||||||
|
subcontractorIdInput.value = ''; // reset hidden id
|
||||||
|
pmcDropdown.innerHTML = '<option value="">Select PMC No</option>'; // reset PMC dropdown
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
fetch(`/search_subcontractor?query=${encodeURIComponent(query)}`)
|
fetch(`/search_subcontractor?query=${encodeURIComponent(query)}`)
|
||||||
.then(response => response.json())
|
.then(response => response.json())
|
||||||
.then(data => {
|
.then(data => {
|
||||||
list.innerHTML = '';
|
subcontractorList.innerHTML = '';
|
||||||
data.results.forEach(item => {
|
data.results.forEach(item => {
|
||||||
const div = document.createElement("div");
|
const div = document.createElement("div");
|
||||||
div.setAttribute("data-id", item.id);
|
div.setAttribute("data-id", item.id);
|
||||||
div.textContent = item.name;
|
div.textContent = item.name;
|
||||||
list.appendChild(div);
|
subcontractorList.appendChild(div);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
// Handle subcontractor selection
|
// --------------------------
|
||||||
document.getElementById("subcontractor_list").addEventListener("click", function (e) {
|
// Subcontractor selection
|
||||||
|
// --------------------------
|
||||||
|
subcontractorList.addEventListener("click", function (e) {
|
||||||
const selectedId = e.target.getAttribute("data-id");
|
const selectedId = e.target.getAttribute("data-id");
|
||||||
const selectedName = e.target.textContent;
|
const selectedName = e.target.textContent;
|
||||||
|
|
||||||
if (selectedId) {
|
if (selectedId) {
|
||||||
document.getElementById("subcontractor_id").value = selectedId;
|
// Set hidden field for backend
|
||||||
document.getElementById("subcontractor").value = selectedName;
|
subcontractorIdInput.value = selectedId;
|
||||||
document.getElementById("subcontractor_list").innerHTML = "";
|
|
||||||
|
|
||||||
// Update PMC dropdown for selected subcontractor
|
// Set text input to selected name
|
||||||
|
subcontractorInput.value = selectedName;
|
||||||
|
|
||||||
|
// Clear the autocomplete list
|
||||||
|
subcontractorList.innerHTML = "";
|
||||||
|
|
||||||
|
// Fetch and populate PMC dropdown
|
||||||
fetch(`/get_pmc_nos_by_subcontractor/${encodeURIComponent(selectedId)}`)
|
fetch(`/get_pmc_nos_by_subcontractor/${encodeURIComponent(selectedId)}`)
|
||||||
.then(response => response.json())
|
.then(response => response.json())
|
||||||
.then(data => {
|
.then(data => {
|
||||||
const pmcDropdown = document.getElementById("PMC_No");
|
|
||||||
pmcDropdown.innerHTML = '<option value="">Select PMC No</option>';
|
pmcDropdown.innerHTML = '<option value="">Select PMC No</option>';
|
||||||
|
|
||||||
data.pmc_nos.forEach(pmc => {
|
data.pmc_nos.forEach(pmc => {
|
||||||
const option = document.createElement("option");
|
const option = document.createElement("option");
|
||||||
option.value = pmc;
|
option.value = pmc;
|
||||||
@@ -167,6 +185,22 @@ document.addEventListener('DOMContentLoaded', function() {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// --------------------------
|
||||||
|
// Form submit validation
|
||||||
|
// --------------------------
|
||||||
|
form.addEventListener('submit', function(e) {
|
||||||
|
if (!subcontractorIdInput.value) {
|
||||||
|
e.preventDefault();
|
||||||
|
alert("Please select a subcontractor from the list.");
|
||||||
|
subcontractorInput.focus();
|
||||||
|
} else if (!pmcDropdown.value) {
|
||||||
|
e.preventDefault();
|
||||||
|
alert("Please select a PMC No.");
|
||||||
|
pmcDropdown.focus();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|||||||
@@ -1,59 +1,73 @@
|
|||||||
{% extends 'base.html' %}
|
{% extends 'base.html' %}
|
||||||
{% block content %}
|
{% block content %}
|
||||||
|
|
||||||
<head>
|
<head>
|
||||||
<title>Manage Hold Types</title>
|
<title>Manage Hold Types</title>
|
||||||
<script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
|
<script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
|
||||||
<!-- <script src="{{ url_for('static', filename='js/hold_types.js') }}"></script> -->
|
<!-- <script src="{{ url_for('static', filename='js/hold_types.js') }}"></script> -->
|
||||||
<link rel="stylesheet" type="text/css" href="{{ url_for('static', filename='css/style.css') }}">
|
<link rel="stylesheet" type="text/css" href="{{ url_for('static', filename='css/style.css') }}">
|
||||||
<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>
|
||||||
|
|
||||||
<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">
|
<div id="addForm">
|
||||||
<h2>Add Hold Types</h2>
|
<h2>Add Hold Types</h2>
|
||||||
<form id="holdTypeForm" method="POST" action="{{ url_for('hold_types.add_hold_type') }}">
|
<form id="holdTypeForm" method="POST" action="{{ url_for('hold_types.add_hold_type') }}">
|
||||||
<label>Enter Hold Amount Type:</label>
|
<label>Enter Hold Amount Type:</label>
|
||||||
<input type="text" id="hold_type" name="hold_type" placeholder="Enter Type" required>
|
<input type="text" id="hold_type" name="hold_type" placeholder="Enter Type" required>
|
||||||
<span id="holdTypeMessage"></span>
|
<span id="holdTypeMessage"></span>
|
||||||
<button type="submit" value="Add Hold Type" id="successPopup">Add Hold Type</button>
|
<button type="submit" value="Add Hold Type" id="successPopup">Add Hold Type</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="addTable" style="display: none;">
|
<div id="addTable" style="display: none;">
|
||||||
<div class="search-container">
|
<div class="search-container">
|
||||||
<h2>Hold Type List</h2>
|
<h2>Hold Type List</h2>
|
||||||
<input type="text" id="searchBar" placeholder="Searching..." onkeyup="searchTable()">
|
<input type="text" id="searchBar" placeholder="Searching..." onkeyup="searchTable()">
|
||||||
</div>
|
</div>
|
||||||
<table id="sortableTable" border="1">
|
<table id="sortableTable" border="1">
|
||||||
<tr>
|
<tr>
|
||||||
<th>ID</th>
|
<th>ID</th>
|
||||||
<th class="sortable-header">
|
<th class="sortable-header">
|
||||||
Hold Type
|
Hold Type
|
||||||
<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>
|
||||||
{% for htd in Hold_Types_data %}
|
{% for htd in Hold_Types_data %}
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{ htd['hold_type_id'] }}</td>
|
<td>{{ htd['hold_type_id'] }}</td>
|
||||||
<td>{{ htd['hold_type'] }}</td>
|
<td>{{ htd['hold_type'] }}</td>
|
||||||
<td><a href="{{ url_for('hold_types.edit_hold_type', id=htd['hold_type_id']) }}">Edit</a></td>
|
<td style="text-align:center;">
|
||||||
<td><button class="delete-button" data-id="{{ htd['hold_type_id'] }}">Delete</button></td>
|
<a href="{{ url_for('hold_types.edit_hold_type', id=htd['hold_type_id']) }}">
|
||||||
</tr>
|
<img src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit"
|
||||||
{% endfor %}
|
class="icon">
|
||||||
</table>
|
</a>
|
||||||
|
</td>
|
||||||
|
|
||||||
<a href="/">Back to Dashboard</a>
|
<td style="text-align:center;">
|
||||||
</div>
|
<a href="{{ url_for('hold_types.delete_hold_type', id=htd['hold_type_id']) }}"
|
||||||
|
onclick="return confirm('Are you sure you want to delete this hold type?');">
|
||||||
|
|
||||||
|
<img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" alt="Delete"
|
||||||
|
class="icon">
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{% endfor %}
|
||||||
|
</table>
|
||||||
|
|
||||||
|
<a href="/">Back to Dashboard</a>
|
||||||
|
</div>
|
||||||
</body>
|
</body>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
|||||||
@@ -1,176 +1,322 @@
|
|||||||
{% extends 'base.html' %}
|
{% extends 'base.html' %} {% block content %}
|
||||||
{% 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 rel="stylesheet" type="text/css" href="{{ url_for('static', filename='css/invoice.css') }}">
|
<link
|
||||||
<script src="{{ url_for('static', filename='js/invoice.js') }}"></script>
|
rel="stylesheet"
|
||||||
<script src="{{ url_for('static', filename='js/holdAmount.js') }}"></script>
|
type="text/css"
|
||||||
<script src="{{ url_for('static', filename='js/search_on_table.js') }}"></script>
|
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/holdAmount.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 class="alert alert-success alert-dismissible fade show mt-3" role="alert">
|
<div
|
||||||
|
class="alert alert-success alert-dismissible fade show mt-3"
|
||||||
|
role="alert"
|
||||||
|
>
|
||||||
✅ Invoice added successfully!
|
✅ Invoice added successfully!
|
||||||
<button type="button" class="btn-close" data-bs-dismiss="alert" aria-label="Close"></button>
|
<button
|
||||||
</div>
|
type="button"
|
||||||
{% endif %}
|
class="btn-close"
|
||||||
|
data-bs-dismiss="alert"
|
||||||
|
aria-label="Close"
|
||||||
<!-- Flash Messages -->
|
></button>
|
||||||
{% with messages = get_flashed_messages(with_categories=true) %}
|
</div>
|
||||||
{% if messages %}
|
|
||||||
<div class="flash-messages">
|
|
||||||
{% for category, message in messages %}
|
|
||||||
<div class="alert {{ category }}">{{ message }}</div>
|
|
||||||
{% endfor %}
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
{% endif %}
|
||||||
{% endwith %}
|
|
||||||
|
|
||||||
<div class="button-container">
|
<!-- Flash Messages -->
|
||||||
|
{% with messages = get_flashed_messages(with_categories=true) %} {% if
|
||||||
|
messages %}
|
||||||
|
<div class="flash-messages">
|
||||||
|
{% for category, message in messages %}
|
||||||
|
<div class="alert {{ category }}">{{ message }}</div>
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
{% endif %} {% endwith %}
|
||||||
|
|
||||||
|
<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 id="invoiceForm" action="{{ url_for('invoice.add_invoice') }}" method="POST">
|
<form
|
||||||
<div class="row1">
|
id="invoiceForm"
|
||||||
<div>
|
action="{{ url_for('invoice.add_invoice') }}"
|
||||||
<label for="subcontractor">Subcontractor Name:</label>
|
method="POST"
|
||||||
<input type="text" id="subcontractor" name="subcontractor" required autocomplete="off"/>
|
>
|
||||||
<input type="hidden" id="subcontractor_id" name="subcontractor_id"/>
|
<div class="row1">
|
||||||
<div id="subcontractor_list"></div>
|
<div>
|
||||||
</div>
|
<label for="subcontractor">Subcontractor Name:</label>
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
id="subcontractor"
|
||||||
|
name="subcontractor"
|
||||||
|
required
|
||||||
|
autocomplete="off"
|
||||||
|
/>
|
||||||
|
<input type="hidden" id="subcontractor_id" name="subcontractor_id" />
|
||||||
|
<div id="subcontractor_list"></div>
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="row2">
|
<div class="row2">
|
||||||
<div>
|
<div>
|
||||||
<label for="village">Village Name:</label>
|
<label for="village">Village Name:</label>
|
||||||
<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 }}">{{ village.Village_Name }}</option>
|
<option value="{{ village.Village_Name }}">
|
||||||
{% endfor %}
|
{{ village.Village_Name }}
|
||||||
</select>
|
</option>
|
||||||
</div>
|
{% endfor %}
|
||||||
<div>
|
</select>
|
||||||
<label for="pmc_no">PMC No:</label>
|
|
||||||
<input type="text" id="pmc_no" name="pmc_no" required/>
|
|
||||||
<div id="pmc_no_list" class="autocomplete-list"></div>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
<div>
|
||||||
<div class="row2">
|
<label for="pmc_no">PMC No:</label>
|
||||||
<div>
|
<input type="text" id="pmc_no" name="pmc_no" required />
|
||||||
<label for="work_type">Work Type:</label>
|
<div id="pmc_no_list" class="autocomplete-list"></div>
|
||||||
<input type="text" id="work_type" name="work_type" required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="invoice_details">Invoice Details:</label>
|
|
||||||
<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="work_type">Work Type:</label>
|
||||||
<input type="text" id="invoice_no" name="invoice_no" required/>
|
<input type="text" id="work_type" name="work_type" required />
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="invoice_date">Invoice Date:</label>
|
|
||||||
<input type="date" id="invoice_date" name="invoice_date" required/>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
<div>
|
||||||
<div class="row3">
|
<label for="invoice_details">Invoice Details:</label>
|
||||||
<div>
|
<textarea
|
||||||
<label for="basic_amount">Basic Amount:</label>
|
id="invoice_details"
|
||||||
<input type="number" step="0.01" id="basic_amount" name="basic_amount" placeholder="₹ - 00.00" required/>
|
name="invoice_details"
|
||||||
</div>
|
required
|
||||||
<div>
|
></textarea>
|
||||||
<label for="debit_amount">Debit Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="debit_amount" name="debit_amount" placeholder="₹ - 00.00" required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="after_debit_amount">After Debit Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="after_debit_amount" name="after_debit_amount" placeholder="₹ - 00.00" readonly required/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="row3">
|
|
||||||
<div class="percentage-field">
|
|
||||||
<label for="gst_percentage">GST %:</label>
|
|
||||||
<input type="number" step="0.01" id="gst_percentage" name="gst_percentage" placeholder="%"/>
|
|
||||||
<label for="gst_amount">GST Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="gst_amount" name="gst_amount" placeholder="₹ - 00.00" readonly required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="amount">Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="amount" name="amount" placeholder="₹ - 00.00" readonly required/>
|
|
||||||
</div>
|
|
||||||
<div class="percentage-field">
|
|
||||||
<label for="tds_percentage">TDS %:</label>
|
|
||||||
<input type="number" step="0.01" id="tds_percentage" name="tds_percentage" placeholder="%"/>
|
|
||||||
<label for="tds_amount">TDS Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="tds_amount" name="tds_amount" placeholder="₹ - 00.00" readonly required/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="row3">
|
|
||||||
<div class="percentage-field">
|
|
||||||
<label for="sd_percentage">SD %:</label>
|
|
||||||
<input type="number" step="0.01" id="sd_percentage" name="sd_percentage" placeholder="%"/>
|
|
||||||
<label for="sd_amount">SD Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="sd_amount" name="sd_amount" placeholder="₹ - 00.00" readonly required>
|
|
||||||
</div>
|
|
||||||
<div class="percentage-field">
|
|
||||||
<label for="commission_percentage">On Commission %:</label>
|
|
||||||
<input type="number" step="0.01" id="commission_percentage" name="commission_percentage" placeholder="%"/>
|
|
||||||
<label for="on_commission">On Commission:</label>
|
|
||||||
<input type="number" step="0.01" id="on_commission" name="on_commission" placeholder="₹ - 00.00" readonly required>
|
|
||||||
</div>
|
|
||||||
<div class="percentage-field">
|
|
||||||
<label for="hydro_percentage">Hydro Testing %:</label>
|
|
||||||
<input type="number" step="0.01" id="hydro_percentage" name="hydro_percentage" placeholder="%"/>
|
|
||||||
<label for="hydro_testing">Hydro Testing:</label>
|
|
||||||
<input type="number" step="0.01" id="hydro_testing" name="hydro_testing" placeholder="₹ - 00.00" readonly required>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="hold-row">
|
|
||||||
<button type="button" id="add_hold_amount" class="button">+ Add Hold Amount</button>
|
|
||||||
</div>
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<!-- Dynamically added hold amount fields -->
|
<div class="row2">
|
||||||
<div id="hold_amount_container"></div>
|
<div>
|
||||||
|
<label for="invoice_no">Invoice No:</label>
|
||||||
<div class="row2">
|
<input type="text" id="invoice_no" name="invoice_no" required />
|
||||||
<div>
|
|
||||||
<label for="gst_sd_amount">GST SD Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="gst_sd_amount" name="gst_sd_amount" placeholder="₹ - 00.00" required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="final_amount">Final Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="final_amount" name="final_amount" placeholder="₹ - 00.00" required/>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="invoice_date">Invoice Date:</label>
|
||||||
|
<input type="date" id="invoice_date" name="invoice_date" required />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<button type="submit" class="button">Submit</button>
|
<div class="row3">
|
||||||
|
<div>
|
||||||
|
<label for="basic_amount">Basic Amount:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="basic_amount"
|
||||||
|
name="basic_amount"
|
||||||
|
placeholder="₹ - 00.00"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="debit_amount">Debit Amount:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="debit_amount"
|
||||||
|
name="debit_amount"
|
||||||
|
placeholder="₹ - 00.00"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="after_debit_amount">After Debit Amount:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="after_debit_amount"
|
||||||
|
name="after_debit_amount"
|
||||||
|
placeholder="₹ - 00.00"
|
||||||
|
readonly
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row3">
|
||||||
|
<div class="percentage-field">
|
||||||
|
<label for="gst_percentage">GST %:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="gst_percentage"
|
||||||
|
name="gst_percentage"
|
||||||
|
placeholder="%"
|
||||||
|
/>
|
||||||
|
<label for="gst_amount">GST Amount:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="gst_amount"
|
||||||
|
name="gst_amount"
|
||||||
|
placeholder="₹ - 00.00"
|
||||||
|
readonly
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="amount">Amount:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="amount"
|
||||||
|
name="amount"
|
||||||
|
placeholder="₹ - 00.00"
|
||||||
|
readonly
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="percentage-field">
|
||||||
|
<label for="tds_percentage">TDS %:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="tds_percentage"
|
||||||
|
name="tds_percentage"
|
||||||
|
placeholder="%"
|
||||||
|
/>
|
||||||
|
<label for="tds_amount">TDS Amount:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="tds_amount"
|
||||||
|
name="tds_amount"
|
||||||
|
placeholder="₹ - 00.00"
|
||||||
|
readonly
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="row3">
|
||||||
|
<div class="percentage-field">
|
||||||
|
<label for="sd_percentage">SD %:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="sd_percentage"
|
||||||
|
name="sd_percentage"
|
||||||
|
placeholder="%"
|
||||||
|
/>
|
||||||
|
<label for="sd_amount">SD Amount:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="sd_amount"
|
||||||
|
name="sd_amount"
|
||||||
|
placeholder="₹ - 00.00"
|
||||||
|
readonly
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="percentage-field">
|
||||||
|
<label for="commission_percentage">On Commission %:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="commission_percentage"
|
||||||
|
name="commission_percentage"
|
||||||
|
placeholder="%"
|
||||||
|
/>
|
||||||
|
<label for="on_commission">On Commission:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="on_commission"
|
||||||
|
name="on_commission"
|
||||||
|
placeholder="₹ - 00.00"
|
||||||
|
readonly
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="percentage-field">
|
||||||
|
<label for="hydro_percentage">Hydro Testing %:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="hydro_percentage"
|
||||||
|
name="hydro_percentage"
|
||||||
|
placeholder="%"
|
||||||
|
/>
|
||||||
|
<label for="hydro_testing">Hydro Testing:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="hydro_testing"
|
||||||
|
name="hydro_testing"
|
||||||
|
placeholder="₹ - 00.00"
|
||||||
|
readonly
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="hold-row">
|
||||||
|
<button type="button" id="add_hold_amount" class="button">
|
||||||
|
+ Add Hold Amount
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Dynamically added hold amount fields -->
|
||||||
|
<div id="hold_amount_container"></div>
|
||||||
|
|
||||||
|
<div class="row2">
|
||||||
|
<div>
|
||||||
|
<label for="gst_sd_amount">GST SD Amount:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="gst_sd_amount"
|
||||||
|
name="gst_sd_amount"
|
||||||
|
placeholder="₹ - 00.00"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="final_amount">Final Amount:</label>
|
||||||
|
<input
|
||||||
|
type="number"
|
||||||
|
step="0.01"
|
||||||
|
id="final_amount"
|
||||||
|
name="final_amount"
|
||||||
|
placeholder="₹ - 00.00"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<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 type="text" id="searchBar" placeholder="Searching..." onkeyup="searchTable()">
|
<input
|
||||||
{% if invoices %}
|
type="text"
|
||||||
<table class="invoice-table">
|
id="searchBar"
|
||||||
|
placeholder="Searching..."
|
||||||
|
onkeyup="searchTable()"
|
||||||
|
/>
|
||||||
|
{% if invoices %}
|
||||||
|
<table class="invoice-table">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th>Invoice Id</th>
|
<th>Invoice Id</th>
|
||||||
<th>SubContractor Name</th>
|
<th>SubContractor Name</th>
|
||||||
<th>PMC No</th>
|
<th>PMC No</th>
|
||||||
@@ -192,11 +338,11 @@
|
|||||||
<th>Final Amount</th>
|
<th>Final Amount</th>
|
||||||
<th>Update</th>
|
<th>Update</th>
|
||||||
<th>Delete</th>
|
<th>Delete</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
{% for invoice in invoices %}
|
{% for invoice in invoices %}
|
||||||
<tr>
|
<tr>
|
||||||
<td>{{ invoice.Invoice_Id }}</td>
|
<td>{{ invoice.Invoice_Id }}</td>
|
||||||
<td>{{ invoice.Contractor_Name }}</td>
|
<td>{{ invoice.Contractor_Name }}</td>
|
||||||
<td>{{ invoice.PMC_No }}</td>
|
<td>{{ invoice.PMC_No }}</td>
|
||||||
@@ -217,62 +363,77 @@
|
|||||||
<td>{{ invoice.GST_SD_Amount }}</td>
|
<td>{{ invoice.GST_SD_Amount }}</td>
|
||||||
<td>{{ invoice.Final_Amount }}</td>
|
<td>{{ invoice.Final_Amount }}</td>
|
||||||
<td>
|
<td>
|
||||||
<!-- Edit -->
|
<!-- Edit -->
|
||||||
<a href="{{ url_for('invoice.edit_invoice', invoice_id=invoice.Invoice_Id) }}">
|
<a
|
||||||
<img src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit" class="icon">
|
href="{{ url_for('invoice.edit_invoice', invoice_id=invoice.Invoice_Id) }}"
|
||||||
</a>
|
>
|
||||||
|
<img
|
||||||
|
src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}"
|
||||||
|
alt="Edit"
|
||||||
|
class="icon edit-btn"
|
||||||
|
data-id="{{ invoice.Invoice_Id }}"
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
</td>
|
</td>
|
||||||
<td>
|
<td>
|
||||||
<!-- Delete -->
|
<a
|
||||||
<a href="{{ url_for('invoice.delete_invoice_route', invoice_id=invoice.Invoice_Id) }}" onclick="return confirm('Are you sure?')">
|
href="javascript:void(0);"
|
||||||
<img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" alt="Delete" class="icon">
|
onclick="deleteInvoice({{ invoice.Invoice_Id }}, this)"
|
||||||
</a>
|
>
|
||||||
|
<img
|
||||||
|
src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}"
|
||||||
|
alt="Delete"
|
||||||
|
class="icon"
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
{% 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('commission_percentage');
|
const commissionPercentage = document.getElementById(
|
||||||
const onCommission = document.getElementById('on_commission');
|
"commission_percentage",
|
||||||
const hydroPercentage = document.getElementById('hydro_percentage');
|
);
|
||||||
const hydroTesting = document.getElementById('hydro_testing');
|
const onCommission = document.getElementById("on_commission");
|
||||||
const gstSdAmount = document.getElementById('gst_sd_amount');
|
const hydroPercentage = document.getElementById("hydro_percentage");
|
||||||
const finalAmount = document.getElementById('final_amount');
|
const hydroTesting = document.getElementById("hydro_testing");
|
||||||
|
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() {
|
||||||
const basic = parseFloat(basicAmount.value) || 0;
|
const basic = parseFloat(basicAmount.value) || 0;
|
||||||
const debit = parseFloat(debitAmount.value) || 0;
|
const debit = parseFloat(debitAmount.value) || 0;
|
||||||
const afterDebit = basic - debit;
|
const afterDebit = basic - debit;
|
||||||
afterDebitAmount.value = afterDebit.toFixed(2);
|
afterDebitAmount.value = afterDebit.toFixed(2);
|
||||||
calculateGST();
|
calculateGST();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calculate GST and Amount
|
// Calculate GST and Amount
|
||||||
function calculateGST() {
|
function calculateGST() {
|
||||||
const baseAmount = parseFloat(afterDebitAmount.value) || 0;
|
const baseAmount = parseFloat(afterDebitAmount.value) || 0;
|
||||||
|
|
||||||
if (gstPercentage.value) {
|
if (gstPercentage.value) {
|
||||||
const gstPerc = parseFloat(gstPercentage.value) || 0;
|
const gstPerc = parseFloat(gstPercentage.value) || 0;
|
||||||
const gstAmt = (baseAmount * gstPerc) / 100;
|
const gstAmt = (baseAmount * gstPerc) / 100;
|
||||||
gstAmount.value = gstAmt.toFixed(2);
|
gstAmount.value = gstAmt.toFixed(2);
|
||||||
@@ -280,91 +441,96 @@ document.addEventListener('DOMContentLoaded', function() {
|
|||||||
|
|
||||||
// Calculate Amount (After Debit + GST)
|
// Calculate Amount (After Debit + GST)
|
||||||
amount.value = (baseAmount + gstAmt).toFixed(2);
|
amount.value = (baseAmount + gstAmt).toFixed(2);
|
||||||
} else {
|
} else {
|
||||||
amount.value = baseAmount.toFixed(2);
|
amount.value = baseAmount.toFixed(2);
|
||||||
|
}
|
||||||
|
|
||||||
|
calculateOtherDeductions();
|
||||||
}
|
}
|
||||||
|
|
||||||
calculateOtherDeductions();
|
// Calculate other deductions (TDS, SD, Commission, Hydro)
|
||||||
}
|
function calculateOtherDeductions() {
|
||||||
|
const baseAmount = parseFloat(afterDebitAmount.value) || 0;
|
||||||
|
|
||||||
// Calculate other deductions (TDS, SD, Commission, Hydro)
|
// Calculate TDS
|
||||||
function calculateOtherDeductions() {
|
if (tdsPercentage.value) {
|
||||||
const baseAmount = parseFloat(afterDebitAmount.value) || 0;
|
|
||||||
|
|
||||||
// Calculate TDS
|
|
||||||
if (tdsPercentage.value) {
|
|
||||||
const tdsPerc = parseFloat(tdsPercentage.value) || 0;
|
const tdsPerc = parseFloat(tdsPercentage.value) || 0;
|
||||||
const tdsAmt = (baseAmount * tdsPerc) / 100;
|
const tdsAmt = (baseAmount * tdsPerc) / 100;
|
||||||
tdsAmount.value = tdsAmt.toFixed(2);
|
tdsAmount.value = tdsAmt.toFixed(2);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calculate SD
|
// Calculate SD
|
||||||
if (sdPercentage.value) {
|
if (sdPercentage.value) {
|
||||||
const sdPerc = parseFloat(sdPercentage.value) || 0;
|
const sdPerc = parseFloat(sdPercentage.value) || 0;
|
||||||
const sdAmt = (baseAmount * sdPerc) / 100;
|
const sdAmt = (baseAmount * sdPerc) / 100;
|
||||||
sdAmountInput.value = sdAmt.toFixed(2);
|
sdAmountInput.value = sdAmt.toFixed(2);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calculate Commission
|
// Calculate Commission
|
||||||
if (commissionPercentage.value) {
|
if (commissionPercentage.value) {
|
||||||
const commPerc = parseFloat(commissionPercentage.value) || 0;
|
const commPerc = parseFloat(commissionPercentage.value) || 0;
|
||||||
const commAmt = (baseAmount * commPerc) / 100;
|
const commAmt = (baseAmount * commPerc) / 100;
|
||||||
onCommission.value = commAmt.toFixed(2);
|
onCommission.value = commAmt.toFixed(2);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Calculate Hydro Testing
|
// Calculate Hydro Testing
|
||||||
if (hydroPercentage.value) {
|
if (hydroPercentage.value) {
|
||||||
const hydroPerc = parseFloat(hydroPercentage.value) || 0;
|
const hydroPerc = parseFloat(hydroPercentage.value) || 0;
|
||||||
const hydroAmt = (baseAmount * hydroPerc) / 100;
|
const hydroAmt = (baseAmount * hydroPerc) / 100;
|
||||||
hydroTesting.value = hydroAmt.toFixed(2);
|
hydroTesting.value = hydroAmt.toFixed(2);
|
||||||
|
}
|
||||||
|
|
||||||
|
calculateFinalAmount();
|
||||||
}
|
}
|
||||||
|
|
||||||
calculateFinalAmount();
|
// Calculate final amount
|
||||||
}
|
function calculateFinalAmount() {
|
||||||
|
const amt = parseFloat(amount.value) || 0;
|
||||||
|
const tds = parseFloat(tdsAmount.value) || 0;
|
||||||
|
const sd = parseFloat(sdAmountInput.value) || 0;
|
||||||
|
const commission = parseFloat(onCommission.value) || 0;
|
||||||
|
const hydro = parseFloat(hydroTesting.value) || 0;
|
||||||
|
const gstSd = parseFloat(gstSdAmount.value) || 0;
|
||||||
|
|
||||||
// Calculate final amount
|
// Get hold amounts
|
||||||
function calculateFinalAmount() {
|
let totalHold = 0;
|
||||||
const amt = parseFloat(amount.value) || 0;
|
document
|
||||||
const tds = parseFloat(tdsAmount.value) || 0;
|
.querySelectorAll('input[name="hold_amount[]"]')
|
||||||
const sd = parseFloat(sdAmountInput.value) || 0;
|
.forEach((input) => {
|
||||||
const commission = parseFloat(onCommission.value) || 0;
|
totalHold += parseFloat(input.value) || 0;
|
||||||
const hydro = parseFloat(hydroTesting.value) || 0;
|
});
|
||||||
const gstSd = parseFloat(gstSdAmount.value) || 0;
|
|
||||||
|
|
||||||
// Get hold amounts
|
// Final Amount = Amount - TDS - SD - Commission - Hydro - GST SD - Hold Amounts
|
||||||
let totalHold = 0;
|
const final = amt - tds - sd - commission - hydro - gstSd - totalHold;
|
||||||
document.querySelectorAll('input[name="hold_amount[]"]').forEach(input => {
|
finalAmount.value = final.toFixed(2);
|
||||||
totalHold += parseFloat(input.value) || 0;
|
}
|
||||||
});
|
|
||||||
|
|
||||||
// Final Amount = Amount - TDS - SD - Commission - Hydro - GST SD - Hold Amounts
|
// Add event listeners
|
||||||
const final = amt - tds - sd - commission - hydro - gstSd - totalHold;
|
basicAmount.addEventListener("input", calculateAfterDebitAmount);
|
||||||
finalAmount.value = final.toFixed(2);
|
debitAmount.addEventListener("input", calculateAfterDebitAmount);
|
||||||
}
|
|
||||||
|
|
||||||
// Add event listeners
|
// Percentage fields
|
||||||
basicAmount.addEventListener('input', calculateAfterDebitAmount);
|
gstPercentage.addEventListener("input", calculateGST);
|
||||||
debitAmount.addEventListener('input', calculateAfterDebitAmount);
|
tdsPercentage.addEventListener("input", calculateOtherDeductions);
|
||||||
|
sdPercentage.addEventListener("input", calculateOtherDeductions);
|
||||||
|
commissionPercentage.addEventListener(
|
||||||
|
"input",
|
||||||
|
calculateOtherDeductions,
|
||||||
|
);
|
||||||
|
hydroPercentage.addEventListener("input", calculateOtherDeductions);
|
||||||
|
|
||||||
// Percentage fields
|
// Listen for changes in hold amounts
|
||||||
gstPercentage.addEventListener('input', calculateGST);
|
document.addEventListener("holdAmountChanged", calculateFinalAmount);
|
||||||
tdsPercentage.addEventListener('input', calculateOtherDeductions);
|
});
|
||||||
sdPercentage.addEventListener('input', calculateOtherDeductions);
|
|
||||||
commissionPercentage.addEventListener('input', calculateOtherDeductions);
|
|
||||||
hydroPercentage.addEventListener('input', calculateOtherDeductions);
|
|
||||||
|
|
||||||
// Listen for changes in hold amounts
|
// Optional JS for auto-hiding flash
|
||||||
document.addEventListener('holdAmountChanged', calculateFinalAmount);
|
setTimeout(() => {
|
||||||
});
|
document
|
||||||
|
.querySelectorAll(".alert")
|
||||||
// Optional JS for auto-hiding flash
|
.forEach((el) => (el.style.display = "none"));
|
||||||
setTimeout(() => {
|
}, 5000);
|
||||||
document.querySelectorAll('.alert').forEach(el => el.style.display = 'none');
|
</script>
|
||||||
}, 5000);
|
</div>
|
||||||
</script>
|
|
||||||
|
|
||||||
|
|
||||||
</div>
|
|
||||||
</body>
|
</body>
|
||||||
|
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
{% extends 'base.html' %}
|
{% extends 'base.html' %}
|
||||||
{% block content %}
|
{% 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">
|
||||||
@@ -8,91 +9,104 @@
|
|||||||
<script src="{{ url_for('static', filename='js/search_on_table.js') }}"></script>
|
<script src="{{ url_for('static', filename='js/search_on_table.js') }}"></script>
|
||||||
<script src="{{ url_for('static', filename='js/invoice.js') }}"></script>
|
<script src="{{ url_for('static', filename='js/invoice.js') }}"></script>
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
|
|
||||||
<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 id="addForm" style="display: none;">
|
|
||||||
<h2>Add Payment</h2>
|
|
||||||
|
|
||||||
<form action="/add_payment" method="POST" onsubmit="showSuccessAlert(event)">
|
|
||||||
<div class="row1">
|
|
||||||
<div>
|
|
||||||
<label for="subcontractor">Subcontractor Name:</label>
|
|
||||||
<input type="text" id="subcontractor" name="subcontractor" required autocomplete="off"/>
|
|
||||||
<input type="hidden" id="subcontractor_id" name="subcontractor_id"/>
|
|
||||||
<div id="subcontractor_list" class="autocomplete-items"></div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<label for="PMC_No">PMC No:</label><br>
|
|
||||||
<select id="PMC_No" name="PMC_No" required>
|
|
||||||
<option value="">Select PMC No</option>
|
|
||||||
</select><br><br>
|
|
||||||
|
|
||||||
<label for="invoice_No">Invoice No:</label><br>
|
|
||||||
<input type="number" step="0.01" id="invoice_No" name="invoice_No" ><br><br>
|
|
||||||
|
|
||||||
<label for="Payment_Amount">Amount:</label><br>
|
|
||||||
<input type="number" step="0.01" id="Payment_Amount" name="Payment_Amount" required oninput="calculateTDSAndTotal()"><br><br>
|
|
||||||
|
|
||||||
<label for="TDS_Percentage">TDS Percentage (%):</label><br>
|
|
||||||
<input type="number" step="0.01" id="TDS_Percentage" name="TDS_Percentage" oninput="calculateTDSAndTotal()"><br><br>
|
|
||||||
|
|
||||||
<label for="TDS_Payment_Amount">TDS Amount:</label><br>
|
|
||||||
<input type="number" step="0.01" id="TDS_Payment_Amount" name="TDS_Payment_Amount" required readonly><br><br>
|
|
||||||
|
|
||||||
<label for="total_amount">Total Amount:</label><br>
|
|
||||||
<input type="number" step="0.01" id="total_amount" name="total_amount" required readonly><br><br>
|
|
||||||
|
|
||||||
|
|
||||||
<label for="utr">UTR:</label><br>
|
|
||||||
<input type="text" id="utr" name="utr"><br><br>
|
|
||||||
|
|
||||||
<button type="submit">Submit Payment</button>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div id="successPopup" class="success-popup">
|
|
||||||
<i>✔</i> Payment added successfully!
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div id="addTable" style="display: none;">
|
|
||||||
<div class="search-container">
|
|
||||||
<h2>Payment History</h2>
|
|
||||||
<input type="text" id="searchBar" placeholder="Searching..." onkeyup="searchTable()">
|
|
||||||
</div>
|
</div>
|
||||||
<table id="sortableTable" border="1">
|
|
||||||
<thead>
|
<div id="addForm" style="display: none;">
|
||||||
<tr>
|
<h2>Add Payment</h2>
|
||||||
<th class="sortable-header">Payment ID</th>
|
|
||||||
<th class="sortable-header">PMC No</th>
|
<form action="/add_payment" method="POST" onsubmit="showSuccessAlert(event)">
|
||||||
<th>Invoice No</th>
|
<div class="row1">
|
||||||
<th>Payment Amount</th>
|
<div>
|
||||||
<th>TDS Amount</th>
|
<label for="subcontractor">Subcontractor Name:</label>
|
||||||
<th>Total Amount</th>
|
<input type="text" id="subcontractor" name="subcontractor" required autocomplete="off" />
|
||||||
<th>UTR</th>
|
<input type="hidden" id="subcontractor_id" name="subcontractor_id" />
|
||||||
<th>Update</th>
|
<div id="subcontractor_list" class="autocomplete-items"></div>
|
||||||
<th>Delete</th>
|
</div>
|
||||||
</tr>
|
</div>
|
||||||
</thead>
|
|
||||||
<tbody>
|
<label for="PMC_No">PMC No:</label><br>
|
||||||
{% for payment in payments %}
|
<select id="PMC_No" name="PMC_No" required>
|
||||||
<tr>
|
<option value="">Select PMC No</option>
|
||||||
<td>{{ payment[0] }}</td>
|
</select><br><br>
|
||||||
<td>{{ payment[1] }}</td>
|
|
||||||
<td>{{ payment[2] }}</td>
|
<label for="invoice_No">Invoice No:</label><br>
|
||||||
<td>{{ payment[3] }}</td>
|
<input type="number" step="0.01" id="invoice_No" name="invoice_No"><br><br>
|
||||||
<td>{{ payment[4] }}</td>
|
|
||||||
<td>{{ payment[5] }}</td>
|
<label for="Payment_Amount">Amount:</label><br>
|
||||||
<td>{{ payment[6] }}</td>
|
<input type="number" step="0.01" id="Payment_Amount" name="Payment_Amount" required
|
||||||
<td><a href="/edit_payment/{{ payment[0] }}"><img src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit" class="icon"></a></td>
|
oninput="calculateTDSAndTotal()"><br><br>
|
||||||
<td><a href="/delete_payment/{{ payment[0] }}" onclick="return confirm('Are you sure you want to delete this payment?')"><img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" alt="Delete" class="icon"></a></td>
|
|
||||||
</tr>
|
<label for="TDS_Percentage">TDS Percentage (%):</label><br>
|
||||||
<!-- <tr>
|
<input type="number" step="0.01" id="TDS_Percentage" name="TDS_Percentage"
|
||||||
|
oninput="calculateTDSAndTotal()"><br><br>
|
||||||
|
|
||||||
|
<label for="TDS_Payment_Amount">TDS Amount:</label><br>
|
||||||
|
<input type="number" step="0.01" id="TDS_Payment_Amount" name="TDS_Payment_Amount" required
|
||||||
|
readonly><br><br>
|
||||||
|
|
||||||
|
<label for="total_amount">Total Amount:</label><br>
|
||||||
|
<input type="number" step="0.01" id="total_amount" name="total_amount" required readonly><br><br>
|
||||||
|
|
||||||
|
|
||||||
|
<label for="utr">UTR:</label><br>
|
||||||
|
<input type="text" id="utr" name="utr"><br><br>
|
||||||
|
|
||||||
|
<button type="submit">Submit Payment</button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="successPopup" class="success-popup">
|
||||||
|
<i>✔</i> Payment added successfully!
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="addTable" style="display: none;">
|
||||||
|
<div class="search-container">
|
||||||
|
<h2>Payment History</h2>
|
||||||
|
<input type="text" id="searchBar" placeholder="Searching..." onkeyup="searchTable()">
|
||||||
|
</div>
|
||||||
|
<table id="sortableTable" border="1">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th class="sortable-header">Payment ID</th>
|
||||||
|
<th class="sortable-header">PMC No</th>
|
||||||
|
<th>Invoice No</th>
|
||||||
|
<th>Payment Amount</th>
|
||||||
|
<th>TDS Amount</th>
|
||||||
|
<th>Total Amount</th>
|
||||||
|
<th>UTR</th>
|
||||||
|
<th>Update</th>
|
||||||
|
<th>Delete</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{% for payment in payments %}
|
||||||
|
<tr>
|
||||||
|
<td>{{ payment[0] }}</td>
|
||||||
|
<td>{{ payment[1] }}</td>
|
||||||
|
<td>{{ payment[2] }}</td>
|
||||||
|
<td>{{ payment[3] }}</td>
|
||||||
|
<td>{{ payment[4] }}</td>
|
||||||
|
<td>{{ payment[5] }}</td>
|
||||||
|
<td>{{ payment[6] }}</td>
|
||||||
|
<td><a href="/edit_payment/{{ payment[0] }}"><img
|
||||||
|
src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit"
|
||||||
|
class="icon"></a></td>
|
||||||
|
|
||||||
|
<td>
|
||||||
|
<a href="{{ url_for('payment_bp.delete_payment', payment_id=payment[0]) }}"
|
||||||
|
onclick="return confirm('Are you sure you want to delete this Payment?')">
|
||||||
|
<img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" alt="Delete"
|
||||||
|
class="icon">
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<!-- <tr>
|
||||||
<td>{{ payment['Payment_Id'] }}</td>
|
<td>{{ payment['Payment_Id'] }}</td>
|
||||||
<td>{{ payment['PMC_No'] }}</td>
|
<td>{{ payment['PMC_No'] }}</td>
|
||||||
<td>{{ payment['invoice_no'] }}</td>
|
<td>{{ payment['invoice_no'] }}</td>
|
||||||
@@ -103,91 +117,91 @@
|
|||||||
<td><a href="/edit_payment/{{ payment['Payment_Id'] }}"><img src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit" class="icon"></a></td>
|
<td><a href="/edit_payment/{{ payment['Payment_Id'] }}"><img src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit" class="icon"></a></td>
|
||||||
<td><a href="/delete_payment/{{ payment['Payment_Id'] }}" onclick="return confirm('Are you sure you want to delete this payment?')"><img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" alt="Delete" class="icon"></a></td>
|
<td><a href="/delete_payment/{{ payment['Payment_Id'] }}" onclick="return confirm('Are you sure you want to delete this payment?')"><img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}" alt="Delete" class="icon"></a></td>
|
||||||
</tr> -->
|
</tr> -->
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
document.getElementById("subcontractor").addEventListener("input", function () {
|
document.getElementById("subcontractor").addEventListener("input", function () {
|
||||||
const query = this.value;
|
const query = this.value;
|
||||||
const list = document.getElementById("subcontractor_list");
|
const list = document.getElementById("subcontractor_list");
|
||||||
|
|
||||||
if (query.length < 2) {
|
if (query.length < 2) {
|
||||||
list.innerHTML = '';
|
list.innerHTML = '';
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
fetch(`/search_subcontractor?query=${encodeURIComponent(query)}`)
|
fetch(`/search_subcontractor?query=${encodeURIComponent(query)}`)
|
||||||
.then(response => response.json())
|
.then(response => response.json())
|
||||||
.then(data => {
|
.then(data => {
|
||||||
list.innerHTML = '';
|
list.innerHTML = '';
|
||||||
data.results.forEach(item => {
|
data.results.forEach(item => {
|
||||||
const div = document.createElement("div");
|
const div = document.createElement("div");
|
||||||
div.setAttribute("data-id", item.id);
|
div.setAttribute("data-id", item.id);
|
||||||
div.textContent = item.name;
|
div.textContent = item.name;
|
||||||
list.appendChild(div);
|
list.appendChild(div);
|
||||||
});
|
});
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
document.getElementById("subcontractor_list").addEventListener("click", function (e) {
|
|
||||||
const selectedId = e.target.getAttribute("data-id");
|
|
||||||
const selectedName = e.target.textContent;
|
|
||||||
|
|
||||||
if (selectedId) {
|
|
||||||
document.getElementById("subcontractor_id").value = selectedId;
|
|
||||||
document.getElementById("subcontractor").value = selectedName;
|
|
||||||
document.getElementById("subcontractor_list").innerHTML = ""; // hide the list
|
|
||||||
|
|
||||||
console.log("Contractor id is", selectedId);
|
|
||||||
|
|
||||||
// Fetch PMC numbers
|
|
||||||
fetch(`/get_pmc_nos_by_subcontractor/${encodeURIComponent(selectedId)}`)
|
|
||||||
.then(response => response.json())
|
|
||||||
.then(data => {
|
|
||||||
console.log("Fetched PMC Nos:", data.pmc_nos);
|
|
||||||
const pmcDropdown = document.getElementById("PMC_No");
|
|
||||||
pmcDropdown.innerHTML = "";
|
|
||||||
|
|
||||||
const defaultOption = document.createElement("option");
|
|
||||||
defaultOption.value = "";
|
|
||||||
defaultOption.textContent = "Select PMC No";
|
|
||||||
pmcDropdown.appendChild(defaultOption);
|
|
||||||
|
|
||||||
data.pmc_nos.forEach(pmc => {
|
|
||||||
const option = document.createElement("option");
|
|
||||||
option.value = pmc;
|
|
||||||
option.textContent = pmc;
|
|
||||||
pmcDropdown.appendChild(option);
|
|
||||||
});
|
});
|
||||||
|
});
|
||||||
|
|
||||||
if (data.pmc_nos.length === 0) {
|
document.getElementById("subcontractor_list").addEventListener("click", function (e) {
|
||||||
alert("No PMC Nos found for this subcontractor.");
|
const selectedId = e.target.getAttribute("data-id");
|
||||||
}
|
const selectedName = e.target.textContent;
|
||||||
})
|
|
||||||
.catch(error => {
|
|
||||||
console.error("Error fetching PMC Nos:", error);
|
|
||||||
alert("Failed to fetch PMC numbers.");
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<script>
|
if (selectedId) {
|
||||||
function calculateTDSAndTotal() {
|
document.getElementById("subcontractor_id").value = selectedId;
|
||||||
const amount = parseFloat(document.getElementById("Payment_Amount").value) || 0;
|
document.getElementById("subcontractor").value = selectedName;
|
||||||
const tdsPercent = parseFloat(document.getElementById("TDS_Percentage").value) || 0;
|
document.getElementById("subcontractor_list").innerHTML = ""; // hide the list
|
||||||
|
|
||||||
const tdsAmount = (amount * tdsPercent / 100).toFixed(2);
|
console.log("Contractor id is", selectedId);
|
||||||
const totalAmount = (amount - tdsAmount).toFixed(2);
|
|
||||||
|
|
||||||
document.getElementById("TDS_Payment_Amount").value = tdsAmount;
|
// Fetch PMC numbers
|
||||||
document.getElementById("total_amount").value = totalAmount;
|
fetch(`/get_pmc_nos_by_subcontractor/${encodeURIComponent(selectedId)}`)
|
||||||
}
|
.then(response => response.json())
|
||||||
</script>
|
.then(data => {
|
||||||
|
console.log("Fetched PMC Nos:", data.pmc_nos);
|
||||||
|
const pmcDropdown = document.getElementById("PMC_No");
|
||||||
|
pmcDropdown.innerHTML = "";
|
||||||
|
|
||||||
|
const defaultOption = document.createElement("option");
|
||||||
|
defaultOption.value = "";
|
||||||
|
defaultOption.textContent = "Select PMC No";
|
||||||
|
pmcDropdown.appendChild(defaultOption);
|
||||||
|
|
||||||
|
data.pmc_nos.forEach(pmc => {
|
||||||
|
const option = document.createElement("option");
|
||||||
|
option.value = pmc;
|
||||||
|
option.textContent = pmc;
|
||||||
|
pmcDropdown.appendChild(option);
|
||||||
|
});
|
||||||
|
|
||||||
|
if (data.pmc_nos.length === 0) {
|
||||||
|
alert("No PMC Nos found for this subcontractor.");
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error("Error fetching PMC Nos:", error);
|
||||||
|
alert("Failed to fetch PMC numbers.");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
function calculateTDSAndTotal() {
|
||||||
|
const amount = parseFloat(document.getElementById("Payment_Amount").value) || 0;
|
||||||
|
const tdsPercent = parseFloat(document.getElementById("TDS_Percentage").value) || 0;
|
||||||
|
|
||||||
|
const tdsAmount = (amount * tdsPercent / 100).toFixed(2);
|
||||||
|
const totalAmount = (amount - tdsAmount).toFixed(2);
|
||||||
|
|
||||||
|
document.getElementById("TDS_Payment_Amount").value = tdsAmount;
|
||||||
|
document.getElementById("total_amount").value = totalAmount;
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
|
||||||
<script src="{{ url_for('static', filename='js/showSuccessAlert.js') }}"></script>
|
<script src="{{ url_for('static', filename='js/showSuccessAlert.js') }}"></script>
|
||||||
</body>
|
</body>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
@@ -1,99 +1,129 @@
|
|||||||
{% extends 'base.html' %}
|
{% extends 'base.html' %} {% block content %}
|
||||||
{% 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 rel="stylesheet" type="text/css" href="{{ url_for('static', filename='css/style.css') }}">
|
<link
|
||||||
<script src="{{ url_for('static', filename='js/village.js') }}"></script>
|
rel="stylesheet"
|
||||||
<script src="{{ url_for('static', filename='js/search_on_table.js') }}"></script>
|
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/search_on_table.js') }}"></script>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
|
<!-- Button Container to Center Buttons -->
|
||||||
<!-- Button Container to Center Buttons -->
|
<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">
|
||||||
<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>
|
||||||
<form id="villageForm" method="POST">
|
<form id="villageForm" method="POST">
|
||||||
<label for="state_Id">State:</label>
|
<label for="state_Id">State:</label>
|
||||||
<select id="state_Id" name="state_Id" required>
|
<select id="state_Id" name="state_Id" required>
|
||||||
<option value="" disabled selected>Select State</option>
|
<option value="" disabled selected>Select State</option>
|
||||||
{% for state in states %}
|
{% for state in states %}
|
||||||
<option value="{{ state[0] }}">{{ state[1] }}</option>
|
<option value="{{ state[0] }}">{{ state[1] }}</option>
|
||||||
{% endfor %}
|
{% endfor %}
|
||||||
</select>
|
</select>
|
||||||
|
|
||||||
<label for="district_Id">District:</label>
|
<label for="district_Id">District:</label>
|
||||||
<select id="district_Id" name="district_Id" required disabled>
|
<select id="district_Id" name="district_Id" required disabled>
|
||||||
<option value="" disabled selected>Select District</option>
|
<option value="" disabled selected>Select District</option>
|
||||||
</select>
|
</select>
|
||||||
|
|
||||||
<label for="block_Id">Block:</label>
|
<label for="block_Id">Block:</label>
|
||||||
<select id="block_Id" name="block_Id" required disabled>
|
<select id="block_Id" name="block_Id" required disabled>
|
||||||
<option value="" disabled selected>Select Block</option>
|
<option value="" disabled selected>Select Block</option>
|
||||||
</select>
|
</select>
|
||||||
|
|
||||||
<label for="Village_Name">Village Name:</label>
|
<label for="Village_Name">Village Name:</label>
|
||||||
<input type="text" id="Village_Name" name="Village_Name" placeholder="Enter Village Name" required>
|
<input
|
||||||
<span id="villageMessage"></span>
|
type="text"
|
||||||
|
id="Village_Name"
|
||||||
|
name="Village_Name"
|
||||||
|
placeholder="Enter Village Name"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
<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 type="text" id="searchBar" placeholder="Searching..." onkeyup="searchTable()">
|
<input
|
||||||
|
type="text"
|
||||||
|
id="searchBar"
|
||||||
|
placeholder="Searching..."
|
||||||
|
onkeyup="searchTable()"
|
||||||
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<table id="sortableTable" border="1">
|
<table id="sortableTable" border="1">
|
||||||
<tr>
|
<tr>
|
||||||
<th>Village Sr No</th>
|
<th>Village Sr No</th>
|
||||||
<th class="sortable-header">
|
<th class="sortable-header">
|
||||||
Village Name
|
Village 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></th>
|
</span>
|
||||||
<th class="sortable-header">Block Name
|
</th>
|
||||||
<span class="sort-buttons">
|
<th class="sortable-header">
|
||||||
<span class="sort-asc">⬆️</span>
|
Block Name
|
||||||
<span class="sort-desc">⬇️</span>
|
<span class="sort-buttons">
|
||||||
</span></th>
|
<span class="sort-asc">⬆️</span>
|
||||||
<th>Update</th>
|
<span class="sort-desc">⬇️</span>
|
||||||
<th>Delete</th>
|
</span>
|
||||||
</tr>
|
</th>
|
||||||
{% for village in villages %}
|
<th>Update</th>
|
||||||
<tr>
|
<th>Delete</th>
|
||||||
<td>{{ village[0] }}</td>
|
</tr>
|
||||||
<td>{{ village[1] }}</td>
|
{% for village in villages %}
|
||||||
<td>{{ village[2] }}</td>
|
<tr>
|
||||||
<td>
|
<td>{{ village[0] }}</td>
|
||||||
<a href="{{ url_for('village.edit_village', village_id=village[0]) }}">
|
<td>{{ village[1] }}</td>
|
||||||
<img src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}" alt="Edit"
|
<td>{{ village[2] }}</td>
|
||||||
class="icon">
|
<td>
|
||||||
</a>
|
<a
|
||||||
</td>
|
href="{{ url_for('village.edit_village', village_id=village[0]) }}"
|
||||||
<td>
|
>
|
||||||
<a href="#"
|
<img
|
||||||
onclick="deleteVillage({{ village[0] }}); return false;">
|
src="{{ url_for('static', filename='images/icons/pen_blue_icon.png') }}"
|
||||||
|
alt="Edit"
|
||||||
|
class="icon"
|
||||||
|
/>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<a href="javascript:void(0);"
|
||||||
|
onclick="deleteVillage({{ village[0] }}, this)">
|
||||||
<img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}"
|
<img src="{{ url_for('static', filename='images/icons/bin_red_icon.png') }}"
|
||||||
alt="Delete" class="icon">
|
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 %}
|
||||||
|
|||||||
@@ -9,37 +9,37 @@
|
|||||||
<body>
|
<body>
|
||||||
<h2>Edit GST Release</h2>
|
<h2>Edit GST Release</h2>
|
||||||
|
|
||||||
<form action="/edit_gst_release/{{ gst_release_data[0] }}" method="POST">
|
<form action="/edit_gst_release/{{ gst_release_data.gst_release_id }}" method="POST">
|
||||||
<!-- <label for="invoice_id">Invoice Id:</label><br>-->
|
|
||||||
<!-- <input type="number" id="invoice_id" name="invoice_id" value="{{ gst_release_data[0] }}" required><br><br>-->
|
|
||||||
|
|
||||||
|
<!-- PMC Number -->
|
||||||
<label for="PMC_No">PMC No :</label><br>
|
<label for="PMC_No">PMC No :</label><br>
|
||||||
<input type="number" id="PMC_No" name="PMC_No" value="{{ gst_release_data[1] }}" required><br><br>
|
<input type="text" id="PMC_No" name="PMC_No" value="{{ gst_release_data.pmc_no }}" required><br><br>
|
||||||
|
|
||||||
<label for="invoice_No">Invoice No:</label><br>
|
<!-- Invoice Number -->
|
||||||
<input type="number" step="0.01" id="invoice_No" name="invoice_No" value="{{ gst_release_data[2] }}"
|
<label for="invoice_no">Invoice No:</label><br>
|
||||||
required><br><br>
|
<input type="text" id="invoice_no" name="invoice_no" value="{{ gst_release_data.invoice_no }}" required><br><br>
|
||||||
|
|
||||||
|
<!-- Basic Amount -->
|
||||||
|
<label for="Basic_Amount">Basic Amount:</label><br>
|
||||||
|
<input type="number" step="0.01" id="Basic_Amount" name="Basic_Amount" value="{{ gst_release_data.basic_amount }}" required><br><br>
|
||||||
|
|
||||||
<label for="basic_amount">Basic Amount:</label><br>
|
<!-- Final Amount -->
|
||||||
<input type="number" step="0.01" id="basic_amount" name="basic_amount" value="{{ gst_release_data[3] }}"
|
<label for="Final_Amount">Final Amount:</label><br>
|
||||||
required><br><br>
|
<input type="number" step="0.01" id="Final_Amount" name="Final_Amount" value="{{ gst_release_data.final_amount }}" required><br><br>
|
||||||
|
|
||||||
<label for="final_amount">Final Amount:</label><br>
|
<!-- Total Amount -->
|
||||||
<input type="number" step="0.01" id="final_amount" name="final_amount" value="{{ gst_release_data[4] }}"
|
<label for="Total_Amount">Total Amount:</label><br>
|
||||||
required><br><br>
|
<input type="number" step="0.01" id="Total_Amount" name="Total_Amount" value="{{ gst_release_data.total_amount }}" required><br><br>
|
||||||
|
|
||||||
<label for="total_amount">Total Amount:</label><br>
|
<!-- UTR -->
|
||||||
<input type="number" step="0.01" id="total_amount" name="total_amount" value="{{ gst_release_data[5] }}"
|
<label for="UTR">UTR:</label><br>
|
||||||
required><br><br>
|
<input type="text" id="UTR" name="UTR" value="{{ gst_release_data.utr }}" readonly required><br><br>
|
||||||
|
|
||||||
<label for="utr">UTR:</label><br>
|
<!-- Hidden Contractor ID -->
|
||||||
<input type="text" id="utr" name="utr" value="{{ gst_release_data[6] }}"
|
<input type="hidden" id="Contractor_ID" name="Contractor_ID" value="{{ gst_release_data.contractor_id }}">
|
||||||
required readonly><br><br>
|
|
||||||
|
|
||||||
<button type="submit">Update GST Release</button>
|
<button type="submit">Update GST Release</button>
|
||||||
</form>
|
</form>
|
||||||
|
|
||||||
|
|
||||||
</body>
|
</body>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
|
|
||||||
{% extends 'base.html' %}
|
{% extends 'base.html' %}
|
||||||
|
|
||||||
{% block content %}
|
{% block content %}
|
||||||
|
|
||||||
<head>
|
<head>
|
||||||
<title>Edit Hold Type</title>
|
<title>Edit Hold Type</title>
|
||||||
<link rel="stylesheet" type="text/css" href="{{ url_for('static', filename='css/style.css') }}">
|
<link rel="stylesheet" type="text/css" href="{{ url_for('static', filename='css/style.css') }}">
|
||||||
@@ -10,10 +10,10 @@
|
|||||||
|
|
||||||
<body>
|
<body>
|
||||||
<h2>Edit Hold Type</h2>
|
<h2>Edit Hold Type</h2>
|
||||||
<form id="updateHoldTypeForm">
|
<form id="updateHoldTypeForm" method="POST">
|
||||||
<input type="hidden" id="hold_type_id" value="{{ hold_type[0] }}">
|
<input type="hidden" id="hold_type_id" value="{{ hold_type.hold_type_id }}">
|
||||||
<label for="edit_hold_type">Hold Type:</label>
|
<label for="edit_hold_type">Hold Type:</label>
|
||||||
<input type="text" id="edit_hold_type" name="hold_type" value="{{ hold_type[1] }}" required>
|
<input type="text" id="edit_hold_type" name="hold_type" value="{{ hold_type.hold_type }}" required>
|
||||||
<button type="submit">Update</button>
|
<button type="submit">Update</button>
|
||||||
<a href="{{ url_for('hold_types.add_hold_type') }}"></a>
|
<a href="{{ url_for('hold_types.add_hold_type') }}"></a>
|
||||||
</form>
|
</form>
|
||||||
@@ -37,7 +37,7 @@
|
|||||||
formData.append('hold_type', newHoldType);
|
formData.append('hold_type', newHoldType);
|
||||||
|
|
||||||
$.ajax({
|
$.ajax({
|
||||||
url: '/update_hold_type/' + holdTypeId,
|
url: '/edit_hold_type/' + holdTypeId,
|
||||||
method: 'POST',
|
method: 'POST',
|
||||||
data: formData,
|
data: formData,
|
||||||
processData: false,
|
processData: false,
|
||||||
@@ -55,8 +55,3 @@
|
|||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -2,168 +2,185 @@
|
|||||||
{% block content %}
|
{% 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>Edit Invoice</title>
|
<title>Edit Invoice</title>
|
||||||
<link rel="stylesheet" type="text/css" href="{{ url_for('static', filename='css/invoice.css') }}">
|
<link rel="stylesheet" type="text/css" href="{{ url_for('static', filename='css/invoice.css') }}">
|
||||||
<link rel="stylesheet" type="text/css" href="{{ url_for('static', filename='css/style1.css') }}">
|
<link rel="stylesheet" type="text/css" href="{{ url_for('static', filename='css/style1.css') }}">
|
||||||
|
|
||||||
<script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
|
<script src="https://code.jquery.com/jquery-3.6.0.min.js"></script>
|
||||||
</head>
|
</head>
|
||||||
|
|
||||||
<body>
|
<body>
|
||||||
|
|
||||||
<div id="editForm">
|
<div id="editForm">
|
||||||
<h2>Edit Invoice</h2>
|
<h2>Edit Invoice</h2>
|
||||||
|
|
||||||
<!-- Success Alert Box -->
|
<!-- Success Alert Box -->
|
||||||
<div id="invoiceSuccessAlert" class="success-alert" style="display:none;">
|
<div id="invoiceSuccessAlert" class="success-alert" style="display:none;">
|
||||||
Invoice successfully updated!
|
Invoice successfully updated!
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<form id="invoiceForm" action="{{ url_for('invoice.edit_invoice', invoice_id=invoice.Invoice_Id) }}"
|
||||||
|
method="POST">
|
||||||
|
|
||||||
|
<!-- Subcontractor Field -->
|
||||||
|
<div class="row1">
|
||||||
|
<div>
|
||||||
|
<label for="subcontractor">Subcontractor Name:</label>
|
||||||
|
<input class="form-control" list="subcontractor_list" id="subcontractor" name="subcontractor"
|
||||||
|
value="{{ invoice.Contractor_Name }}" placeholder="Type to search..." required>
|
||||||
|
<input type="hidden" id="subcontractor_id" name="subcontractor_id"
|
||||||
|
value="{{ invoice.Subcontractor_Id }}">
|
||||||
|
<datalist id="subcontractor_list"></datalist>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Village and PMC No Fields -->
|
||||||
|
<div class="row2">
|
||||||
|
<div>
|
||||||
|
<label for="village">Village Name:</label>
|
||||||
|
<input type="text" id="village" name="village" value="{{ invoice.Village_Name }}" required />
|
||||||
|
<datalist id="village_list"></datalist>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="pmc_no">PMC No:</label>
|
||||||
|
<input type="text" id="pmc_no" name="pmc_no" value="{{ invoice.PMC_No }}" required />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Work Type and Invoice Details -->
|
||||||
|
<div class="row2">
|
||||||
|
<div>
|
||||||
|
<label for="work_type">Work Type:</label>
|
||||||
|
<input type="text" id="work_type" name="work_type" value="{{ invoice.Work_Type }}" required />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="invoice_details">Invoice Details:</label>
|
||||||
|
<textarea id="invoice_details" name="invoice_details"
|
||||||
|
required>{{ invoice.Invoice_Details }}</textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Invoice No and Invoice Date -->
|
||||||
|
<div class="row2">
|
||||||
|
<div>
|
||||||
|
<label for="invoice_no">Invoice No:</label>
|
||||||
|
<input type="text" id="invoice_no" name="invoice_no" value="{{ invoice.invoice_no }}" required />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="invoice_date">Invoice Date:</label>
|
||||||
|
<input type="date" id="invoice_date" name="invoice_date" value="{{ invoice.Invoice_Date }}"
|
||||||
|
required />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Amount Fields -->
|
||||||
|
<div class="row3">
|
||||||
|
<div>
|
||||||
|
<label for="basic_amount">Basic Amount:</label>
|
||||||
|
<input type="number" step="0.01" id="basic_amount" name="basic_amount"
|
||||||
|
value="{{ invoice.Basic_Amount }}" required />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="debit_amount">Debit Amount:</label>
|
||||||
|
<input type="number" step="0.01" id="debit_amount" name="debit_amount"
|
||||||
|
value="{{ invoice.Debit_Amount }}" required />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="after_debit_amount">After Debit Amount:</label>
|
||||||
|
<input type="number" step="0.01" id="after_debit_amount" name="after_debit_amount"
|
||||||
|
value="{{ invoice.After_Debit_Amount }}" required />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- GST, TDS, and Other Amounts -->
|
||||||
|
<div class="row3">
|
||||||
|
<div>
|
||||||
|
<label for="amount">Amount:</label>
|
||||||
|
<input type="number" step="0.01" id="amount" name="amount" value="{{ invoice.Amount }}" required />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="gst_amount">GST Amount:</label>
|
||||||
|
<input type="number" step="0.01" id="gst_amount" name="gst_amount" value="{{ invoice.GST_Amount }}"
|
||||||
|
required />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="tds_amount">TDS Amount:</label>
|
||||||
|
<input type="number" step="0.01" id="tds_amount" name="tds_amount" value="{{ invoice.TDS_Amount }}"
|
||||||
|
required />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- SD, On Commission, Hydro Testing -->
|
||||||
|
<div class="row3">
|
||||||
|
<div>
|
||||||
|
<label for="sd_amount">SD Amount:</label>
|
||||||
|
<input type="number" step="0.01" id="sd_amount" name="sd_amount" value="{{ invoice.SD_Amount }}"
|
||||||
|
required />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="on_commission">On Commission:</label>
|
||||||
|
<input type="number" step="0.01" id="on_commission" name="on_commission"
|
||||||
|
value="{{ invoice.On_Commission }}" required />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="hydro_testing">Hydro Testing:</label>
|
||||||
|
<input type="number" step="0.01" id="hydro_testing" name="hydro_testing"
|
||||||
|
value="{{ invoice.Hydro_Testing }}" required />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Hold Amount Section -->
|
||||||
|
<div id="hold_amount_container">
|
||||||
|
{% set seen_types = [] %}
|
||||||
|
{% for hold in invoice.hold_amounts %}
|
||||||
|
{% if hold.hold_type not in seen_types %}
|
||||||
|
{% set _ = seen_types.append(hold.hold_type) %}
|
||||||
|
<div class="hold-amount-row">
|
||||||
|
<label for="hold_type_{{ loop.index }}">Hold Type:</label>
|
||||||
|
<input type="text" id="hold_type_{{ loop.index }}" name="hold_type[]" value="{{ hold.hold_type }}"
|
||||||
|
required />
|
||||||
|
<label for="hold_amount_{{ loop.index }}">Hold Amount:</label>
|
||||||
|
<input type="number" step="0.01" id="hold_amount_{{ loop.index }}" name="hold_amount[]"
|
||||||
|
value="{{ hold.hold_amount }}" required />
|
||||||
|
<button type="button" class="remove-hold-amount">Remove</button>
|
||||||
|
</div>
|
||||||
|
{% endif %}
|
||||||
|
{% endfor %}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<div class="hold-row">
|
||||||
|
<button type="button" id="add_hold_amount" class="button">+ Add Hold Amount</button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Final Amounts -->
|
||||||
|
<div class="row2">
|
||||||
|
<div>
|
||||||
|
<label for="gst_sd_amount">GST SD Amount:</label>
|
||||||
|
<input type="number" step="0.01" id="gst_sd_amount" name="gst_sd_amount"
|
||||||
|
value="{{ invoice.GST_SD_Amount }}" required />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<label for="final_amount">Final Amount:</label>
|
||||||
|
<input type="number" step="0.01" id="final_amount" name="final_amount"
|
||||||
|
value="{{ invoice.Final_Amount }}" required />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Submit Button -->
|
||||||
|
<button type="submit" class="button">Update</button>
|
||||||
|
</form>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<form id="invoiceForm" action="{{ url_for('invoice.edit_invoice', invoice_id=invoice.Invoice_Id) }}" method="POST">
|
<!-- JS for dynamic hold amount rows -->
|
||||||
|
<script>
|
||||||
<!-- Subcontractor Field -->
|
$(document).ready(function () {
|
||||||
<div class="row1">
|
// Add new hold amount row
|
||||||
<div>
|
$("#add_hold_amount").click(function () {
|
||||||
<label for="subcontractor">Subcontractor Name:</label>
|
const index = $("#hold_amount_container .hold-amount-row").length;
|
||||||
<input class="form-control" list="subcontractor_list" id="subcontractor" name="subcontractor"
|
const newRow = `
|
||||||
value="{{ invoice.Contractor_Name }}" placeholder="Type to search..." required>
|
|
||||||
<input type="hidden" id="subcontractor_id" name="subcontractor_id" value="{{ invoice.Subcontractor_Id }}">
|
|
||||||
<datalist id="subcontractor_list"></datalist>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Village and PMC No Fields -->
|
|
||||||
<div class="row2">
|
|
||||||
<div>
|
|
||||||
<label for="village">Village Name:</label>
|
|
||||||
<input type="text" id="village" name="village" value="{{ invoice.Village_Name }}" required/>
|
|
||||||
<datalist id="village_list"></datalist>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="pmc_no">PMC No:</label>
|
|
||||||
<input type="text" id="pmc_no" name="pmc_no" value="{{ invoice.PMC_No }}" required/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Work Type and Invoice Details -->
|
|
||||||
<div class="row2">
|
|
||||||
<div>
|
|
||||||
<label for="work_type">Work Type:</label>
|
|
||||||
<input type="text" id="work_type" name="work_type" value="{{ invoice.Work_Type }}" required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="invoice_details">Invoice Details:</label>
|
|
||||||
<textarea id="invoice_details" name="invoice_details" required>{{ invoice.Invoice_Details }}</textarea>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Invoice No and Invoice Date -->
|
|
||||||
<div class="row2">
|
|
||||||
<div>
|
|
||||||
<label for="invoice_no">Invoice No:</label>
|
|
||||||
<input type="text" id="invoice_no" name="invoice_no" value="{{ invoice.Invoice_No }}" required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="invoice_date">Invoice Date:</label>
|
|
||||||
<input type="date" id="invoice_date" name="invoice_date" value="{{ invoice.Invoice_Date }}" required/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Amount Fields -->
|
|
||||||
<div class="row3">
|
|
||||||
<div>
|
|
||||||
<label for="basic_amount">Basic Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="basic_amount" name="basic_amount" value="{{ invoice.Basic_Amount }}" required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="debit_amount">Debit Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="debit_amount" name="debit_amount" value="{{ invoice.Debit_Amount }}" required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="after_debit_amount">After Debit Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="after_debit_amount" name="after_debit_amount" value="{{ invoice.After_Debit_Amount }}" required/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- GST, TDS, and Other Amounts -->
|
|
||||||
<div class="row3">
|
|
||||||
<div>
|
|
||||||
<label for="amount">Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="amount" name="amount" value="{{ invoice.Amount }}" required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="gst_amount">GST Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="gst_amount" name="gst_amount" value="{{ invoice.GST_Amount }}" required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="tds_amount">TDS Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="tds_amount" name="tds_amount" value="{{ invoice.TDS_Amount }}" required/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- SD, On Commission, Hydro Testing -->
|
|
||||||
<div class="row3">
|
|
||||||
<div>
|
|
||||||
<label for="sd_amount">SD Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="sd_amount" name="sd_amount" value="{{ invoice.SD_Amount }}" required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="on_commission">On Commission:</label>
|
|
||||||
<input type="number" step="0.01" id="on_commission" name="on_commission" value="{{ invoice.On_Commission }}" required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="hydro_testing">Hydro Testing:</label>
|
|
||||||
<input type="number" step="0.01" id="hydro_testing" name="hydro_testing" value="{{ invoice.Hydro_Testing }}" required/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Hold Amount Section -->
|
|
||||||
<div id="hold_amount_container">
|
|
||||||
{% set seen_types = [] %}
|
|
||||||
{% for hold in invoice.hold_amounts %}
|
|
||||||
{% if hold.hold_type not in seen_types %}
|
|
||||||
{% set _ = seen_types.append(hold.hold_type) %}
|
|
||||||
<div class="hold-amount-row">
|
|
||||||
<label for="hold_type_{{ loop.index }}">Hold Type:</label>
|
|
||||||
<input type="text" id="hold_type_{{ loop.index }}" name="hold_type[]" value="{{ hold.hold_type }}" required/>
|
|
||||||
<label for="hold_amount_{{ loop.index }}">Hold Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="hold_amount_{{ loop.index }}" name="hold_amount[]" value="{{ hold.hold_amount }}" required/>
|
|
||||||
<button type="button" class="remove-hold-amount">Remove</button>
|
|
||||||
</div>
|
|
||||||
{% endif %}
|
|
||||||
{% endfor %}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
|
|
||||||
<div class="hold-row">
|
|
||||||
<button type="button" id="add_hold_amount" class="button">+ Add Hold Amount</button>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Final Amounts -->
|
|
||||||
<div class="row2">
|
|
||||||
<div>
|
|
||||||
<label for="gst_sd_amount">GST SD Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="gst_sd_amount" name="gst_sd_amount" value="{{ invoice.GST_SD_Amount }}" required/>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<label for="final_amount">Final Amount:</label>
|
|
||||||
<input type="number" step="0.01" id="final_amount" name="final_amount" value="{{ invoice.Final_Amount }}" required/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- Submit Button -->
|
|
||||||
<button type="submit" class="button">Update</button>
|
|
||||||
</form>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- JS for dynamic hold amount rows -->
|
|
||||||
<script>
|
|
||||||
$(document).ready(function() {
|
|
||||||
// Add new hold amount row
|
|
||||||
$("#add_hold_amount").click(function() {
|
|
||||||
const index = $("#hold_amount_container .hold-amount-row").length;
|
|
||||||
const newRow = `
|
|
||||||
<div class="hold-amount-row">
|
<div class="hold-amount-row">
|
||||||
<label for="hold_type_${index}">Hold Type:</label>
|
<label for="hold_type_${index}">Hold Type:</label>
|
||||||
<input type="text" id="hold_type_${index}" name="hold_type[]" required/>
|
<input type="text" id="hold_type_${index}" name="hold_type[]" required/>
|
||||||
@@ -172,36 +189,38 @@ $(document).ready(function() {
|
|||||||
<button type="button" class="remove-hold-amount">Remove</button>
|
<button type="button" class="remove-hold-amount">Remove</button>
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
$("#hold_amount_container").append(newRow);
|
$("#hold_amount_container").append(newRow);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Remove hold amount row
|
// Remove hold amount row
|
||||||
$(document).on("click", ".remove-hold-amount", function() {
|
$(document).on("click", ".remove-hold-amount", function () {
|
||||||
$(this).closest(".hold-amount-row").remove();
|
$(this).closest(".hold-amount-row").remove();
|
||||||
});
|
});
|
||||||
|
|
||||||
// Submit form via AJAX
|
// Submit form via AJAX
|
||||||
$("#invoiceForm").submit(function(e) {
|
$("#invoiceForm").submit(function (e) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
|
||||||
$.ajax({
|
$.ajax({
|
||||||
type: "POST",
|
type: "POST",
|
||||||
url: $(this).attr("action"),
|
url: $(this).attr("action"),
|
||||||
data: $(this).serialize(),
|
data: $(this).serialize(),
|
||||||
success: function(response) {
|
dataType: 'json', // ensure JSON is returned
|
||||||
if(response.status === "success") {
|
success: function (response) {
|
||||||
$("#invoiceSuccessAlert").fadeIn().delay(3000).fadeOut();
|
if (response.status === "success") {
|
||||||
alert("Invoice updated successfully!"); // <-- Popup alert
|
alert("Invoice updated successfully!"); // <-- Popup alert
|
||||||
}
|
|
||||||
|
|
||||||
},
|
// ✅ Redirect to Add Invoice page (table part visible)
|
||||||
error: function(xhr) {
|
window.location.href = "{{ url_for('invoice.add_invoice') }}#addTable";
|
||||||
alert("Error: " + xhr.responseJSON.message);
|
}
|
||||||
}
|
},
|
||||||
|
error: function (xhr) {
|
||||||
|
alert("Error: " + xhr.responseJSON?.message || "Something went wrong!");
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
</script>
|
||||||
});
|
|
||||||
</script>
|
|
||||||
|
|
||||||
</body>
|
</body>
|
||||||
{% endblock %}
|
{% endblock %}
|
||||||
@@ -28,25 +28,17 @@
|
|||||||
|
|
||||||
<button type="submit">Update Village</button>
|
<button type="submit">Update Village</button>
|
||||||
</form>
|
</form>
|
||||||
</div>
|
<!-- Flash Messages (hidden, used for JS popup) -->
|
||||||
|
<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 id="flash-message" data-category="{{ category }}" style="display:none;">{{ message }}</div>
|
<div class="flash-message" data-category="{{ category }}">{{ 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 %}
|
||||||
|
|||||||
Reference in New Issue
Block a user